@nostr-dev-kit/ndk
Version:
NDK - Nostr Development Kit. Includes AI Guardrails to catch common mistakes during development.
285 lines (229 loc) • 14.1 kB
text/typescript
import { describe, expect, it } from "vitest";
import { generateContentTags } from "./content-tagger";
import type { ContentTaggingOptions, NDKEvent, NDKTag } from "./index.js";
describe("await generateContentTags", () => {
it("doesnt retag events that were quoted", async () => {
const content = "testing a quoted event nostr:note1d2mheza0d5z5yycucu94nw37e6gyl4vwl6gavyku86rshkhexq6q30s0g7";
const tags: NDKTag[] = [];
tags.push(["q", "6ab77c8baf6d0542131cc70b59ba3ece904fd58efe91d612dc3e870bdaf93034"]);
const { content: processedContent, tags: processedTags } = await generateContentTags(content, tags);
expect(processedContent).toEqual(content);
expect(processedTags).toEqual([["q", "6ab77c8baf6d0542131cc70b59ba3ece904fd58efe91d612dc3e870bdaf93034"]]);
});
it("replaces valid tags and store decoded data in the tags array", async () => {
const content =
"This is a sample content with @npub1l2vyh47mk2p0qlsku7hg0vn29faehy9hy34ygaclpn66ukqp3afqutajft and @note1d2mheza0d5z5yycucu94nw37e6gyl4vwl6gavyku86rshkhexq6q30s0g7 tags.";
const tags: NDKTag[] = [];
const { content: processedContent, tags: processedTags } = await generateContentTags(content, tags);
expect(processedContent).toEqual(
"This is a sample content with nostr:npub1l2vyh47mk2p0qlsku7hg0vn29faehy9hy34ygaclpn66ukqp3afqutajft and nostr:note1d2mheza0d5z5yycucu94nw37e6gyl4vwl6gavyku86rshkhexq6q30s0g7 tags.",
);
expect(processedTags.length).toEqual(2);
expect(processedTags).toEqual([
["p", "fa984bd7dbb282f07e16e7ae87b26a2a7b9b90b7246a44771f0cf5ae58018f52"],
["q", "6ab77c8baf6d0542131cc70b59ba3ece904fd58efe91d612dc3e870bdaf93034", ""],
]);
});
it("does not replace invalid tags and leave the tags array unchanged", async () => {
const content = "This is a sample content with an @invalidTag.";
const tags: NDKTag[] = [];
const { content: processedContent, tags: processedTags } = await generateContentTags(content, tags);
expect(processedContent).toEqual(content);
expect(processedTags.length).toEqual(0);
});
it("handles existing tags and update indexes accordingly", async () => {
const content =
"This is a sample content with @npub1l2vyh47mk2p0qlsku7hg0vn29faehy9hy34ygaclpn66ukqp3afqutajft and @note1d2mheza0d5z5yycucu94nw37e6gyl4vwl6gavyku86rshkhexq6q30s0g7 tags.";
const tags: NDKTag[] = [["p", "existing_p"]];
const { content: processedContent, tags: processedTags } = await generateContentTags(content, tags);
expect(processedContent).toEqual(
"This is a sample content with nostr:npub1l2vyh47mk2p0qlsku7hg0vn29faehy9hy34ygaclpn66ukqp3afqutajft and nostr:note1d2mheza0d5z5yycucu94nw37e6gyl4vwl6gavyku86rshkhexq6q30s0g7 tags.",
);
expect(processedTags.length).toEqual(3);
expect(processedTags).toEqual([
["p", "existing_p"],
["p", "fa984bd7dbb282f07e16e7ae87b26a2a7b9b90b7246a44771f0cf5ae58018f52"],
["q", "6ab77c8baf6d0542131cc70b59ba3ece904fd58efe91d612dc3e870bdaf93034", ""],
]);
});
it("does not replace an event without a nostr: or @ prefix", async () => {
const content =
"note13jgf85r2yxjmww8f6gweque5g388agfztrppmgenur7zvh8e929s4cptur " +
"@note13jgf85r2yxjmww8f6gweque5g388agfztrppmgenur7zvh8e929s4cptur";
const { content: processedContent, tags: processedTags } = await generateContentTags(content, []);
expect(processedContent).toEqual(
"note13jgf85r2yxjmww8f6gweque5g388agfztrppmgenur7zvh8e929s4cptur " +
"nostr:note13jgf85r2yxjmww8f6gweque5g388agfztrppmgenur7zvh8e929s4cptur",
);
expect(processedTags).toEqual([["q", "8c9093d06a21a5b738e9d21d907334444e7ea12258c21da333e0fc265cf92a8b", ""]]);
});
it("uses a mention marker", async () => {
const content = "Hello world, nostr:note13jgf85r2yxjmww8f6gweque5g388agfztrppmgenur7zvh8e929s4cptur";
const { content: processedContent, tags: processedTags } = await generateContentTags(content, []);
expect(processedContent).toEqual(
"Hello world, nostr:note13jgf85r2yxjmww8f6gweque5g388agfztrppmgenur7zvh8e929s4cptur",
);
expect(processedTags).toEqual([["q", "8c9093d06a21a5b738e9d21d907334444e7ea12258c21da333e0fc265cf92a8b", ""]]);
});
it("handles replaceable event mentions", async () => {
const content =
"Hello world, nostr:naddr1qqrrywpn8y6rvq3qvwymuey3u7mf860ndrkw3r7dz30s0srg6tqmhtjzg7umtm6rn5eqxpqqqp65wnen4nu";
const { content: processedContent, tags: processedTags } = await generateContentTags(content, []);
expect(processedContent).toEqual(
"Hello world, nostr:naddr1qqrrywpn8y6rvq3qvwymuey3u7mf860ndrkw3r7dz30s0srg6tqmhtjzg7umtm6rn5eqxpqqqp65wnen4nu",
);
expect(processedTags).toEqual([
["q", "30023:6389be6491e7b693e9f368ece88fcd145f07c068d2c1bbae4247b9b5ef439d32:283946", ""],
["p", "6389be6491e7b693e9f368ece88fcd145f07c068d2c1bbae4247b9b5ef439d32"],
]);
});
it("uses the relay marker on nevents", async () => {
const content =
"Hello @nevent1qqsykmtd4zplclt56wfkkzjwplyw6q8a2p0fqf8p46wedlqayct0uhqppemhxue69uhkummn9ekx7mp0qgstryqhtnh5qlze8utuz42gpmdp5th87myyx78w7r5hx5lc0mjexqqrqsqqqqqpyazpkc";
const { content: processedContent, tags: processedTags } = await generateContentTags(content, []);
expect(processedContent).toEqual(
"Hello nostr:nevent1qqsykmtd4zplclt56wfkkzjwplyw6q8a2p0fqf8p46wedlqayct0uhqppemhxue69uhkummn9ekx7mp0qgstryqhtnh5qlze8utuz42gpmdp5th87myyx78w7r5hx5lc0mjexqqrqsqqqqqpyazpkc",
);
expect(processedTags).toEqual([
["q", "4b6d6da883fc7d74d3936b0a4e0fc8ed00fd505e9024e1ae9d96fc1d2616fe5c", "wss://nos.lol/"],
["p", "b190175cef407c593f17c155480eda1a2ee7f6c84378eef0e97353f87ee59300"],
]);
});
it("uses the relay marker on naddr", async () => {
const content =
"Hello @naddr1qqrrywpn8y6rvqgdwaehxw309ahx7uewd3hkcq3qvwymuey3u7mf860ndrkw3r7dz30s0srg6tqmhtjzg7umtm6rn5eqxpqqqp65wj55zc8";
const { content: processedContent, tags: processedTags } = await generateContentTags(content, []);
expect(processedContent).toEqual(
"Hello nostr:naddr1qqrrywpn8y6rvqgdwaehxw309ahx7uewd3hkcq3qvwymuey3u7mf860ndrkw3r7dz30s0srg6tqmhtjzg7umtm6rn5eqxpqqqp65wj55zc8",
);
expect(processedTags).toEqual([
["q", "30023:6389be6491e7b693e9f368ece88fcd145f07c068d2c1bbae4247b9b5ef439d32:283946", "wss://nos.lol"],
["p", "6389be6491e7b693e9f368ece88fcd145f07c068d2c1bbae4247b9b5ef439d32"],
]);
});
it("finds and store #tags in the tags array", async () => {
const content = "This is a sample content with #tag1 and #tag2.";
const tags: NDKTag[] = [];
const { content: processedContent, tags: processedTags } = await generateContentTags(content, tags);
expect(processedContent).toEqual("This is a sample content with #tag1 and #tag2.");
expect(processedTags.length).toEqual(2);
expect(processedTags).toEqual([
["t", "tag1"],
["t", "tag2"],
]);
});
it("handles mixed types of tags", async () => {
const content =
"This is a sample content with #tag1 and @npub1l2vyh47mk2p0qlsku7hg0vn29faehy9hy34ygaclpn66ukqp3afqutajft.";
const tags: NDKTag[] = [];
const { content: processedContent, tags: processedTags } = await generateContentTags(content, tags);
expect(processedContent).toEqual(
"This is a sample content with #tag1 and nostr:npub1l2vyh47mk2p0qlsku7hg0vn29faehy9hy34ygaclpn66ukqp3afqutajft.",
);
expect(processedTags.length).toEqual(2);
expect(processedTags).toEqual([
["p", "fa984bd7dbb282f07e16e7ae87b26a2a7b9b90b7246a44771f0cf5ae58018f52"],
["t", "tag1"],
]);
});
describe("ContentTaggingOptions", () => {
it("skips content tagging when skipContentTagging is true", async () => {
const content = "Hello @npub1l2vyh47mk2p0qlsku7hg0vn29faehy9hy34ygaclpn66ukqp3afqutajft #tag1";
const opts: ContentTaggingOptions = { skipContentTagging: true };
const { content: processedContent, tags: processedTags } = await generateContentTags(content, [], opts);
expect(processedContent).toEqual(content);
expect(processedTags).toEqual([]);
});
it("always uses q-tag for note references", async () => {
const content = "Check this out: nostr:note1d2mheza0d5z5yycucu94nw37e6gyl4vwl6gavyku86rshkhexq6q30s0g7";
const { tags: processedTags } = await generateContentTags(content, []);
expect(processedTags).toEqual([
["q", "6ab77c8baf6d0542131cc70b59ba3ece904fd58efe91d612dc3e870bdaf93034", ""],
]);
});
it("skips p-tags on q-tags when pTagOnQTags is false", async () => {
const content =
"Hello @nevent1qqsykmtd4zplclt56wfkkzjwplyw6q8a2p0fqf8p46wedlqayct0uhqppemhxue69uhkummn9ekx7mp0qgstryqhtnh5qlze8utuz42gpmdp5th87myyx78w7r5hx5lc0mjexqqrqsqqqqqpyazpkc";
const opts: ContentTaggingOptions = { pTagOnQTags: false };
const { tags: processedTags } = await generateContentTags(content, [], opts);
expect(processedTags).toEqual([
["q", "4b6d6da883fc7d74d3936b0a4e0fc8ed00fd505e9024e1ae9d96fc1d2616fe5c", "wss://nos.lol/"],
]);
});
it("includes p-tags on q-tags by default", async () => {
const content =
"Hello @nevent1qqsykmtd4zplclt56wfkkzjwplyw6q8a2p0fqf8p46wedlqayct0uhqppemhxue69uhkummn9ekx7mp0qgstryqhtnh5qlze8utuz42gpmdp5th87myyx78w7r5hx5lc0mjexqqrqsqqqqqpyazpkc";
const { tags: processedTags } = await generateContentTags(content, []);
expect(processedTags).toEqual([
["q", "4b6d6da883fc7d74d3936b0a4e0fc8ed00fd505e9024e1ae9d96fc1d2616fe5c", "wss://nos.lol/"],
["p", "b190175cef407c593f17c155480eda1a2ee7f6c84378eef0e97353f87ee59300"],
]);
});
it("do not copy p-tags from target event when copyPTagsFromTarget is true but pubkeys are invalid", async () => {
const content = "Reply to event";
const opts: ContentTaggingOptions = { copyPTagsFromTarget: true };
const mockEvent = {
getMatchingTags: (tag: string) => {
if (tag === "p") {
return [
["p", "pubkey1"],
["p", "pubkey2"],
];
}
return [];
},
} as unknown as NDKEvent;
const { tags: processedTags } = await generateContentTags(content, [], opts, mockEvent);
expect(processedTags).toEqual([]);
});
it("copies p-tags from target event when copyPTagsFromTarget is true and pubkeys are valid", async () => {
const content = "Reply to event";
const opts: ContentTaggingOptions = { copyPTagsFromTarget: true };
const mockEvent = {
getMatchingTags: (tag: string) => {
if (tag === "p") {
return [
["p", "fa984bd7dbb282f07e16e7ae87b26a2a7b9b90b7246a44771f0cf5ae58018f52"],
["p", "b190175cef407c593f17c155480eda1a2ee7f6c84378eef0e97353f87ee59300"],
];
}
return [];
},
} as unknown as NDKEvent;
const { tags: processedTags } = await generateContentTags(content, [], opts, mockEvent);
expect(processedTags).toEqual([
["p", "fa984bd7dbb282f07e16e7ae87b26a2a7b9b90b7246a44771f0cf5ae58018f52"],
["p", "b190175cef407c593f17c155480eda1a2ee7f6c84378eef0e97353f87ee59300"],
]);
});
it("excludes hashtag processing when excluded in filters", async () => {
const content = "Hello #bitcoin #nostr";
const opts: ContentTaggingOptions = {
filters: { excludeTypes: ["hashtag"] },
};
const { tags: processedTags } = await generateContentTags(content, [], opts);
expect(processedTags).toEqual([]);
});
it("only includes specified types when includeTypes is set", async () => {
const content =
"Hello @npub1l2vyh47mk2p0qlsku7hg0vn29faehy9hy34ygaclpn66ukqp3afqutajft and #bitcoin and nostr:note1d2mheza0d5z5yycucu94nw37e6gyl4vwl6gavyku86rshkhexq6q30s0g7";
const opts: ContentTaggingOptions = {
filters: { includeTypes: ["npub"] },
};
const { tags: processedTags } = await generateContentTags(content, [], opts);
expect(processedTags).toEqual([
["p", "fa984bd7dbb282f07e16e7ae87b26a2a7b9b90b7246a44771f0cf5ae58018f52"],
["t", "bitcoin"],
]);
});
it("excludes specified types when excludeTypes is set", async () => {
const content =
"Hello @npub1l2vyh47mk2p0qlsku7hg0vn29faehy9hy34ygaclpn66ukqp3afqutajft and nostr:note1d2mheza0d5z5yycucu94nw37e6gyl4vwl6gavyku86rshkhexq6q30s0g7";
const opts: ContentTaggingOptions = {
filters: { excludeTypes: ["note"] },
};
const { tags: processedTags } = await generateContentTags(content, [], opts);
expect(processedTags).toEqual([["p", "fa984bd7dbb282f07e16e7ae87b26a2a7b9b90b7246a44771f0cf5ae58018f52"]]);
});
});
});