@fedify/botkit 0.3.0-dev.108

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (94) hide show
  1. package/LICENSE +661 -0
  2. package/README.md +75 -0
  3. package/dist/bot-impl.d.ts +111 -0
  4. package/dist/bot-impl.d.ts.map +1 -0
  5. package/dist/bot-impl.js +602 -0
  6. package/dist/bot-impl.js.map +1 -0
  7. package/dist/bot-impl.test.d.ts +2 -0
  8. package/dist/bot-impl.test.js +1642 -0
  9. package/dist/bot-impl.test.js.map +1 -0
  10. package/dist/bot.d.ts +270 -0
  11. package/dist/bot.d.ts.map +1 -0
  12. package/dist/bot.js +118 -0
  13. package/dist/bot.js.map +1 -0
  14. package/dist/bot.test.d.ts +2 -0
  15. package/dist/bot.test.js +80 -0
  16. package/dist/bot.test.js.map +1 -0
  17. package/dist/components/Layout.d.ts +26 -0
  18. package/dist/components/Layout.d.ts.map +1 -0
  19. package/dist/components/Layout.js +36 -0
  20. package/dist/components/Layout.js.map +1 -0
  21. package/dist/components/Message.d.ts +21 -0
  22. package/dist/components/Message.d.ts.map +1 -0
  23. package/dist/components/Message.js +99 -0
  24. package/dist/components/Message.js.map +1 -0
  25. package/dist/components/Message.test.d.ts +2 -0
  26. package/dist/components/Message.test.js +32 -0
  27. package/dist/components/Message.test.js.map +1 -0
  28. package/dist/deno.js +73 -0
  29. package/dist/deno.js.map +1 -0
  30. package/dist/emoji.d.ts +87 -0
  31. package/dist/emoji.d.ts.map +1 -0
  32. package/dist/emoji.js +37 -0
  33. package/dist/emoji.js.map +1 -0
  34. package/dist/emoji.test.d.ts +2 -0
  35. package/dist/emoji.test.js +109 -0
  36. package/dist/emoji.test.js.map +1 -0
  37. package/dist/events.d.ts +110 -0
  38. package/dist/events.d.ts.map +1 -0
  39. package/dist/events.js +4 -0
  40. package/dist/follow-impl.d.ts +23 -0
  41. package/dist/follow-impl.d.ts.map +1 -0
  42. package/dist/follow-impl.js +51 -0
  43. package/dist/follow-impl.js.map +1 -0
  44. package/dist/follow-impl.test.d.ts +2 -0
  45. package/dist/follow-impl.test.js +110 -0
  46. package/dist/follow-impl.test.js.map +1 -0
  47. package/dist/follow.d.ts +44 -0
  48. package/dist/follow.d.ts.map +1 -0
  49. package/dist/follow.js +4 -0
  50. package/dist/message-impl.d.ts +54 -0
  51. package/dist/message-impl.d.ts.map +1 -0
  52. package/dist/message-impl.js +439 -0
  53. package/dist/message-impl.js.map +1 -0
  54. package/dist/message-impl.test.d.ts +2 -0
  55. package/dist/message-impl.test.js +519 -0
  56. package/dist/message-impl.test.js.map +1 -0
  57. package/dist/message.d.ts +223 -0
  58. package/dist/message.d.ts.map +1 -0
  59. package/dist/message.js +8 -0
  60. package/dist/mod.d.ts +13 -0
  61. package/dist/mod.js +13 -0
  62. package/dist/pages.d.ts +20 -0
  63. package/dist/pages.d.ts.map +1 -0
  64. package/dist/pages.js +361 -0
  65. package/dist/pages.js.map +1 -0
  66. package/dist/reaction.d.ts +90 -0
  67. package/dist/reaction.d.ts.map +1 -0
  68. package/dist/reaction.js +7 -0
  69. package/dist/repository.d.ts +323 -0
  70. package/dist/repository.d.ts.map +1 -0
  71. package/dist/repository.js +483 -0
  72. package/dist/repository.js.map +1 -0
  73. package/dist/repository.test.d.ts +2 -0
  74. package/dist/repository.test.js +336 -0
  75. package/dist/repository.test.js.map +1 -0
  76. package/dist/session-impl.d.ts +32 -0
  77. package/dist/session-impl.d.ts.map +1 -0
  78. package/dist/session-impl.js +195 -0
  79. package/dist/session-impl.js.map +1 -0
  80. package/dist/session-impl.test.d.ts +20 -0
  81. package/dist/session-impl.test.d.ts.map +1 -0
  82. package/dist/session-impl.test.js +464 -0
  83. package/dist/session-impl.test.js.map +1 -0
  84. package/dist/session.d.ts +139 -0
  85. package/dist/session.d.ts.map +1 -0
  86. package/dist/session.js +4 -0
  87. package/dist/text.d.ts +391 -0
  88. package/dist/text.d.ts.map +1 -0
  89. package/dist/text.js +640 -0
  90. package/dist/text.js.map +1 -0
  91. package/dist/text.test.d.ts +2 -0
  92. package/dist/text.test.js +473 -0
  93. package/dist/text.test.js.map +1 -0
  94. package/package.json +137 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"text.js","names":["value: unknown","session: Session<TContextData>","text: Text<\"block\" | \"inline\", TContextData>","actor: Actor | URL","Mention","strings: TemplateStringsArray","#strings","#values","Emoji","paraState: \"opened\" | \"closed\"","objects: Object[]","text: string","text","_session: Session<TContextData>","label: string | ((session: Session<TContextData>) => Promise<string>)","actor: Actor | ((session: Session<TContextData>) => Promise<Object | null>)","#label","#actor","#cachedObject","#labelPromise","#actorPromise","#getLabel","#getActor","Link","a: string | Actor | URL","b?: Actor | URL","tag: string","#tag","text: Text<\"inline\", TContextData> | string","#text","label: Text<\"inline\", TContextData> | string","href: URL | string","#href","label: Text<\"inline\", TContextData> | string | URL","href?: URL | string","#code","code: Text<\"inline\", TContextData> | string","code","#emoji","emoji: Emoji | DeferredCustomEmoji<TContextData>","#content","#markdownIt","#mentions","#hashtags","content: string","options: MarkdownTextOptions","mentionPlugin","handle: string","env: MarkdownEnv","env","_handle: string","_env: MarkdownEnv","hashtagPlugin","hashtag: string","_hashtag: string","#getMentionedActors","#actors","actors: Record<string, Object>","actors: Record<string, string | null>","hashtag"],"sources":["../src/text.ts"],"sourcesContent":["// BotKit by Fedify: A framework for creating ActivityPub bots\n// Copyright (C) 2025 Hong Minhee <https://hongminhee.org/>\n//\n// This program is free software: you can redistribute it and/or modify\n// it under the terms of the GNU Affero General Public License as\n// published by the Free Software Foundation, either version 3 of the\n// License, or (at your option) any later version.\n//\n// This program is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n// GNU Affero General Public License for more details.\n//\n// You should have received a copy of the GNU Affero General Public License\n// along with this program. If not, see <https://www.gnu.org/licenses/>.\nimport {\n type Actor,\n Emoji,\n getActorHandle,\n isActor,\n Link,\n Mention,\n type Object,\n} from \"@fedify/fedify\";\nimport { Hashtag } from \"@fedify/fedify/vocab\";\nimport { hashtag as hashtagPlugin } from \"@fedify/markdown-it-hashtag\";\nimport {\n mention as mentionPlugin,\n toFullHandle,\n} from \"@fedify/markdown-it-mention\";\nimport { encode } from \"html-entities\";\nimport MarkdownIt from \"markdown-it\";\nimport type { DeferredCustomEmoji } from \"./emoji.ts\";\nimport type { Session } from \"./session.ts\";\n\n/**\n * A tree structure representing a text with formatting. It does not only\n * render the text but also extract tags (e.g., mentions) from it.\n * @typeParam TContextData The type of the context data.\n */\nexport interface Text<TType extends \"block\" | \"inline\", TContextData> {\n /**\n * The type of the text. It can be either `\"block\"` or `\"inline\"`.\n */\n readonly type: TType;\n\n /**\n * Render a text tree as HTML.\n * @param session The bot session.\n * @returns An async iterable of HTML chunks.\n */\n getHtml(session: Session<TContextData>): AsyncIterable<string>;\n\n /**\n * Extract tags (e.g., mentions) from a text tree.\n * @param session The bot session\n * @returns An async iterable of tags.\n */\n getTags(session: Session<TContextData>): AsyncIterable<Link | Object>;\n\n /**\n * Gets cached objects. The result of this method depends on\n * whether {@link getHtml} or {@link getTags} has been called before.\n * It's used for optimizing the post rendering process, e.g., reusing\n * once fetched remote objects.\n * @returns The cached objects. The order of the objects does not matter.\n */\n getCachedObjects(): Object[];\n}\n\n/**\n * Checks if a value is a {@link Text} tree.\n * @param value The value to check.\n * @returns `true` if the value is a {@link Text} tree, `false` otherwise.\n * @typeParam TContextData The type of the context data.\n */\nexport function isText<TContextData>(\n value: unknown,\n): value is Text<\"block\" | \"inline\", TContextData> {\n return typeof value === \"object\" && value !== null && \"getHtml\" in value &&\n \"getTags\" in value && typeof value.getHtml === \"function\" &&\n typeof value.getTags === \"function\" && \"type\" in value &&\n (value.type === \"block\" || value.type === \"inline\");\n}\n\n/**\n * Checks if a given `actor` is mentioned in a `text`.\n * @param session The bot session.\n * @param text The text object to check.\n * @param actor The actor to check. It can be either an `Actor` object or\n * an actor URI.\n * @returns `true` if the actor is mentioned in the text, `false` otherwise.\n */\nexport async function mentions<TContextData>(\n session: Session<TContextData>,\n text: Text<\"block\" | \"inline\", TContextData>,\n actor: Actor | URL,\n): Promise<boolean> {\n if (isActor(actor)) {\n if (actor.id == null) return false;\n actor = actor.id;\n }\n for await (const tag of text.getTags(session)) {\n if (tag instanceof Mention && tag.href?.href === actor.href) return true;\n }\n return false;\n}\n\n/**\n * A text tree that renders a template string with values. You normally\n * don't need to instantiate this directly; use the {@link text} function\n * instead.\n * @typeParam TContextData The type of the context data.\n */\nexport class TemplatedText<TContextData>\n implements Text<\"block\", TContextData> {\n readonly type = \"block\";\n #strings: TemplateStringsArray;\n #values: Text<\"block\" | \"inline\", TContextData>[];\n\n /**\n * Creates a text tree with a template string and values.\n * @param strings The template strings.\n * @param values The values to interpolate.\n */\n constructor(strings: TemplateStringsArray, ...values: unknown[]) {\n this.#strings = strings;\n this.#values = values.map((v) => {\n if (isText<TContextData>(v)) return v;\n if (v instanceof URL) return link(v);\n if (isActor(v)) return mention(v);\n if (v instanceof Emoji) return customEmoji(v);\n return new PlainText(String(v));\n });\n }\n\n async *getHtml(session: Session<TContextData>): AsyncIterable<string> {\n let paraState: \"opened\" | \"closed\" = \"closed\";\n for (let i = 0; i < this.#strings.length; i++) {\n const paragraphs = this.#strings[i].split(/([ \\t]*\\r?\\n){2,}/g);\n let p = 0;\n for (const para of paragraphs) {\n if (p > 0 && paraState === \"opened\") {\n yield \"</p>\";\n paraState = \"closed\";\n }\n const lines = para.split(\"\\n\");\n let l = 0;\n for (const line of lines) {\n if (line.trim() === \"\") continue;\n if (l < 1 && paraState === \"closed\") {\n yield \"<p>\";\n paraState = \"opened\";\n }\n if (l > 0) yield \"<br>\";\n yield encode(line);\n l++;\n }\n p++;\n }\n if (i < this.#values.length) {\n const value = this.#values[i];\n if (value.type === \"block\" && paraState === \"opened\") {\n yield \"</p>\";\n paraState = \"closed\";\n } else if (value.type === \"inline\" && paraState === \"closed\") {\n yield \"<p>\";\n paraState = \"opened\";\n }\n yield* value.getHtml(session);\n }\n }\n if (paraState === \"opened\") yield \"</p>\";\n }\n\n async *getTags(session: Session<TContextData>): AsyncIterable<Link | Object> {\n for (const value of this.#values) {\n if (!isText<TContextData>(value)) continue;\n yield* value.getTags(session);\n }\n }\n\n getCachedObjects(): Object[] {\n const objects: Object[] = [];\n for (const value of this.#values) {\n if (!isText<TContextData>(value)) continue;\n objects.push(...value.getCachedObjects());\n }\n return objects;\n }\n}\n\n/**\n * A template string tag that creates a {@link Text} tree.\n *\n * Basically, it only interpolates values into the template string and\n * escapes HTML characters, except for line breaks and paragraphs.\n * For example, the below code:\n *\n * ```ts\n * text`Hello, <${em(\"World\")}>!\\n\\nGoodbye!`\n * ```\n *\n * will be rendered as:\n *\n * ```html\n * <p>Hello, &lt;<em>World</em>&gt;!</p>\n * <p>Goodbye!</p>\n * ```\n *\n * @typeParam TContextData The type of the context data.\n * @param strings The template strings.\n * @param values The values to interpolate.\n * @returns A {@link Text} tree.\n */\nexport function text<TContextData>(\n strings: TemplateStringsArray,\n ...values: unknown[]\n): Text<\"block\", TContextData> {\n return new TemplatedText<TContextData>(strings, ...values);\n}\n\n/**\n * A text tree that renders a plain text. You normally don't need to\n * instantiate this directly; use the {@link plainText} function instead.\n * @typeParam TContextData The type of the context data.\n */\nexport class PlainText<TContextData> implements Text<\"inline\", TContextData> {\n readonly type = \"inline\";\n readonly text: string;\n\n /**\n * Creates a {@link PlainText} tree with a plain text.\n * @param text The plain text.\n */\n constructor(text: string) {\n this.text = text;\n }\n\n async *getHtml(_session: Session<TContextData>): AsyncIterable<string> {\n let first = true;\n for (const line of this.text.split(\"\\n\")) {\n if (!first) yield \"<br>\";\n yield encode(line);\n first = false;\n }\n }\n\n async *getTags(\n _session: Session<TContextData>,\n ): AsyncIterable<Link | Object> {\n }\n\n getCachedObjects(): Object[] {\n return [];\n }\n}\n\n/**\n * A function that creates a {@link PlainText} tree. It only does two simple\n * things:\n *\n * - Escaping the given text so that it can be safely rendered as HTML\n * - Splitting the text by line breaks and rendering them as hard line breaks\n * @typeParam TContextData The type of the context data.\n * @param text The plain text.\n * @returns A {@link PlainText} tree.\n */\nexport function plainText<TContextData>(\n text: string,\n): Text<\"inline\", TContextData> {\n return new PlainText(text);\n}\n\n/**\n * A text tree that renders a mention. You normally don't need to\n * instantiate this directly; use the {@link mention} function instead.\n * @typeParam TContextData The type of the context data.\n */\nexport class MentionText<TContextData> implements Text<\"inline\", TContextData> {\n readonly type = \"inline\";\n #label: string | ((session: Session<TContextData>) => Promise<string>);\n #actor: Actor | ((session: Session<TContextData>) => Promise<Object | null>);\n #cachedObject?: Object;\n #labelPromise?: Promise<string>;\n #actorPromise?: Promise<Object | null>;\n\n /**\n * Creates a {@link MentionText} tree with a label and an actor.\n * @param label The label of the mention.\n * @param actor The actor which the mention refers to.\n */\n constructor(\n label: string | ((session: Session<TContextData>) => Promise<string>),\n actor: Actor | ((session: Session<TContextData>) => Promise<Object | null>),\n ) {\n this.#label = label;\n this.#actor = actor;\n if (isActor(actor)) this.#cachedObject = actor;\n }\n\n #getLabel(session: Session<TContextData>): Promise<string> {\n if (typeof this.#label === \"string\") return Promise.resolve(this.#label);\n if (this.#labelPromise != null) return this.#labelPromise;\n return this.#labelPromise = this.#label(session);\n }\n\n #getActor(session: Session<TContextData>): Promise<Object | null> {\n if (isActor(this.#actor)) return Promise.resolve(this.#actor);\n if (this.#actorPromise != null) return this.#actorPromise;\n return this.#actorPromise = this.#actor(session).then((actor) => {\n if (actor != null) this.#cachedObject = actor;\n return actor;\n });\n }\n\n async *getHtml(session: Session<TContextData>): AsyncIterable<string> {\n const label = await this.#getLabel(session);\n const actor = await this.#getActor(session);\n const url = !isActor(actor)\n ? null\n : actor.url == null\n ? actor.id\n : actor.url instanceof Link\n ? actor.url.href\n : actor.url;\n if (url == null) {\n yield encode(label);\n return;\n }\n yield '<a href=\"';\n yield encode(url.href);\n yield '\" translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\">';\n if (label.startsWith(\"@\")) {\n yield \"@<span>\";\n yield encode(label.substring(1));\n yield \"</span>\";\n } else {\n yield encode(label);\n }\n yield \"</a>\";\n }\n\n async *getTags(session: Session<TContextData>): AsyncIterable<Link | Object> {\n const label = await this.#getLabel(session);\n const actor = await this.#getActor(session);\n if (isActor(actor)) {\n yield new Mention({\n name: label,\n href: actor.id,\n });\n }\n }\n\n getCachedObjects(): Object[] {\n return this.#cachedObject == null ? [] : [this.#cachedObject];\n }\n}\n\n/**\n * Mentions an actor by its fediverse handle. You can use this function\n * to create a {@link MentionText} tree. The label of the mention will be\n * the same as the handle.\n *\n * If the given handle does not refer to an actor, the returned tree consists\n * of a plain text with the handle without any link.\n * @typeParam TContextData The type of the context data.\n * @param handle The handle of the actor.\n * @returns A {@link MentionText} tree.\n */\nexport function mention<TContextData>(\n handle: string,\n): Text<\"inline\", TContextData>;\n\n/**\n * Mentions an actor. You can use this function to create a {@link MentionText}\n * from an actor object. The label of the mention will be the fediverse handle\n * of the actor.\n * @typeParam TContextData The type of the context data.\n * @param actor The actor to mention.\n * @returns A {@link MentionText} tree.\n */\nexport function mention<TContextData>(\n actor: Actor | URL,\n): Text<\"inline\", TContextData>;\n\n/**\n * Mentions an actor with a custom label. You can use this function to create\n * a {@link MentionText} tree from an actor object with a custom label.\n *\n * If the given actor is a URL and the URL does not refer to an actor,\n * the returned tree consists of a plain text with the URL without any link.\n * @typeParam TContextData The type of the context data.\n * @param label The label of the mention.\n * @param actor The actor to mention.\n */\nexport function mention<TContextData>(\n label: string,\n actor: Actor | URL,\n): Text<\"inline\", TContextData>;\n\nexport function mention<TContextData>(\n a: string | Actor | URL,\n b?: Actor | URL,\n): Text<\"inline\", TContextData> {\n if (b != null) {\n // (label: string, actor: Actor | URL)\n return new MentionText<TContextData>(\n a as string,\n isActor(b) ? b : async (session) => {\n if (session.actorId.href === b.href) return await session.getActor();\n const documentLoader = await session.context.getDocumentLoader(\n session.bot,\n );\n return await session.context.lookupObject(b, { documentLoader });\n },\n );\n } else if (typeof a === \"string\") {\n // (handle: string)\n return new MentionText<TContextData>(\n a,\n async (session) => {\n if (session.actorHandle === a) return await session.getActor();\n const documentLoader = await session.context.getDocumentLoader(\n session.bot,\n );\n return await session.context.lookupObject(a, { documentLoader });\n },\n );\n } else if (isActor(a)) {\n // (actor: Actor)\n return new MentionText<TContextData>(\n (session) =>\n a.id?.href === session.actorId.href\n ? Promise.resolve(session.actorHandle)\n : getActorHandle(a, session.context),\n a,\n );\n }\n // (actor: URL)\n return new MentionText<TContextData>(\n (session) =>\n a.href === session.actorId.href\n ? Promise.resolve(session.actorHandle)\n : getActorHandle(a, session.context),\n async (session) => {\n if (a.href === session.actorId.href) return await session.getActor();\n const documentLoader = await session.context.getDocumentLoader(\n session.bot,\n );\n return await session.context.lookupObject(a, { documentLoader });\n },\n );\n}\n\n/**\n * A text tree that renders a hashtag. You normally don't need to\n * instantiate this directly; use the {@link hashtag} function instead.\n * @typeParam TContextData The type of the context data.\n */\nexport class HashtagText<TContextData> implements Text<\"inline\", TContextData> {\n readonly type = \"inline\";\n #tag: string;\n\n /**\n * Creates a {@link HashtagText} tree with a tag.\n * @param tag The hashtag. It does not matter whether it starts with `\"#\"`.\n */\n constructor(tag: string) {\n this.#tag = tag.trimStart().replace(/^#/, \"\").trim().replace(/\\s+/g, \" \");\n }\n\n async *getHtml(session: Session<TContextData>): AsyncIterable<string> {\n yield '<a href=\"';\n yield encode(session.context.origin);\n yield \"/tags/\";\n yield encode(encodeURIComponent(this.#tag.toLowerCase()));\n yield '\" class=\"mention hashtag\" rel=\"tag\" target=\"_blank\">#<span>';\n yield this.#tag;\n yield \"</span></a>\";\n }\n\n async *getTags(session: Session<TContextData>): AsyncIterable<Link | Object> {\n yield new Hashtag({\n href: new URL(\n `/tags/${encodeURIComponent(this.#tag.toLowerCase())}`,\n session.context.origin,\n ),\n name: `#${this.#tag.toLowerCase()}`,\n });\n }\n\n getCachedObjects(): Object[] {\n return [];\n }\n}\n\n/**\n * Creates a hashtag. You can use this function to create a {@link HashtagText}\n * tree.\n * @param tag The hashtag. It does not matter whether it starts with `\"#\"`.\n * @returns A {@link HashtagText} tree.\n */\nexport function hashtag<TContextData>(\n tag: string,\n): Text<\"inline\", TContextData> {\n return new HashtagText(tag);\n}\n\n/**\n * A text tree that renders a `<strong>` text. You normally don't need to\n * instantiate this directly; use the {@link strong} function instead.\n * @typeParam TContextData The type of the context data.\n */\nexport class StrongText<TContextData> implements Text<\"inline\", TContextData> {\n readonly type = \"inline\";\n #text: Text<\"inline\", TContextData>;\n\n /**\n * Creates a {@link StrongText} tree with a text.\n * @param text The text to render as `<strong>`.\n */\n constructor(text: Text<\"inline\", TContextData> | string) {\n this.#text = typeof text === \"string\" ? new PlainText(text) : text;\n }\n\n async *getHtml(session: Session<TContextData>): AsyncIterable<string> {\n yield \"<strong>\";\n yield* this.#text.getHtml(session);\n yield \"</strong>\";\n }\n\n getTags(session: Session<TContextData>): AsyncIterable<Link | Object> {\n return this.#text.getTags(session);\n }\n\n getCachedObjects(): Object[] {\n return [];\n }\n}\n\n/**\n * Applies `<strong>` tag to a text. You can use this function to create a\n * {@link StrongText} tree.\n * @typeParam TContextData The type of the context data.\n * @param text The text to render as `<strong>`. It can be a plain text or\n * another text tree.\n * @returns A {@link StrongText} tree.\n */\nexport function strong<TContextData>(\n text: Text<\"inline\", TContextData> | string,\n): Text<\"inline\", TContextData> {\n return new StrongText(text);\n}\n\n/**\n * A text tree that renders an `<em>` text. You normally don't need to\n * instantiate this directly; use the {@link em} function instead.\n * @typeParam TContextData The type of the context data.\n */\nexport class EmText<TContextData> implements Text<\"inline\", TContextData> {\n readonly type = \"inline\";\n #text: Text<\"inline\", TContextData>;\n\n constructor(text: Text<\"inline\", TContextData> | string) {\n this.#text = typeof text === \"string\" ? new PlainText(text) : text;\n }\n\n async *getHtml(session: Session<TContextData>): AsyncIterable<string> {\n yield \"<em>\";\n yield* this.#text.getHtml(session);\n yield \"</em>\";\n }\n\n getTags(session: Session<TContextData>): AsyncIterable<Link | Object> {\n return this.#text.getTags(session);\n }\n\n getCachedObjects(): Object[] {\n return [];\n }\n}\n\n/**\n * Applies `<em>` tag to a text. You can use this function to create an\n * {@link EmText} tree.\n * @typeParam TContextData The type of the context data.\n * @param text The text to render as `<em>`. It can be a plain text or\n * another text tree.\n * @returns A {@link EmText} tree.\n */\nexport function em<TContextData>(\n text: Text<\"inline\", TContextData> | string,\n): Text<\"inline\", TContextData> {\n return new EmText(text);\n}\n\n/**\n * A text tree that renders a link. You normally don't need to instantiate\n * this directly; use the {@link link} function instead.\n * @typeParam TContextData The type of the context data.\n */\nexport class LinkText<TContextData> implements Text<\"inline\", TContextData> {\n readonly type = \"inline\";\n #label: Text<\"inline\", TContextData>;\n #href: URL;\n\n /**\n * Creates a {@link LinkText} tree with a label and a URL.\n * @param label The label of the link.\n * @param href The URL of the link. It has to be an absolute URL.\n */\n constructor(\n label: Text<\"inline\", TContextData> | string,\n href: URL | string,\n ) {\n this.#label = typeof label === \"string\" ? new PlainText(label) : label;\n this.#href = typeof href === \"string\" ? new URL(href) : href;\n }\n\n async *getHtml(session: Session<TContextData>): AsyncIterable<string> {\n yield '<a href=\"';\n yield encode(this.#href.href);\n yield '\" target=\"_blank\">';\n yield* this.#label.getHtml(session);\n yield \"</a>\";\n }\n\n getTags(session: Session<TContextData>): AsyncIterable<Link | Object> {\n return this.#label.getTags(session);\n }\n\n getCachedObjects(): Object[] {\n return this.#label.getCachedObjects();\n }\n}\n\n/**\n * Creates a link to the given `href` with the `label`. You can use this\n * function to create a {@link LinkText} tree.\n * @typeParam TContextData The type of the context data.\n * @param label The displayed label of the link.\n * @param href The link target. It has to be an absolute URL.\n * @returns A {@link LinkText} tree.\n */\nexport function link<TContextData>(\n label: Text<\"inline\", TContextData> | string,\n href: URL | string,\n): Text<\"inline\", TContextData>;\n\n/**\n * Creates a link to the given `url` with no label. You can use this function\n * to create a {@link LinkText} tree. The label of the link will be the same\n * as the given `url`.\n * @param url The link target. It has to be an absolute URL.\n * @returns A {@link LinkText} tree.\n */\nexport function link<TContextData>(\n url: URL | string,\n): Text<\"inline\", TContextData>;\n\nexport function link<TContextData>(\n label: Text<\"inline\", TContextData> | string | URL,\n href?: URL | string,\n): Text<\"inline\", TContextData> {\n return href == null\n ? new LinkText(String(label), label as string)\n : new LinkText(\n isText<TContextData>(label) ? label : label.toString(),\n href,\n );\n}\n\n/**\n * A text tree that renders a inline code. You normally don't need to\n * instantiate this directly; use the {@link code} function instead.\n * @typeParam TContextData The type of the context data.\n */\nexport class CodeText<TContextData> implements Text<\"inline\", TContextData> {\n readonly type = \"inline\";\n readonly #code: Text<\"inline\", TContextData>;\n\n /**\n * Creates a {@link CodeText} tree with a code.\n * @param code The code to render.\n */\n constructor(code: Text<\"inline\", TContextData> | string) {\n this.#code = typeof code === \"string\" ? new PlainText(code) : code;\n }\n\n async *getHtml(session: Session<TContextData>): AsyncIterable<string> {\n yield \"<code>\";\n yield* this.#code.getHtml(session);\n yield \"</code>\";\n }\n\n getTags(session: Session<TContextData>): AsyncIterable<Link | Object> {\n return this.#code.getTags(session);\n }\n\n getCachedObjects(): Object[] {\n return [];\n }\n}\n\n/**\n * Applies `<code>` tag to a text. You can use this function to create\n * a {@link CodeText} tree.\n * @param code The code to render.\n * @returns A {@link CodeText} tree.\n */\nexport function code<TContextData>(\n code: Text<\"inline\", TContextData> | string,\n): Text<\"inline\", TContextData> {\n return new CodeText(code);\n}\n\n/**\n * A text tree that renders a custom emoji. You normally don't need to\n * instantiate this directly; use the {@link customEmoji} function instead.\n * @typeParam TContextData The type of the context data.\n * @since 0.2.0\n */\nexport class CustomEmojiText<TContextData>\n implements Text<\"inline\", TContextData> {\n readonly type = \"inline\";\n readonly #emoji: Emoji | DeferredCustomEmoji<TContextData>;\n\n /**\n * Creates a {@link CustomEmojiText} tree with a custom emoji.\n * @param emoji The custom emoji to render.\n */\n constructor(emoji: Emoji | DeferredCustomEmoji<TContextData>) {\n this.#emoji = emoji;\n }\n\n /**\n * Gets the emoji object. If the emoji is a deferred emoji, it will\n * be resolved with the given session.\n * @param session The bot session.\n * @returns The emoji object.\n */\n getEmoji(session: Session<TContextData>): Emoji {\n if (typeof this.#emoji === \"function\") return this.#emoji(session);\n return this.#emoji;\n }\n\n async *getHtml(session: Session<TContextData>): AsyncIterable<string> {\n const emoji = this.getEmoji(session);\n if (emoji.name == null) return;\n yield \"\\u200b\"; // zero-width space for segmentation\n yield encode(emoji.name.toString());\n yield \"\\u200b\";\n }\n\n async *getTags(\n session: Session<TContextData>,\n ): AsyncIterable<Link | Object> {\n yield this.getEmoji(session);\n }\n\n getCachedObjects(): Object[] {\n return [];\n }\n}\n\n/**\n * Renders a custom emoji. You can use this function to create a\n * {@link CustomEmojiText} tree.\n * @param emoji The custom emoji to render. See also {@link Bot.addCustomEmojis}\n * method.\n * @returns A {@link CustomEmojiText} tree.\n * @since 0.2.0\n */\nexport function customEmoji<TContextData>(\n emoji: Emoji | DeferredCustomEmoji<TContextData>,\n): Text<\"inline\", TContextData> {\n return new CustomEmojiText(emoji);\n}\n\n/**\n * The options for rendering a Markdown text.\n */\nexport interface MarkdownTextOptions {\n /**\n * Whether to render mentions in the Markdown text.\n * @default {true}\n */\n readonly mentions?: boolean;\n\n /**\n * Whether to render hashtags in the Markdown text.\n * @default {true}\n */\n readonly hashtags?: boolean;\n\n /**\n * Whether to automatically linkify URLs in the Markdown text.\n * @default {true}\n */\n readonly linkify?: boolean;\n}\n\ninterface MarkdownEnv {\n mentions: string[];\n hashtags: string[];\n origin: string;\n actors?: Record<string, string | null>;\n}\n\n/**\n * A text tree that renders a Markdown text. You normally don't need to\n * instantiate this directly; use the {@link markdown} function instead.\n */\nexport class MarkdownText<TContextData> implements Text<\"block\", TContextData> {\n readonly type = \"block\";\n readonly #content: string;\n readonly #markdownIt: MarkdownIt;\n readonly #mentions?: string[];\n readonly #hashtags?: string[];\n #actors?: Record<string, Object>;\n\n /**\n * Creates a {@link MarkdownText} tree with a Markdown content.\n * @param content The Markdown content.\n * @param options The options for rendering the Markdown content.\n */\n constructor(content: string, options: MarkdownTextOptions = {}) {\n this.#content = content;\n const md = MarkdownIt({\n html: false,\n linkify: options.linkify ?? true,\n });\n if (options.mentions ?? true) {\n md.use(mentionPlugin, {\n link(handle: string, env: MarkdownEnv) {\n if (env.actors == null) return `acct:${handle}`;\n return env.actors[handle] ?? null;\n },\n linkAttributes(_handle: string, _env: MarkdownEnv) {\n return {\n translate: \"no\",\n class: \"h-card u-url mention\",\n target: \"_blank\",\n };\n },\n label: toFullHandle,\n });\n const env: MarkdownEnv = {\n mentions: [],\n hashtags: [],\n origin: \"http://localhost\",\n };\n md.render(content, env);\n this.#mentions = env.mentions;\n }\n if (options.hashtags ?? true) {\n md.use(hashtagPlugin, {\n link(hashtag: string, env: MarkdownEnv) {\n const tag = hashtag.substring(1).toLowerCase();\n return new URL(`/tags/${encodeURIComponent(tag)}`, env.origin).href;\n },\n linkAttributes(_hashtag: string, _env: MarkdownEnv) {\n return {\n class: \"mention hashtag\",\n rel: \"tag\",\n target: \"_blank\",\n };\n },\n label(hashtag: string, _env: MarkdownEnv) {\n const tag = hashtag.substring(1);\n return `#<span>${tag}</span>`;\n },\n });\n const env: MarkdownEnv = {\n mentions: [],\n hashtags: [],\n origin: \"http://localhost\",\n };\n md.render(content, env);\n this.#hashtags = env.hashtags;\n }\n this.#markdownIt = md;\n }\n\n async #getMentionedActors(\n session: Session<TContextData>,\n ): Promise<Record<string, Object>> {\n if (this.#mentions == null) return {};\n if (this.#actors != null) return this.#actors;\n const documentLoader = await session.context.getDocumentLoader(session.bot);\n const objects = await Promise.all(\n this.#mentions.map((m) =>\n m === session.actorHandle\n ? session.getActor()\n : session.context.lookupObject(m, { documentLoader })\n ),\n );\n const actors: Record<string, Object> = {};\n for (let i = 0; i < this.#mentions.length; i++) {\n const object = objects[i];\n if (object != null) actors[this.#mentions[i]] = object;\n }\n this.#actors = actors;\n return actors;\n }\n\n async *getHtml(session: Session<TContextData>): AsyncIterable<string> {\n if (this.#mentions == null) {\n yield this.#markdownIt.render(this.#content);\n return;\n }\n const actors: Record<string, string | null> = globalThis.Object.fromEntries(\n globalThis.Object.entries(\n await this.#getMentionedActors(session),\n ).filter(([_, obj]) => isActor(obj)).map((\n [handle, actor],\n ) =>\n [\n handle,\n (actor.url instanceof Link\n ? actor.url.href?.href\n : actor.url?.href) ?? actor.id?.href ?? null,\n ] satisfies [string, string | null]\n ).filter(([_, url]) => url != null),\n );\n const env: MarkdownEnv = {\n mentions: [],\n hashtags: [],\n origin: session.context.origin,\n actors,\n };\n yield this.#markdownIt.render(this.#content, env);\n }\n\n async *getTags(session: Session<TContextData>): AsyncIterable<Link | Object> {\n if (this.#mentions == null) return;\n const actors = await this.#getMentionedActors(session);\n for (const [handle, object] of globalThis.Object.entries(actors)) {\n if (!isActor(object) || object.id == null) continue;\n yield new Mention({\n name: handle,\n href: object.id,\n });\n }\n if (this.#hashtags != null) {\n for (const hashtag of this.#hashtags) {\n const tag = hashtag.substring(1).toLowerCase();\n yield new Hashtag({\n name: `#${tag}`,\n href: new URL(\n `/tags/${encodeURIComponent(tag)}`,\n session.context.origin,\n ),\n });\n }\n }\n }\n\n getCachedObjects(): Object[] {\n return this.#actors == null ? [] : globalThis.Object.values(this.#actors);\n }\n}\n\n/**\n * Renders a Markdown text. You can use this function to create\n * a {@link MarkdownText} tree. The mentions in the Markdown text\n * will be rendered as links unless the `mentions` option is set to\n * `false`.\n * @param content The Markdown content.\n * @param options The options for rendering the Markdown content.\n * @returns A {@link MarkdownText} tree.\n */\nexport function markdown<TContextData>(\n content: string,\n options: MarkdownTextOptions = {},\n): Text<\"block\", TContextData> {\n return new MarkdownText(content, options);\n}\n"],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import { Temporal, toTemporalInstant } from "@js-temporal/polyfill";
