fix: allow Renderer class in marked.use (#3118)

This commit is contained in:
Tony Brix 2023-12-08 00:17:36 -07:00 committed by GitHub
parent 44f40384b1
commit a28743391a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 67 additions and 5 deletions

View File

@ -142,9 +142,13 @@ export class Marked {
if (pack.renderer) { if (pack.renderer) {
const renderer = this.defaults.renderer || new _Renderer(this.defaults); const renderer = this.defaults.renderer || new _Renderer(this.defaults);
for (const prop in pack.renderer) { for (const prop in pack.renderer) {
if (!(prop in renderer) || prop === 'options') { if (!(prop in renderer)) {
throw new Error(`renderer '${prop}' does not exist`); throw new Error(`renderer '${prop}' does not exist`);
} }
if (prop === 'options') {
// ignore options property
continue;
}
const rendererProp = prop as Exclude<keyof _Renderer, 'options'>; const rendererProp = prop as Exclude<keyof _Renderer, 'options'>;
const rendererFunc = pack.renderer[rendererProp] as GenericRendererFunction; const rendererFunc = pack.renderer[rendererProp] as GenericRendererFunction;
const prevRenderer = renderer[rendererProp] as GenericRendererFunction; const prevRenderer = renderer[rendererProp] as GenericRendererFunction;
@ -162,9 +166,13 @@ export class Marked {
if (pack.tokenizer) { if (pack.tokenizer) {
const tokenizer = this.defaults.tokenizer || new _Tokenizer(this.defaults); const tokenizer = this.defaults.tokenizer || new _Tokenizer(this.defaults);
for (const prop in pack.tokenizer) { for (const prop in pack.tokenizer) {
if (!(prop in tokenizer) || ['options', 'rules', 'lexer'].includes(prop)) { if (!(prop in tokenizer)) {
throw new Error(`tokenizer '${prop}' does not exist`); throw new Error(`tokenizer '${prop}' does not exist`);
} }
if (['options', 'rules', 'lexer'].includes(prop)) {
// ignore options, rules, and lexer properties
continue;
}
const tokenizerProp = prop as Exclude<keyof _Tokenizer, 'options' | 'rules' | 'lexer'>; const tokenizerProp = prop as Exclude<keyof _Tokenizer, 'options' | 'rules' | 'lexer'>;
const tokenizerFunc = pack.tokenizer[tokenizerProp] as UnknownFunction; const tokenizerFunc = pack.tokenizer[tokenizerProp] as UnknownFunction;
const prevTokenizer = tokenizer[tokenizerProp] as UnknownFunction; const prevTokenizer = tokenizer[tokenizerProp] as UnknownFunction;
@ -185,9 +193,13 @@ export class Marked {
if (pack.hooks) { if (pack.hooks) {
const hooks = this.defaults.hooks || new _Hooks(); const hooks = this.defaults.hooks || new _Hooks();
for (const prop in pack.hooks) { for (const prop in pack.hooks) {
if (!(prop in hooks) || prop === 'options') { if (!(prop in hooks)) {
throw new Error(`hook '${prop}' does not exist`); throw new Error(`hook '${prop}' does not exist`);
} }
if (prop === 'options') {
// ignore options property
continue;
}
const hooksProp = prop as Exclude<keyof _Hooks, 'options'>; const hooksProp = prop as Exclude<keyof _Hooks, 'options'>;
const hooksFunc = pack.hooks[hooksProp] as UnknownFunction; const hooksFunc = pack.hooks[hooksProp] as UnknownFunction;
const prevHook = hooks[hooksProp] as UnknownFunction; const prevHook = hooks[hooksProp] as UnknownFunction;

View File

@ -1,5 +1,5 @@
import { marked, Marked, Renderer } from '../../lib/marked.esm.js'; import { marked, Marked, Renderer, Tokenizer, Hooks } from '../../lib/marked.esm.js';
import { describe, it } from 'node:test'; import { describe, it, beforeEach } from 'node:test';
import assert from 'node:assert'; import assert from 'node:assert';
describe('Marked', () => { describe('Marked', () => {
@ -89,4 +89,54 @@ describe('Marked', () => {
assert.strictEqual(html, 'test'); assert.strictEqual(html, 'test');
}); });
describe('use class objects', () => {
beforeEach(() => {
marked.setOptions(marked.getDefaults());
});
it('allow new Renderer()', () => {
const marked1 = new Marked();
const newRenderer = new Renderer();
newRenderer.heading = () => 'im marked renderer';
marked1.use({ renderer: newRenderer });
marked.use({ renderer: newRenderer });
assert.strictEqual(marked1.parse('# header'), 'im marked renderer');
assert.strictEqual(marked.parse('# header'), 'im marked renderer');
});
it('allow new Tokenizer()', () => {
const marked1 = new Marked();
const newTokenizer = new Tokenizer();
newTokenizer.heading = function(src) {
return {
type: 'heading',
raw: src,
depth: 1,
text: 'im marked tokenizer',
tokens: this.lexer.inlineTokens('im marked tokenizer')
};
};
marked1.use({ tokenizer: newTokenizer });
marked.use({ tokenizer: newTokenizer });
assert.strictEqual(marked1.parse('# header'), '<h1>im marked tokenizer</h1>\n');
assert.strictEqual(marked.parse('# header'), '<h1>im marked tokenizer</h1>\n');
});
it('allow new Hooks()', () => {
const marked1 = new Marked();
const newHooks = new Hooks();
newHooks.preprocess = () => 'im marked hooks';
marked1.use({ hooks: newHooks });
marked.use({ hooks: newHooks });
assert.strictEqual(marked1.parse('# header'), '<p>im marked hooks</p>\n');
assert.strictEqual(marked.parse('# header'), '<p>im marked hooks</p>\n');
});
});
}); });