@mtcute/markdown-parser 0.16.7 → 0.17.0

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.
package/esm/index.js DELETED
@@ -1,403 +0,0 @@
1
- import Long from 'long';
2
- const MENTION_REGEX = /^tg:\/\/user\?id=(\d+)(?:&hash=(-?[0-9a-fA-F]+)(?:&|$)|&|$)/;
3
- const EMOJI_REGEX = /^tg:\/\/emoji\?id=(-?\d+)/;
4
- const TAG_BOLD = '**';
5
- const TAG_ITALIC = '__';
6
- const TAG_UNDERLINE = '--';
7
- const TAG_STRIKE = '~~';
8
- const TAG_SPOILER = '||';
9
- const TAG_CODE = '`';
10
- const TAG_PRE = '```';
11
- const TO_BE_ESCAPED = /[*_\-~`[\\\]|]/g;
12
- /**
13
- * Escape a string to be safely used in Markdown.
14
- *
15
- * > **Note**: this function is in most cases not needed, as `md` function
16
- * > handles all `string`s passed to it automatically as plain text.
17
- */
18
- function escape(str) {
19
- return str.replace(TO_BE_ESCAPED, s => `\\${s}`);
20
- }
21
- /**
22
- * Add Markdown formatting to the text given the plain text and entities contained in it.
23
- */
24
- function unparse(input) {
25
- if (typeof input === 'string')
26
- return escape(input);
27
- let text = input.text;
28
- const entities = input.entities ?? [];
29
- // keep track of positions of inserted escape symbols
30
- const escaped = [];
31
- text = text.replace(TO_BE_ESCAPED, (s, pos) => {
32
- escaped.push(pos);
33
- return `\\${s}`;
34
- });
35
- const hasEscaped = escaped.length > 0;
36
- const insert = [];
37
- for (const entity of entities) {
38
- const type = entity._;
39
- let start = entity.offset;
40
- let end = start + entity.length;
41
- if (start > text.length)
42
- continue;
43
- if (start < 0)
44
- start = 0;
45
- if (end > text.length)
46
- end = text.length;
47
- if (hasEscaped) {
48
- // determine number of escape chars since the beginning of the string
49
- let escapedPos = 0;
50
- while (escapedPos < escaped.length && escaped[escapedPos] < start) {
51
- escapedPos += 1;
52
- }
53
- start += escapedPos;
54
- while (escapedPos < escaped.length && escaped[escapedPos] <= end) {
55
- escapedPos += 1;
56
- }
57
- end += escapedPos;
58
- }
59
- let startTag;
60
- let endTag;
61
- switch (type) {
62
- case 'messageEntityBold':
63
- startTag = endTag = TAG_BOLD;
64
- break;
65
- case 'messageEntityItalic':
66
- startTag = endTag = TAG_ITALIC;
67
- break;
68
- case 'messageEntityUnderline':
69
- startTag = endTag = TAG_UNDERLINE;
70
- break;
71
- case 'messageEntityStrike':
72
- startTag = endTag = TAG_STRIKE;
73
- break;
74
- case 'messageEntitySpoiler':
75
- startTag = endTag = TAG_SPOILER;
76
- break;
77
- case 'messageEntityCode':
78
- startTag = endTag = TAG_CODE;
79
- break;
80
- case 'messageEntityPre':
81
- startTag = TAG_PRE;
82
- if (entity.language) {
83
- startTag += entity.language;
84
- }
85
- startTag += '\n';
86
- endTag = `\n${TAG_PRE}`;
87
- break;
88
- case 'messageEntityTextUrl':
89
- startTag = '[';
90
- endTag = `](${entity.url})`;
91
- break;
92
- case 'messageEntityMentionName':
93
- startTag = '[';
94
- endTag = `](tg://user?id=${entity.userId})`;
95
- break;
96
- case 'messageEntityCustomEmoji':
97
- startTag = '[';
98
- endTag = `](tg://emoji?id=${entity.documentId.toString()})`;
99
- break;
100
- default:
101
- continue;
102
- }
103
- insert.push([start, startTag]);
104
- insert.push([end, endTag]);
105
- }
106
- // sort by offset desc
107
- insert.sort((a, b) => b[0] - a[0]);
108
- for (const [offset, tag] of insert) {
109
- text = text.substr(0, offset) + tag + text.substr(offset);
110
- }
111
- return text;
112
- }
113
- function parse(strings, ...sub) {
114
- const entities = [];
115
- let result = '';
116
- const stacks = {};
117
- let insideCode = false;
118
- let insidePre = false;
119
- let insideLink = false;
120
- let insideLinkUrl = false;
121
- let pendingLinkUrl = '';
122
- function feed(text) {
123
- const len = text.length;
124
- let pos = 0;
125
- while (pos < len) {
126
- const c = text[pos];
127
- if (c === '\\') {
128
- if (insideLinkUrl) {
129
- pendingLinkUrl += text[pos + 1];
130
- }
131
- else {
132
- result += text[pos + 1];
133
- }
134
- pos += 2;
135
- continue;
136
- }
137
- if (insideCode) {
138
- if (c === '`') {
139
- // we can be certain that we're inside code
140
- const ent = stacks.code.pop();
141
- ent.length = result.length - ent.offset;
142
- entities.push(ent);
143
- insideCode = false;
144
- pos += 1;
145
- }
146
- else {
147
- pos += 1;
148
- result += c;
149
- }
150
- continue;
151
- }
152
- if (insidePre) {
153
- if (c === '`' || (c === '\n' && text[pos + 1] === '`')) {
154
- if (c === '\n')
155
- pos += 1;
156
- if (text[pos + 1] === '`' && text[pos + 2] === '`') {
157
- // we can be certain that we're inside pre
158
- const ent = stacks.pre.pop();
159
- ent.length = result.length - ent.offset;
160
- entities.push(ent);
161
- insidePre = false;
162
- pos += 3;
163
- continue;
164
- // closed with single or double backtick
165
- // i.e. not closed actually! this is totally valid md:
166
- // ```javascript
167
- // const a = ``;
168
- // ```
169
- // compensate that `pos` change we made earliers
170
- }
171
- else if (c === '\n') {
172
- pos -= 1;
173
- }
174
- }
175
- pos += 1;
176
- result += c;
177
- continue;
178
- }
179
- if (insideLink && c === ']') {
180
- // we can be certain that we're inside link
181
- const ent = stacks.link.pop();
182
- if (text[pos + 1] !== '(') {
183
- // [link text]
184
- // ignore this, and add opening [
185
- result = `${result.substr(0, ent.offset)}[${result.substr(ent.offset)}]`;
186
- pos += 1;
187
- insideLink = false;
188
- continue;
189
- }
190
- pos += 2;
191
- insideLink = false;
192
- insideLinkUrl = true;
193
- stacks.link.push(ent);
194
- continue;
195
- }
196
- if (insideLinkUrl) {
197
- pos += 1;
198
- if (c !== ')') {
199
- // not ended yet
200
- pendingLinkUrl += c;
201
- continue;
202
- }
203
- const ent = stacks.link.pop();
204
- let url = pendingLinkUrl;
205
- pendingLinkUrl = '';
206
- insideLinkUrl = false;
207
- if (!url.length)
208
- continue;
209
- ent.length = result.length - ent.offset;
210
- let m = url.match(MENTION_REGEX);
211
- if (m) {
212
- const userId = Number.parseInt(m[1]);
213
- const accessHash = m[2];
214
- if (accessHash) {
215
- ent._ = 'inputMessageEntityMentionName';
216
- ent.userId = {
217
- _: 'inputUser',
218
- userId,
219
- accessHash: Long.fromString(accessHash, false, 16),
220
- };
221
- }
222
- else {
223
- ent._ = 'messageEntityMentionName';
224
- ent.userId = userId;
225
- }
226
- }
227
- else if ((m = EMOJI_REGEX.exec(url))) {
228
- ent._ = 'messageEntityCustomEmoji';
229
- ent.documentId = Long.fromString(m[1]);
230
- }
231
- else {
232
- if (url.match(/^\/\//))
233
- url = `http:${url}`;
234
- ent._ = 'messageEntityTextUrl';
235
- ent.url = url;
236
- }
237
- entities.push(ent);
238
- continue;
239
- }
240
- if (c === '[' && !insideLink) {
241
- pos += 1;
242
- insideLink = true;
243
- if (!('link' in stacks))
244
- stacks.link = [];
245
- // eslint-disable-next-line ts/no-unsafe-argument
246
- stacks.link.push({
247
- offset: result.length,
248
- length: 0,
249
- }); // other fields are added after the second part
250
- continue;
251
- }
252
- if (c === '`') {
253
- const isPre = text[pos + 1] === '`' && text[pos + 2] === '`';
254
- if (isPre) {
255
- pos += 3;
256
- let language = '';
257
- while (pos < text.length && text[pos] !== '\n') {
258
- language += text[pos++];
259
- }
260
- // newline
261
- pos += 1;
262
- if (pos > text.length) {
263
- throw new Error('Malformed PRE entity, expected LF after ```');
264
- }
265
- if (!('pre' in stacks))
266
- stacks.pre = [];
267
- stacks.pre.push({
268
- _: 'messageEntityPre',
269
- offset: result.length,
270
- length: 0,
271
- language,
272
- });
273
- insidePre = true;
274
- }
275
- else {
276
- pos += 1;
277
- if (!('code' in stacks))
278
- stacks.code = [];
279
- stacks.code.push({
280
- _: 'messageEntityCode',
281
- offset: result.length,
282
- length: 0,
283
- });
284
- insideCode = true;
285
- }
286
- continue;
287
- }
288
- if (c === text[pos + 1]) {
289
- // maybe (?) start or end of an entity
290
- let type = null;
291
- switch (c) {
292
- case '_':
293
- type = 'Italic';
294
- break;
295
- case '*':
296
- type = 'Bold';
297
- break;
298
- case '-':
299
- type = 'Underline';
300
- break;
301
- case '~':
302
- type = 'Strike';
303
- break;
304
- case '|':
305
- type = 'Spoiler';
306
- break;
307
- }
308
- if (type) {
309
- if (!(type in stacks))
310
- stacks[type] = [];
311
- const isBegin = stacks[type].length === 0;
312
- if (isBegin) {
313
- stacks[type].push({
314
- _: `messageEntity${type}`,
315
- offset: result.length,
316
- length: 0,
317
- });
318
- }
319
- else {
320
- // valid because isBegin is false
321
- const ent = stacks[type].pop();
322
- ent.length = result.length - ent.offset;
323
- entities.push(ent);
324
- }
325
- pos += 2;
326
- continue;
327
- }
328
- }
329
- if (c === '\n') {
330
- if (pos !== 0) {
331
- result += '\n';
332
- }
333
- const nonWhitespace = text.slice(pos + 1).search(/[^ \t]/);
334
- if (nonWhitespace !== -1) {
335
- pos += nonWhitespace + 1;
336
- }
337
- else {
338
- pos = len;
339
- result = result.trimEnd();
340
- }
341
- continue;
342
- }
343
- // nothing matched => normal character
344
- result += c;
345
- pos += 1;
346
- }
347
- }
348
- if (typeof strings === 'string')
349
- strings = [strings];
350
- sub.forEach((it, idx) => {
351
- feed(strings[idx]);
352
- if (typeof it === 'boolean' || !it)
353
- return;
354
- if (insideLinkUrl) {
355
- if (typeof it === 'string' || typeof it === 'number') {
356
- pendingLinkUrl += it;
357
- }
358
- else if (Long.isLong(it)) {
359
- pendingLinkUrl += it.toString(10);
360
- }
361
- else {
362
- // ignore the entity, only use text
363
- pendingLinkUrl += it.text;
364
- }
365
- return;
366
- }
367
- if (typeof it === 'string' || typeof it === 'number') {
368
- result += it;
369
- }
370
- else if (Long.isLong(it)) {
371
- result += it.toString(10);
372
- }
373
- else {
374
- // TextWithEntities or MessageEntity
375
- const text = it.text;
376
- const innerEntities = 'raw' in it ? [it.raw] : it.entities;
377
- const baseOffset = result.length;
378
- result += text;
379
- if (innerEntities) {
380
- for (const ent of innerEntities) {
381
- entities.push({ ...ent, offset: ent.offset + baseOffset });
382
- }
383
- }
384
- }
385
- });
386
- feed(strings[strings.length - 1]);
387
- for (const [name, stack] of Object.entries(stacks)) {
388
- if (stack.length) {
389
- throw new Error(`Unterminated ${name} entity`);
390
- }
391
- }
392
- return {
393
- text: result,
394
- entities,
395
- };
396
- }
397
- // typedoc doesn't support this yet, so we'll have to do it manually
398
- // https://github.com/TypeStrong/typedoc/issues/2436
399
- export const md = Object.assign(parse, {
400
- escape,
401
- unparse,
402
- });
403
- //# sourceMappingURL=index.js.map
package/esm/index.js.map DELETED
@@ -1 +0,0 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import Long from 'long'\nimport type { InputText, MessageEntity, TextWithEntities, tl } from '@mtcute/core'\n\nconst MENTION_REGEX = /^tg:\\/\\/user\\?id=(\\d+)(?:&hash=(-?[0-9a-fA-F]+)(?:&|$)|&|$)/\nconst EMOJI_REGEX = /^tg:\\/\\/emoji\\?id=(-?\\d+)/\n\nconst TAG_BOLD = '**'\nconst TAG_ITALIC = '__'\nconst TAG_UNDERLINE = '--'\nconst TAG_STRIKE = '~~'\nconst TAG_SPOILER = '||'\nconst TAG_CODE = '`'\nconst TAG_PRE = '```'\n\nconst TO_BE_ESCAPED = /[*_\\-~`[\\\\\\]|]/g\n\n/**\n * Escape a string to be safely used in Markdown.\n *\n * > **Note**: this function is in most cases not needed, as `md` function\n * > handles all `string`s passed to it automatically as plain text.\n */\nfunction escape(str: string): string {\n return str.replace(TO_BE_ESCAPED, s => `\\\\${s}`)\n}\n\n/**\n * Add Markdown formatting to the text given the plain text and entities contained in it.\n */\nfunction unparse(input: InputText): string {\n if (typeof input === 'string') return escape(input)\n\n let text = input.text\n const entities = input.entities ?? []\n\n // keep track of positions of inserted escape symbols\n const escaped: number[] = []\n text = text.replace(TO_BE_ESCAPED, (s, pos: number) => {\n escaped.push(pos)\n\n return `\\\\${s}`\n })\n const hasEscaped = escaped.length > 0\n\n type InsertLater = [number, string]\n const insert: InsertLater[] = []\n\n for (const entity of entities) {\n const type = entity._\n\n let start = entity.offset\n let end = start + entity.length\n\n if (start > text.length) continue\n if (start < 0) start = 0\n if (end > text.length) end = text.length\n\n if (hasEscaped) {\n // determine number of escape chars since the beginning of the string\n let escapedPos = 0\n\n while (escapedPos < escaped.length && escaped[escapedPos] < start) {\n escapedPos += 1\n }\n start += escapedPos\n\n while (escapedPos < escaped.length && escaped[escapedPos] <= end) {\n escapedPos += 1\n }\n end += escapedPos\n }\n\n let startTag\n let endTag: string\n\n switch (type) {\n case 'messageEntityBold':\n startTag = endTag = TAG_BOLD\n break\n case 'messageEntityItalic':\n startTag = endTag = TAG_ITALIC\n break\n case 'messageEntityUnderline':\n startTag = endTag = TAG_UNDERLINE\n break\n case 'messageEntityStrike':\n startTag = endTag = TAG_STRIKE\n break\n case 'messageEntitySpoiler':\n startTag = endTag = TAG_SPOILER\n break\n case 'messageEntityCode':\n startTag = endTag = TAG_CODE\n break\n case 'messageEntityPre':\n startTag = TAG_PRE\n\n if (entity.language) {\n startTag += entity.language\n }\n\n startTag += '\\n'\n endTag = `\\n${TAG_PRE}`\n break\n case 'messageEntityTextUrl':\n startTag = '['\n endTag = `](${entity.url})`\n break\n case 'messageEntityMentionName':\n startTag = '['\n endTag = `](tg://user?id=${entity.userId})`\n break\n case 'messageEntityCustomEmoji':\n startTag = '['\n endTag = `](tg://emoji?id=${entity.documentId.toString()})`\n break\n default:\n continue\n }\n\n insert.push([start, startTag])\n insert.push([end, endTag])\n }\n\n // sort by offset desc\n insert.sort((a, b) => b[0] - a[0])\n\n for (const [offset, tag] of insert) {\n text = text.substr(0, offset) + tag + text.substr(offset)\n }\n\n return text\n}\n\nfunction parse(\n strings: TemplateStringsArray | string,\n ...sub: (InputText | MessageEntity | Long | boolean | number | undefined | null)[]\n): TextWithEntities {\n const entities: tl.TypeMessageEntity[] = []\n let result = ''\n\n const stacks: Record<string, tl.Mutable<tl.TypeMessageEntity>[]> = {}\n\n let insideCode = false\n let insidePre = false\n let insideLink = false\n\n let insideLinkUrl = false\n let pendingLinkUrl = ''\n\n function feed(text: string) {\n const len = text.length\n let pos = 0\n\n while (pos < len) {\n const c = text[pos]\n\n if (c === '\\\\') {\n if (insideLinkUrl) {\n pendingLinkUrl += text[pos + 1]\n } else {\n result += text[pos + 1]\n }\n pos += 2\n continue\n }\n\n if (insideCode) {\n if (c === '`') {\n // we can be certain that we're inside code\n\n const ent = stacks.code.pop()!\n ent.length = result.length - ent.offset\n entities.push(ent)\n insideCode = false\n pos += 1\n } else {\n pos += 1\n result += c\n }\n continue\n }\n\n if (insidePre) {\n if (c === '`' || (c === '\\n' && text[pos + 1] === '`')) {\n if (c === '\\n') pos += 1\n\n if (text[pos + 1] === '`' && text[pos + 2] === '`') {\n // we can be certain that we're inside pre\n\n const ent = stacks.pre.pop()!\n ent.length = result.length - ent.offset\n entities.push(ent)\n insidePre = false\n pos += 3\n continue\n\n // closed with single or double backtick\n // i.e. not closed actually! this is totally valid md:\n // ```javascript\n // const a = ``;\n // ```\n // compensate that `pos` change we made earliers\n } else if (c === '\\n') {\n pos -= 1\n }\n }\n\n pos += 1\n result += c\n continue\n }\n\n if (insideLink && c === ']') {\n // we can be certain that we're inside link\n\n const ent = stacks.link.pop()!\n\n if (text[pos + 1] !== '(') {\n // [link text]\n // ignore this, and add opening [\n result = `${result.substr(0, ent.offset)}[${result.substr(ent.offset)}]`\n pos += 1\n insideLink = false\n continue\n }\n\n pos += 2\n insideLink = false\n insideLinkUrl = true\n stacks.link.push(ent)\n continue\n }\n\n if (insideLinkUrl) {\n pos += 1\n\n if (c !== ')') {\n // not ended yet\n pendingLinkUrl += c\n continue\n }\n\n const ent = stacks.link.pop()!\n\n let url = pendingLinkUrl\n pendingLinkUrl = ''\n insideLinkUrl = false\n\n if (!url.length) continue\n\n ent.length = result.length - ent.offset\n\n let m = url.match(MENTION_REGEX)\n\n if (m) {\n const userId = Number.parseInt(m[1])\n const accessHash = m[2]\n\n if (accessHash) {\n (ent as tl.Mutable<tl.RawInputMessageEntityMentionName>)._ = 'inputMessageEntityMentionName'\n ;(ent as tl.Mutable<tl.RawInputMessageEntityMentionName>).userId = {\n _: 'inputUser',\n userId,\n accessHash: Long.fromString(accessHash, false, 16),\n }\n } else {\n (ent as tl.Mutable<tl.RawMessageEntityMentionName>)._ = 'messageEntityMentionName'\n ;(ent as tl.Mutable<tl.RawMessageEntityMentionName>).userId = userId\n }\n } else if ((m = EMOJI_REGEX.exec(url))) {\n (ent as tl.Mutable<tl.RawMessageEntityCustomEmoji>)._ = 'messageEntityCustomEmoji'\n ;(ent as tl.Mutable<tl.RawMessageEntityCustomEmoji>).documentId = Long.fromString(m[1])\n } else {\n if (url.match(/^\\/\\//)) url = `http:${url}`\n ;(ent as tl.Mutable<tl.RawMessageEntityTextUrl>)._ = 'messageEntityTextUrl'\n ;(ent as tl.Mutable<tl.RawMessageEntityTextUrl>).url = url\n }\n entities.push(ent)\n\n continue\n }\n\n if (c === '[' && !insideLink) {\n pos += 1\n insideLink = true\n if (!('link' in stacks)) stacks.link = []\n // eslint-disable-next-line ts/no-unsafe-argument\n stacks.link.push({\n offset: result.length,\n length: 0,\n } as any) // other fields are added after the second part\n continue\n }\n\n if (c === '`') {\n const isPre = text[pos + 1] === '`' && text[pos + 2] === '`'\n\n if (isPre) {\n pos += 3\n let language = ''\n\n while (pos < text.length && text[pos] !== '\\n') {\n language += text[pos++]\n }\n\n // newline\n pos += 1\n\n if (pos > text.length) {\n throw new Error('Malformed PRE entity, expected LF after ```')\n }\n\n if (!('pre' in stacks)) stacks.pre = []\n stacks.pre.push({\n _: 'messageEntityPre',\n offset: result.length,\n length: 0,\n language,\n })\n insidePre = true\n } else {\n pos += 1\n if (!('code' in stacks)) stacks.code = []\n stacks.code.push({\n _: 'messageEntityCode',\n offset: result.length,\n length: 0,\n })\n insideCode = true\n }\n\n continue\n }\n\n if (c === text[pos + 1]) {\n // maybe (?) start or end of an entity\n let type: 'Italic' | 'Bold' | 'Underline' | 'Strike' | 'Spoiler' | null = null\n\n switch (c) {\n case '_':\n type = 'Italic'\n break\n case '*':\n type = 'Bold'\n break\n case '-':\n type = 'Underline'\n break\n case '~':\n type = 'Strike'\n break\n case '|':\n type = 'Spoiler'\n break\n }\n\n if (type) {\n if (!(type in stacks)) stacks[type] = []\n const isBegin = stacks[type].length === 0\n\n if (isBegin) {\n stacks[type].push({\n _: `messageEntity${type}`,\n offset: result.length,\n length: 0,\n })\n } else {\n // valid because isBegin is false\n\n const ent = stacks[type].pop()!\n ent.length = result.length - ent.offset\n entities.push(ent)\n }\n\n pos += 2\n continue\n }\n }\n\n if (c === '\\n') {\n if (pos !== 0) {\n result += '\\n'\n }\n\n const nonWhitespace = text.slice(pos + 1).search(/[^ \\t]/)\n\n if (nonWhitespace !== -1) {\n pos += nonWhitespace + 1\n } else {\n pos = len\n result = result.trimEnd()\n }\n continue\n }\n\n // nothing matched => normal character\n result += c\n pos += 1\n }\n }\n\n if (typeof strings === 'string') strings = [strings] as unknown as TemplateStringsArray\n\n sub.forEach((it, idx) => {\n feed(strings[idx])\n\n if (typeof it === 'boolean' || !it) return\n\n if (insideLinkUrl) {\n if (typeof it === 'string' || typeof it === 'number') {\n pendingLinkUrl += it\n } else if (Long.isLong(it)) {\n pendingLinkUrl += it.toString(10)\n } else {\n // ignore the entity, only use text\n pendingLinkUrl += it.text\n }\n\n return\n }\n\n if (typeof it === 'string' || typeof it === 'number') {\n result += it\n } else if (Long.isLong(it)) {\n result += it.toString(10)\n } else {\n // TextWithEntities or MessageEntity\n const text = it.text\n const innerEntities = 'raw' in it ? [it.raw] : it.entities\n\n const baseOffset = result.length\n result += text\n\n if (innerEntities) {\n for (const ent of innerEntities) {\n entities.push({ ...ent, offset: ent.offset + baseOffset })\n }\n }\n }\n })\n\n feed(strings[strings.length - 1])\n\n for (const [name, stack] of Object.entries(stacks)) {\n if (stack.length) {\n throw new Error(`Unterminated ${name} entity`)\n }\n }\n\n return {\n text: result,\n entities,\n }\n}\n\n// typedoc doesn't support this yet, so we'll have to do it manually\n// https://github.com/TypeStrong/typedoc/issues/2436\n\nexport const md: {\n /**\n * Tagged template based Markdown-to-entities parser function\n *\n * Additionally, `md` function has two static methods:\n * - `md.escape` - escape a string to be safely used in Markdown\n * (should not be needed in most cases, as `md` function itself handles all `string`s\n * passed to it automatically as plain text)\n * - `md.unparse` - add Markdown formatting to the text given the plain text and entities contained in it\n *\n * @example\n * ```typescript\n * const text = md`**${user.displayName}**`\n * ```\n */\n (\n strings: TemplateStringsArray,\n ...sub: (InputText | MessageEntity | Long | boolean | number | undefined | null)[]\n ): TextWithEntities\n /**\n * A variant taking a plain JS string as input\n * and parsing it.\n *\n * Useful for cases when you already have a string\n * (e.g. from some server) and want to parse it.\n *\n * @example\n * ```typescript\n * const string = '**hello**'\n * const text = md(string)\n * ```\n */\n (string: string): TextWithEntities\n escape: typeof escape\n unparse: typeof unparse\n} = Object.assign(parse, {\n escape,\n unparse,\n})\n"]}