2
+ Date.prototype.toTemporalInstant = toTemporalInstant;
@@ -0,0 +1,473 @@
1
+
2
+ import { Temporal, toTemporalInstant } from "@js-temporal/polyfill";
3
+ Date.prototype.toTemporalInstant = toTemporalInstant;
4
+
5
+ import { BotImpl } from "./bot-impl.js";
6
+ import { CustomEmojiText, code, customEmoji, em, hashtag, isText, link, markdown, mention, mentions, strong, text } from "./text.js";
7
+ import { MemoryKvStore, createFederation } from "@fedify/fedify/federation";
8
+ import { Emoji, Hashtag, Image, Mention, Person } from "@fedify/fedify/vocab";
9
+ import assert from "node:assert";
10
+ import { describe, test } from "node:test";
11
+ import { getDocumentLoader } from "@fedify/fedify/runtime";
12
+ import { importJwk } from "@fedify/fedify/sig";
13
+
14
+ //#region src/text.test.ts
15
+ const defaultDocumentLoader = getDocumentLoader();
16
+ const federation = createFederation({
17
+ kv: new MemoryKvStore(),
18
+ documentLoader(url) {
19
+ const parsed = new URL(url);
20
+ if (parsed.host !== "example.com") return defaultDocumentLoader(url);
21
+ const document = {
22
+ "@context": "https://www.w3.org/ns/activitystreams",
23
+ type: "Person",
24
+ id: url,
25
+ preferredUsername: url.split("/").at(-1),
26
+ url
27
+ };
28
+ return Promise.resolve({
29
+ document,
30
+ documentUrl: url,
31
+ contextUrl: null
32
+ });
33
+ },
34
+ authenticatedDocumentLoaderFactory(_identity) {
35
+ return this.documentLoader;
36
+ }
37
+ });
38
+ const keyPair = {
39
+ privateKey: await importJwk({
40
+ kty: "RSA",
41
+ alg: "RS256",
42
+ n: "15DAu9S9sROZ_NonfZm5S0PSeYuh2POeo0cpvvGnp_T9jQWjMuGdhwOPT7OOD9N-R1IY-2hXkk-RjfWNzbxMoNKvOpz_1MHRg18W5Lw60mIxZFztLKlNhOZS7rVrlJc--jj1wLETEfY5ocYCyRZm25UeT_Q1JSePdnhEGVXo4sSqoUcMV5Bgys5PlISfQj_bHDpcIi9snJ70hOzYTy7k7fNuCHHsK08DN4bZMG58qQNrPFtZW6fgpQiu0kgtUBFgQu-uUNn1h0io-7OhMU2dXeV8lwCILhIFRvrRV9vKQydBejbyYuzFY-Xq98biB9Aox8GD0jJE4tTVj6CpsmaN9yq6nI8ibWjnk87IKdU3jex5zB8chR0cm2tyIWr6dKhCTexmtYTF0pGW7PZ2Dnn31BU3cPkrkplu752D4eQ47BsAspMsHRxXE3NtqlmN02Y-6AIzt1tuPBPHldQHUpxtGrwFh9b9biC2mtb_w6F0oyxVyZsAZnuK2tN0-uX_iMPoC5VLnPrzgWjQiMArivg4u1cQ35hrvuFYFAdNf2WaBDyDNxaCoTD28z9bF1titlbQ48tDw0adZ1Zp8_x12JqA5HNpqDfmyT4KPU_6Ag7J7cGfGeO2vsjcc1oGhqZ-n6JlnSvVImHS5eKC6CDhW7ceuKC_oX-XPWWQQHhPniwM2RE",
43
+ e: "AQAB",
44
+ d: "eAD5ipdQUrfazcyUl3Nwl9nV3hxBqYlWEweW0dmtv-6_CDbPN5AqJfNxYKlQuLbAYevuRGc9-RGasjC1FIdzEUS4kCS-ty5--GeDUysGhABuBrVEw8wsf4PJP2J31WytfpcfGHp7Z1BvnQOioVd7Q1qsWU5WF60CTK1_G6ubzkI1yzrGQCj7-WsJGmEKV9M8o2ZJzC4ihL5o2WcQtGQixeTyqHjjROjjnZHQbwnTFDP3Cs6_3CqFANrol9_eeehyclED9bag3QMyL408ezn-FTugNF_zb9JQZcdTq1mMK_46kVLtdOzipk5klDN_uWHEkg_E1sttVemuShri3ZICDUSd70Y4VeQxNLUKJNBSYdSLmcVgfIHaXMmcrknmBuz23SrGR6JZR4DSJtr3sylR2tlOxpZhJAUZf17f8aZD7EnbR7qcNtjZmf8RyAKrEXLgL6f3jm6FfE_b027kcmLMXL7bJtlTBYnM9MrBnXSsJftHRrmB1Xe-0Hq24Q7ctrhRhFF3Ij8MjNRjdn6NWdIXzltreblLEO44iTJtlWeYtg-C9566F_yWZnjDZEQ8nvBhpCM3iXlRfzhlEebBoBbf-Mf-0hJRRUL3EtGuMueylzeyk4tTzvOfK2FUnVAi8bIhjz4m8RhN7kC4ubqUXZbKwzjeI1dhWyfDphrPiRtCetE",
45
+ p: "_4BmuvmlMZmiD_uGE__iWIHizMe4Ay34WOSNAm0uhjfNgWpWHdxosQQeSehBE8x_CwJGKbCl69ewmEAOYnP3KYpytvcHVEMO7XQAE66p_19PzIy95WY6CR20CFXUykzQ_6gE_nFB39rxcxprxAqaNEGM2PtwpiTSwoyZFWkgSE6O6rwv03vNAMeib3CZU3kdg4mHiZAryk9U6uphZ_lJ2wOSMX_SHkcEAwePoae8E_jIJ0drfIkyR5pSkghThjHIv95hcHuuDodm2W9e7sppv3EpjE81ANKEfkrrauDfX9BrRDC-14hhCK7t7bYJ9bj49nmSSLmLHNw1W2eyvvAHEw",
46
+ q: "1_xoQ6UBnAsql-87v7UddCiJEppk9LkfYb-DBEtBHbTFDvxbE0Px7ooNevgwlMDmi6t4hSkQZEwF5sLFHv1-fNFtpwVGzqu6ekXYfw0FxWRVjJTh5e91G3mOMVprGvgSMk3PRO33dBAhA6MMGIMi72XCGWNpy7ajfawYdpeMO6k6aC93jT9RmoaB_UmwyM2lmzaWP9RKInJ6t_i-eFm05_0qkPdPdg5b3HbNpKyAMLVxA8DAauHvcCx9CsAZ8ZbvL-mC5AX7v7B9iovXCStHpfA5f73M7AGFGRZpt51mvx8pxHVwF2tW6h03HEBz5BX-9YMTaepH369CRovGo4Jvyw",
47
+ dp: "hlBlsN0T7mMpQuWisljOEGEXbTeAkItWBsT_K8thrcUgD2xrIP-BOa1Eju29aD8UeiET6U6nqreUajUiWrdDs17It05dV_p4mnNkpvQnAcyFEq7aFQIMeEZZIhic6ExBgmQ9W9UGIDvkufGlvUUlk1ryRA7KRU0OTp_CyfKdueUyVEvhiHeIaWSJC7RRpgQBc-iUi8hyfMP_jA7ybcoq_St_au4a8ze58C3FX-HhiU47SgrNgoZNHD8QMRyXa_A37EVnS854zcJ4Ws2lRjq6JJ3EjbIF1wzUAeA4qdLVGnViLlLBwGQ9PmdXRKNx0O8QUeHO-NQxQVax5f85hA6CaQ",
48
+ dq: "MVKMpNXrlizeny-cn1zGyx3un3bukwwrZHENhE-DITuEvLVYPwAHIYgZJ_nBblbWzxJrRU1pVt4dguL7jOYqmmpg9gE4eD2zKfUFSY45wSf2eVIOfCnAvnN1y0Nwrgn0bdRi_sSw-6orP99eBcL8mVrNhmqzYDfnAe3o8DwPZBhzJBOi43iQNA9_Y84ONuzvYpCGozDhdRhbeeOt62Hg9BFWRSCU3srMo33l3DMgWv80Pb0os7_ApAckzu2rfwYOvQxAPb44DUBKivcANjHR_Mzs9ITtZP-720zI-4tQSVjeeuSuokp64J-nVCZL0MxNGtfB-S_tFeG56s5EoFZLHQ",
49
+ qi: "1J7CfWYlg4Igsu2N7bhgLzbc1l6A3odyyOlM70uH8P41kCYgpRDdH8Ms8yOJE-F13ha5drICZqsD7IjgG0cZONJ_0xTeka0AYMvCwjuJZ_4CzVFYNICxSHFUI-sCu1p-zb70eXU6fiwOFgzoPbnrwywpbxcTV_8H0XszwPcI3fjrGk6N-hi23Ur1gIjhnri_-x8mzwmtPA1ID1G17U4X93mP7dlYCzGigq8ORbSdZthOKdjtHXITBOgpcTiuyTTwAEqh3xyXscfsgzi0X6olBevJCGeTzOrqQX026JmNVykaS1-o_ea_Y0cD0q6Nxd5TwLZMCLZi1M5PLHhGlJg9MQ",
50
+ key_ops: ["sign"],
51
+ ext: true
52
+ }, "private"),
53
+ publicKey: await importJwk({
54
+ kty: "RSA",
55
+ alg: "RS256",
56
+ n: "15DAu9S9sROZ_NonfZm5S0PSeYuh2POeo0cpvvGnp_T9jQWjMuGdhwOPT7OOD9N-R1IY-2hXkk-RjfWNzbxMoNKvOpz_1MHRg18W5Lw60mIxZFztLKlNhOZS7rVrlJc--jj1wLETEfY5ocYCyRZm25UeT_Q1JSePdnhEGVXo4sSqoUcMV5Bgys5PlISfQj_bHDpcIi9snJ70hOzYTy7k7fNuCHHsK08DN4bZMG58qQNrPFtZW6fgpQiu0kgtUBFgQu-uUNn1h0io-7OhMU2dXeV8lwCILhIFRvrRV9vKQydBejbyYuzFY-Xq98biB9Aox8GD0jJE4tTVj6CpsmaN9yq6nI8ibWjnk87IKdU3jex5zB8chR0cm2tyIWr6dKhCTexmtYTF0pGW7PZ2Dnn31BU3cPkrkplu752D4eQ47BsAspMsHRxXE3NtqlmN02Y-6AIzt1tuPBPHldQHUpxtGrwFh9b9biC2mtb_w6F0oyxVyZsAZnuK2tN0-uX_iMPoC5VLnPrzgWjQiMArivg4u1cQ35hrvuFYFAdNf2WaBDyDNxaCoTD28z9bF1titlbQ48tDw0adZ1Zp8_x12JqA5HNpqDfmyT4KPU_6Ag7J7cGfGeO2vsjcc1oGhqZ-n6JlnSvVImHS5eKC6CDhW7ceuKC_oX-XPWWQQHhPniwM2RE",
57
+ e: "AQAB",
58
+ key_ops: ["verify"],
59
+ ext: true
60
+ }, "public")
61
+ };
62
+ federation.setActorDispatcher("/ap/actor/{identifier}", (ctx, identifier) => {
63
+ return new Person({
64
+ id: ctx.getActorUri(identifier),
65
+ preferredUsername: identifier
66
+ });
67
+ }).setKeyPairsDispatcher((_ctx, _identifier) => {
68
+ return [keyPair];
69
+ });
70
+ const bot = {
71
+ federation,
72
+ identifier: "bot",
73
+ getSession(origin, _contextData) {
74
+ const ctx = typeof origin === "string" || origin instanceof URL ? federation.createContext(new URL(origin)) : origin;
75
+ return {
76
+ bot,
77
+ context: ctx,
78
+ actorId: ctx.getActorUri(bot.identifier),
79
+ actorHandle: `@bot@${ctx.host}`,
80
+ getActor() {
81
+ const actor = new Person({
82
+ id: ctx.getActorUri(bot.identifier),
83
+ preferredUsername: "bot"
84
+ });
85
+ return Promise.resolve(actor);
86
+ },
87
+ follow() {
88
+ throw new Error("Not implemented");
89
+ },
90
+ unfollow() {
91
+ throw new Error("Not implemented");
92
+ },
93
+ follows() {
94
+ throw new Error("Not implemented");
95
+ },
96
+ publish() {
97
+ throw new Error("Not implemented");
98
+ },
99
+ getOutbox() {
100
+ throw new Error("Not implemented");
101
+ }
102
+ };
103
+ },
104
+ fetch(_req) {
105
+ return Promise.resolve(new Response());
106
+ },
107
+ addCustomEmojis(_emojis) {
108
+ return {};
109
+ }
110
+ };
111
+ test("isText()", () => {
112
+ const t = text`Hello, World`;
113
+ assert.ok(isText(t));
114
+ const t2 = em("Hello, World");
115
+ assert.ok(isText(t2));
116
+ assert.deepStrictEqual(isText("Hello, World"), false);
117
+ });
118
+ test("mentions()", async () => {
119
+ const session = bot.getSession("https://example.com");
120
+ const actor = new URL("https://hollo.social/@fedify");
121
+ const actor2 = new URL("https://example.com/users/john");
122
+ const actor3 = new Person({ id: actor });
123
+ const t = text`Hello, world!`;
124
+ assert.deepStrictEqual(await mentions(session, t, actor), false);
125
+ assert.deepStrictEqual(await mentions(session, t, actor2), false);
126
+ assert.deepStrictEqual(await mentions(session, t, actor3), false);
127
+ const m = mention(actor);
128
+ assert.ok(await mentions(session, m, actor));
129
+ assert.deepStrictEqual(await mentions(session, m, actor2), false);
130
+ assert.ok(await mentions(session, m, actor3));
131
+ });
132
+ test("text`...`", async () => {
133
+ const session = bot.getSession("https://example.com");
134
+ const t = text`Hello, <${123}>`;
135
+ assert.deepStrictEqual((await Array.fromAsync(t.getHtml(session))).join(""), "<p>Hello, &lt;123&gt;</p>");
136
+ assert.deepStrictEqual(await Array.fromAsync(t.getTags(session)), []);
137
+ assert.deepStrictEqual(t.getCachedObjects(), []);
138
+ const t2 = text`Hello, ${em("World")}`;
139
+ assert.deepStrictEqual((await Array.fromAsync(t2.getHtml(session))).join(""), "<p>Hello, <em>World</em></p>");
140
+ assert.deepStrictEqual(await Array.fromAsync(t2.getTags(session)), []);
141
+ assert.deepStrictEqual(t2.getCachedObjects(), []);
142
+ const actor = new Person({
143
+ id: new URL("https://example.com/users/john"),
144
+ preferredUsername: "john",
145
+ url: new URL("https://example.com/@john")
146
+ });
147
+ const t3 = text`Hello, ${mention(actor)}`;
148
+ assert.deepStrictEqual((await Array.fromAsync(t3.getHtml(session))).join(""), "<p>Hello, <a href=\"https://example.com/@john\" translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\">@<span>john@example.com</span></a></p>");
149
+ const tags3 = await Array.fromAsync(t3.getTags(session));
150
+ assert.deepStrictEqual(tags3.length, 1);
151
+ assert.ok(tags3[0] instanceof Mention);
152
+ assert.deepStrictEqual(tags3[0].name, "@john@example.com");
153
+ assert.deepStrictEqual(tags3[0].href, new URL("https://example.com/users/john"));
154
+ const cache3 = t3.getCachedObjects();
155
+ assert.deepStrictEqual(cache3.length, 1);
156
+ assert.ok(cache3[0] instanceof Person);
157
+ assert.deepStrictEqual(cache3[0].id, new URL("https://example.com/users/john"));
158
+ const t4 = text`Hello\nworld!`;
159
+ assert.deepStrictEqual((await Array.fromAsync(t4.getHtml(session))).join(""), "<p>Hello<br>world!</p>");
160
+ assert.deepStrictEqual(await Array.fromAsync(t4.getTags(session)), []);
161
+ assert.deepStrictEqual(t4.getCachedObjects(), []);
162
+ const t5 = text`Hello\nworld!\n\nGoodbye!\n\t\n \nHello!`;
163
+ assert.deepStrictEqual((await Array.fromAsync(t5.getHtml(session))).join(""), "<p>Hello<br>world!</p><p>Goodbye!</p><p>Hello!</p>");
164
+ assert.deepStrictEqual(await Array.fromAsync(t5.getTags(session)), []);
165
+ assert.deepStrictEqual(t5.getCachedObjects(), []);
166
+ const t6 = text`\n\n\nHello\nworld\n\n\nGoodbye!\n`;
167
+ assert.deepStrictEqual((await Array.fromAsync(t6.getHtml(session))).join(""), "<p>Hello<br>world</p><p>Goodbye!</p>");
168
+ assert.deepStrictEqual(await Array.fromAsync(t6.getTags(session)), []);
169
+ assert.deepStrictEqual(t6.getCachedObjects(), []);
170
+ const t7 = text`Here's a link: ${new URL("https://fedify.dev/")}.`;
171
+ assert.deepStrictEqual((await Array.fromAsync(t7.getHtml(session))).join(""), "<p>Here&apos;s a link: <a href=\"https://fedify.dev/\" target=\"_blank\">https://fedify.dev/</a>.</p>");
172
+ assert.deepStrictEqual(await Array.fromAsync(t7.getTags(session)), []);
173
+ assert.deepStrictEqual(t7.getCachedObjects(), []);
174
+ const t8 = text`Here's a multiline text:
175
+
176
+ ${"First line.\nSecond line."}`;
177
+ assert.deepStrictEqual((await Array.fromAsync(t8.getHtml(session))).join(""), "<p>Here&apos;s a multiline text:</p><p>First line.<br>Second line.</p>");
178
+ assert.deepStrictEqual(await Array.fromAsync(t8.getTags(session)), []);
179
+ assert.deepStrictEqual(t8.getCachedObjects(), []);
180
+ const t9 = text`Interpolating blocks: ${text`Hello\nworld!`} ... and ... ${text`Goodbye!`}`;
181
+ assert.deepStrictEqual((await Array.fromAsync(t9.getHtml(session))).join(""), "<p>Interpolating blocks: </p><p>Hello<br>world!</p><p> ... and ... </p><p>Goodbye!</p>");
182
+ assert.deepStrictEqual(await Array.fromAsync(t9.getTags(session)), []);
183
+ assert.deepStrictEqual(t9.getCachedObjects(), []);
184
+ const t10 = text`Interpolating blocks:\n\n${text`Hello\nworld!`}\n\n... and ...\n\n${text`Goodbye!`}`;
185
+ assert.deepStrictEqual((await Array.fromAsync(t10.getHtml(session))).join(""), "<p>Interpolating blocks:</p><p>Hello<br>world!</p><p>... and ...</p><p>Goodbye!</p>");
186
+ assert.deepStrictEqual(await Array.fromAsync(t10.getTags(session)), []);
187
+ assert.deepStrictEqual(t10.getCachedObjects(), []);
188
+ });
189
+ test("mention()", async () => {
190
+ const session = bot.getSession("https://example.com");
191
+ const m = mention(new Person({
192
+ id: new URL("https://example.com/users/john"),
193
+ preferredUsername: "john",
194
+ url: new URL("https://example.com/@john")
195
+ }));
196
+ assert.deepStrictEqual((await Array.fromAsync(m.getHtml(session))).join(""), "<a href=\"https://example.com/@john\" translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\">@<span>john@example.com</span></a>");
197
+ const tags = await Array.fromAsync(m.getTags(session));
198
+ assert.deepStrictEqual(tags.length, 1);
199
+ assert.ok(tags[0] instanceof Mention);
200
+ assert.deepStrictEqual(tags[0].name, "@john@example.com");
201
+ assert.deepStrictEqual(tags[0].href, new URL("https://example.com/users/john"));
202
+ const cache = m.getCachedObjects();
203
+ assert.deepStrictEqual(cache.length, 1);
204
+ assert.ok(cache[0] instanceof Person);
205
+ assert.deepStrictEqual(cache[0].id, new URL("https://example.com/users/john"));
206
+ const m2 = mention("Jane Doe", new URL("https://example.com/@jane"));
207
+ assert.deepStrictEqual((await Array.fromAsync(m2.getHtml(session))).join(""), "<a href=\"https://example.com/@jane\" translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\">Jane Doe</a>");
208
+ const tags2 = await Array.fromAsync(m2.getTags(session));
209
+ assert.deepStrictEqual(tags2.length, 1);
210
+ assert.ok(tags2[0] instanceof Mention);
211
+ assert.deepStrictEqual(tags2[0].name, "Jane Doe");
212
+ assert.deepStrictEqual(tags2[0].href, new URL("https://example.com/@jane"));
213
+ const cache2 = m2.getCachedObjects();
214
+ assert.deepStrictEqual(cache2.length, 1);
215
+ assert.ok(cache2[0] instanceof Person);
216
+ assert.deepStrictEqual(cache2[0].id, new URL("https://example.com/@jane"));
217
+ const m3 = mention("John Doe", new Person({
218
+ id: new URL("https://example.com/users/john"),
219
+ preferredUsername: "john",
220
+ url: new URL("https://example.com/@john")
221
+ }));
222
+ assert.deepStrictEqual((await Array.fromAsync(m3.getHtml(session))).join(""), "<a href=\"https://example.com/@john\" translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\">John Doe</a>");
223
+ const tags3 = await Array.fromAsync(m3.getTags(session));
224
+ assert.deepStrictEqual(tags3.length, 1);
225
+ assert.ok(tags3[0] instanceof Mention);
226
+ assert.deepStrictEqual(tags3[0].name, "John Doe");
227
+ assert.deepStrictEqual(tags3[0].href, new URL("https://example.com/users/john"));
228
+ const cache3 = m3.getCachedObjects();
229
+ assert.deepStrictEqual(cache3.length, 1);
230
+ assert.ok(cache3[0] instanceof Person);
231
+ assert.deepStrictEqual(cache3[0].id, new URL("https://example.com/users/john"));
232
+ const m4 = mention("@fedify@hollo.social");
233
+ assert.deepStrictEqual((await Array.fromAsync(m4.getHtml(session))).join(""), "<a href=\"https://hollo.social/@fedify\" translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\">@<span>fedify@hollo.social</span></a>");
234
+ const tags4 = await Array.fromAsync(m4.getTags(session));
235
+ assert.deepStrictEqual(tags4.length, 1);
236
+ assert.ok(tags4[0] instanceof Mention);
237
+ assert.deepStrictEqual(tags4[0].name, "@fedify@hollo.social");
238
+ assert.deepStrictEqual(tags4[0].href, new URL("https://hollo.social/@fedify"));
239
+ const cache4 = m4.getCachedObjects();
240
+ assert.deepStrictEqual(cache4.length, 1);
241
+ assert.ok(cache4[0] instanceof Person);
242
+ assert.deepStrictEqual(cache4[0].id, new URL("https://hollo.social/@fedify"));
243
+ const m5 = mention(new URL("https://hollo.social/@fedify"));
244
+ assert.deepStrictEqual((await Array.fromAsync(m5.getHtml(session))).join(""), "<a href=\"https://hollo.social/@fedify\" translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\">@<span>fedify@hollo.social</span></a>");
245
+ const tags5 = await Array.fromAsync(m5.getTags(session));
246
+ assert.deepStrictEqual(tags5.length, 1);
247
+ assert.ok(tags5[0] instanceof Mention);
248
+ assert.deepStrictEqual(tags5[0].name, "@fedify@hollo.social");
249
+ assert.deepStrictEqual(tags5[0].href, new URL("https://hollo.social/@fedify"));
250
+ const cache5 = m5.getCachedObjects();
251
+ assert.deepStrictEqual(cache5.length, 1);
252
+ assert.ok(cache5[0] instanceof Person);
253
+ assert.deepStrictEqual(cache5[0].id, new URL("https://hollo.social/@fedify"));
254
+ const m6 = mention("@bot@example.com");
255
+ assert.deepStrictEqual((await Array.fromAsync(m6.getHtml(session))).join(""), "<a href=\"https://example.com/ap/actor/bot\" translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\">@<span>bot@example.com</span></a>");
256
+ const tags6 = await Array.fromAsync(m6.getTags(session));
257
+ assert.deepStrictEqual(tags6.length, 1);
258
+ assert.ok(tags6[0] instanceof Mention);
259
+ assert.deepStrictEqual(tags6[0].name, "@bot@example.com");
260
+ assert.deepStrictEqual(tags6[0].href, new URL("https://example.com/ap/actor/bot"));
261
+ const cache6 = m6.getCachedObjects();
262
+ assert.deepStrictEqual(cache6.length, 1);
263
+ assert.ok(cache6[0] instanceof Person);
264
+ assert.deepStrictEqual(cache6[0].id, new URL("https://example.com/ap/actor/bot"));
265
+ const m7 = mention("Example", new URL("https://example.com/ap/actor/bot"));
266
+ assert.deepStrictEqual((await Array.fromAsync(m7.getHtml(session))).join(""), "<a href=\"https://example.com/ap/actor/bot\" translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\">Example</a>");
267
+ const tags7 = await Array.fromAsync(m7.getTags(session));
268
+ assert.deepStrictEqual(tags7.length, 1);
269
+ assert.ok(tags7[0] instanceof Mention);
270
+ assert.deepStrictEqual(tags7[0].name, "Example");
271
+ assert.deepStrictEqual(tags7[0].href, new URL("https://example.com/ap/actor/bot"));
272
+ const cache7 = m7.getCachedObjects();
273
+ assert.deepStrictEqual(cache7.length, 1);
274
+ assert.ok(cache7[0] instanceof Person);
275
+ assert.deepStrictEqual(cache7[0].id, new URL("https://example.com/ap/actor/bot"));
276
+ const m8 = mention(new Person({
277
+ id: new URL("https://example.com/ap/actor/bot"),
278
+ preferredUsername: "bot"
279
+ }));
280
+ assert.deepStrictEqual((await Array.fromAsync(m8.getHtml(session))).join(""), "<a href=\"https://example.com/ap/actor/bot\" translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\">@<span>bot@example.com</span></a>");
281
+ const tags8 = await Array.fromAsync(m8.getTags(session));
282
+ assert.deepStrictEqual(tags8.length, 1);
283
+ assert.ok(tags8[0] instanceof Mention);
284
+ assert.deepStrictEqual(tags8[0].name, "@bot@example.com");
285
+ assert.deepStrictEqual(tags8[0].href, new URL("https://example.com/ap/actor/bot"));
286
+ const cache8 = m8.getCachedObjects();
287
+ assert.deepStrictEqual(cache8.length, 1);
288
+ assert.ok(cache8[0] instanceof Person);
289
+ assert.deepStrictEqual(cache8[0].id, new URL("https://example.com/ap/actor/bot"));
290
+ const m9 = mention(new URL("https://example.com/ap/actor/bot"));
291
+ assert.deepStrictEqual((await Array.fromAsync(m9.getHtml(session))).join(""), "<a href=\"https://example.com/ap/actor/bot\" translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\">@<span>bot@example.com</span></a>");
292
+ const tags9 = await Array.fromAsync(m9.getTags(session));
293
+ assert.deepStrictEqual(tags9.length, 1);
294
+ assert.ok(tags9[0] instanceof Mention);
295
+ assert.deepStrictEqual(tags9[0].name, "@bot@example.com");
296
+ assert.deepStrictEqual(tags9[0].href, new URL("https://example.com/ap/actor/bot"));
297
+ const cache9 = m9.getCachedObjects();
298
+ assert.deepStrictEqual(cache9.length, 1);
299
+ assert.ok(cache9[0] instanceof Person);
300
+ assert.deepStrictEqual(cache9[0].id, new URL("https://example.com/ap/actor/bot"));
301
+ });
302
+ test("hashtag()", async () => {
303
+ const session = bot.getSession("https://example.com");
304
+ const t = hashtag("Fediverse");
305
+ assert.deepStrictEqual((await Array.fromAsync(t.getHtml(session))).join(""), "<a href=\"https://example.com/tags/fediverse\" class=\"mention hashtag\" rel=\"tag\" target=\"_blank\">#<span>Fediverse</span></a>");
306
+ const tags = await Array.fromAsync(t.getTags(session));
307
+ assert.deepStrictEqual(tags.length, 1);
308
+ assert.ok(tags[0] instanceof Hashtag);
309
+ assert.deepStrictEqual(tags[0].name, "#fediverse");
310
+ assert.deepStrictEqual(tags[0].href, new URL("https://example.com/tags/fediverse"));
311
+ assert.deepStrictEqual(t.getCachedObjects(), []);
312
+ });
313
+ test("em()", async () => {
314
+ const session = bot.getSession("https://example.com");
315
+ const t = em("Hello, World");
316
+ assert.deepStrictEqual((await Array.fromAsync(t.getHtml(session))).join(""), "<em>Hello, World</em>");
317
+ assert.deepStrictEqual(await Array.fromAsync(t.getTags(session)), []);
318
+ assert.deepStrictEqual(t.getCachedObjects(), []);
319
+ });
320
+ test("strong()", async () => {
321
+ const session = bot.getSession("https://example.com");
322
+ const t = strong("Hello, World");
323
+ assert.deepStrictEqual((await Array.fromAsync(t.getHtml(session))).join(""), "<strong>Hello, World</strong>");
324
+ assert.deepStrictEqual(await Array.fromAsync(t.getTags(session)), []);
325
+ assert.deepStrictEqual(t.getCachedObjects(), []);
326
+ });
327
+ test("link()", async () => {
328
+ const session = bot.getSession("https://example.com");
329
+ const t = link(em("label"), "https://example.com/");
330
+ assert.deepStrictEqual((await Array.fromAsync(t.getHtml(session))).join(""), "<a href=\"https://example.com/\" target=\"_blank\"><em>label</em></a>");
331
+ assert.deepStrictEqual(await Array.fromAsync(t.getTags(session)), []);
332
+ assert.deepStrictEqual(t.getCachedObjects(), []);
333
+ const t2 = link("label", "https://example.com/");
334
+ assert.deepStrictEqual((await Array.fromAsync(t2.getHtml(session))).join(""), "<a href=\"https://example.com/\" target=\"_blank\">label</a>");
335
+ assert.deepStrictEqual(await Array.fromAsync(t2.getTags(session)), []);
336
+ assert.deepStrictEqual(t2.getCachedObjects(), []);
337
+ const t3 = link("https://example.com/");
338
+ assert.deepStrictEqual((await Array.fromAsync(t3.getHtml(session))).join(""), "<a href=\"https://example.com/\" target=\"_blank\">https://example.com/</a>");
339
+ assert.deepStrictEqual(await Array.fromAsync(t3.getTags(session)), []);
340
+ assert.deepStrictEqual(t3.getCachedObjects(), []);
341
+ const t4 = link(em("label"), "https://example.com/");
342
+ assert.deepStrictEqual((await Array.fromAsync(t4.getHtml(session))).join(""), "<a href=\"https://example.com/\" target=\"_blank\"><em>label</em></a>");
343
+ assert.deepStrictEqual(await Array.fromAsync(t4.getTags(session)), []);
344
+ assert.deepStrictEqual(t4.getCachedObjects(), []);
345
+ });
346
+ test("code()", async () => {
347
+ const session = bot.getSession("https://example.com");
348
+ const t = code("a + b");
349
+ assert.deepStrictEqual((await Array.fromAsync(t.getHtml(session))).join(""), "<code>a + b</code>");
350
+ assert.deepStrictEqual(await Array.fromAsync(t.getTags(session)), []);
351
+ assert.deepStrictEqual(t.getCachedObjects(), []);
352
+ });
353
+ test("markdown()", async () => {
354
+ const session = bot.getSession("https://example.com");
355
+ const md = `Here's a Markdown text.
356
+
357
+ - I can have a list.
358
+ - I can have a **bold** text.
359
+ - I can have an _italic_ text.
360
+ - I can mention @fedify@hollo.social.
361
+ - I can tag #Hashtag.`;
362
+ const t = markdown(md);
363
+ assert.deepStrictEqual((await Array.fromAsync(t.getHtml(session))).join(""), "<p>Here's a Markdown text.</p>\n<ul>\n<li>I can have a list.</li>\n<li>I can have a <strong>bold</strong> text.</li>\n<li>I can have an <em>italic</em> text.</li>\n<li>I can mention <a translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\" href=\"https://hollo.social/@fedify\"><span class=\"at\">@</span><span class=\"user\">fedify</span><span class=\"at\">@</span><span class=\"domain\">hollo.social</span></a>.</li>\n<li>I can tag <a class=\"mention hashtag\" rel=\"tag\" target=\"_blank\" href=\"https://example.com/tags/hashtag\">#<span>Hashtag</span></a>.</li>\n</ul>\n");
364
+ const tags = await Array.fromAsync(t.getTags(session));
365
+ assert.deepStrictEqual(tags.length, 2);
366
+ assert.ok(tags[0] instanceof Mention);
367
+ assert.deepStrictEqual(tags[0].name, "@fedify@hollo.social");
368
+ assert.deepStrictEqual(tags[0].href, new URL("https://hollo.social/@fedify"));
369
+ assert.ok(tags[1] instanceof Hashtag);
370
+ assert.deepStrictEqual(tags[1].name, "#hashtag");
371
+ assert.deepStrictEqual(tags[1].href, new URL("https://example.com/tags/hashtag"));
372
+ const cache = t.getCachedObjects();
373
+ assert.deepStrictEqual(cache.length, 1);
374
+ assert.ok(cache[0] instanceof Person);
375
+ assert.deepStrictEqual(cache[0].id, new URL("https://hollo.social/@fedify"));
376
+ const t2 = markdown("@fedify@hollo.social", { mentions: false });
377
+ assert.deepStrictEqual((await Array.fromAsync(t2.getHtml(session))).join(""), "<p>@fedify@hollo.social</p>\n");
378
+ assert.deepStrictEqual(await Array.fromAsync(t2.getTags(session)), []);
379
+ assert.deepStrictEqual(t2.getCachedObjects(), []);
380
+ const t3 = markdown("@bot@example.com");
381
+ assert.deepStrictEqual((await Array.fromAsync(t3.getHtml(session))).join(""), "<p><a translate=\"no\" class=\"h-card u-url mention\" target=\"_blank\" href=\"https://example.com/ap/actor/bot\"><span class=\"at\">@</span><span class=\"user\">bot</span><span class=\"at\">@</span><span class=\"domain\">example.com</span></a></p>\n");
382
+ const tags3 = await Array.fromAsync(t3.getTags(session));
383
+ assert.deepStrictEqual(tags3.length, 1);
384
+ assert.ok(tags3[0] instanceof Mention);
385
+ assert.deepStrictEqual(tags3[0].name, "@bot@example.com");
386
+ assert.deepStrictEqual(tags3[0].href, new URL("https://example.com/ap/actor/bot"));
387
+ const cache3 = t3.getCachedObjects();
388
+ assert.deepStrictEqual(cache3.length, 1);
389
+ assert.ok(cache3[0] instanceof Person);
390
+ assert.deepStrictEqual(cache3[0].id, new URL("https://example.com/ap/actor/bot"));
391
+ });
392
+ describe("customEmoji(), CustomEmojiText", () => {
393
+ const localBot = new BotImpl({
394
+ kv: new MemoryKvStore(),
395
+ username: "bot"
396
+ });
397
+ const session = localBot.getSession("https://example.com");
398
+ const emojiData = {
399
+ type: "image/png",
400
+ url: "https://example.com/emoji.png"
401
+ };
402
+ const deferredEmoji = localBot.addCustomEmoji("testEmoji", emojiData);
403
+ const emojiObject = deferredEmoji(session);
404
+ test("customEmoji() function", () => {
405
+ const emojiText = customEmoji(deferredEmoji);
406
+ assert.ok(emojiText instanceof CustomEmojiText);
407
+ assert.deepStrictEqual(emojiText.getEmoji(session), emojiObject);
408
+ const emojiTextDirect = customEmoji(emojiObject);
409
+ assert.ok(emojiTextDirect instanceof CustomEmojiText);
410
+ assert.deepStrictEqual(emojiTextDirect.getEmoji(session), emojiObject);
411
+ });
412
+ test("CustomEmojiText.getHtml()", async () => {
413
+ const emojiText = new CustomEmojiText(deferredEmoji);
414
+ assert.deepStrictEqual((await Array.fromAsync(emojiText.getHtml(session))).join(""), "​:testEmoji:​");
415
+ const emojiTextDirect = new CustomEmojiText(emojiObject);
416
+ assert.deepStrictEqual((await Array.fromAsync(emojiTextDirect.getHtml(session))).join(""), "​:testEmoji:​");
417
+ const emojiNoName = new Emoji({
418
+ id: new URL("https://example.com/ap/emoji/noname"),
419
+ icon: new Image({ url: new URL("https://example.com/noname.png") })
420
+ });
421
+ const emojiTextNoName = new CustomEmojiText(emojiNoName);
422
+ assert.deepStrictEqual((await Array.fromAsync(emojiTextNoName.getHtml(session))).join(""), "");
423
+ });
424
+ test("CustomEmojiText.getTags()", async () => {
425
+ const emojiText = new CustomEmojiText(deferredEmoji);
426
+ const tags = await Array.fromAsync(emojiText.getTags(session));
427
+ assert.deepStrictEqual(tags.length, 1);
428
+ assert.ok(tags[0] instanceof Emoji);
429
+ assert.deepStrictEqual(tags[0].id, emojiObject.id);
430
+ assert.deepStrictEqual(tags[0].name, emojiObject.name);
431
+ const emojiTextDirect = new CustomEmojiText(emojiObject);
432
+ const tagsDirect = await Array.fromAsync(emojiTextDirect.getTags(session));
433
+ assert.deepStrictEqual(tagsDirect.length, 1);
434
+ assert.deepStrictEqual(tagsDirect[0], emojiObject);
435
+ });
436
+ test("CustomEmojiText.getCachedObjects()", () => {
437
+ const emojiText = new CustomEmojiText(deferredEmoji);
438
+ assert.deepStrictEqual(emojiText.getCachedObjects(), []);
439
+ const emojiTextDirect = new CustomEmojiText(emojiObject);
440
+ assert.deepStrictEqual(emojiTextDirect.getCachedObjects(), []);
441
+ });
442
+ });
443
+ test("customEmoji()", async () => {
444
+ const localBot = new BotImpl({
445
+ kv: new MemoryKvStore(),
446
+ username: "bot"
447
+ });
448
+ const session = localBot.getSession("https://example.com");
449
+ const emojiData1 = {
450
+ type: "image/png",
451
+ url: "https://example.com/emoji1.png"
452
+ };
453
+ const emojiData2 = {
454
+ type: "image/gif",
455
+ file: "/path/to/emoji2.gif"
456
+ };
457
+ const deferredEmoji1 = localBot.addCustomEmoji("emoji1", emojiData1);
458
+ const deferredEmoji2 = localBot.addCustomEmoji("emoji2", emojiData2);
459
+ const emojiObject1 = deferredEmoji1(session);
460
+ const emojiObject2 = deferredEmoji2(session);
461
+ const t = text`Hello ${customEmoji(deferredEmoji1)} world ${customEmoji(emojiObject2)}!`;
462
+ assert.deepStrictEqual((await Array.fromAsync(t.getHtml(session))).join(""), "<p>Hello ​:emoji1:​ world ​:emoji2:​!</p>");
463
+ const tags = await Array.fromAsync(t.getTags(session));
464
+ assert.deepStrictEqual(tags.length, 2);
465
+ assert.ok(tags[0] instanceof Emoji);
466
+ assert.deepStrictEqual(tags[0].id, emojiObject1.id);
467
+ assert.ok(tags[1] instanceof Emoji);
468
+ assert.deepStrictEqual(tags[1].id, emojiObject2.id);
469
+ assert.deepStrictEqual(t.getCachedObjects(), []);
470
+ });
471
+
472
+ //#endregion
473
+ //# sourceMappingURL=text.test.js.map