From 20cb7f5fdb145b13967d61fe78363c6db5a287f8 Mon Sep 17 00:00:00 2001 From: Daniel Kuznetsov Date: Tue, 10 Dec 2024 18:25:19 +0300 Subject: [PATCH 1/2] feat: implement ways to interact with collect via transform API --- src/transform/index.ts | 47 +++++++++++++++++++++++++++++++++++----- src/transform/typings.ts | 25 +++++++++++++++++++-- 2 files changed, 64 insertions(+), 8 deletions(-) diff --git a/src/transform/index.ts b/src/transform/index.ts index 3ca9dfa3..6588de75 100644 --- a/src/transform/index.ts +++ b/src/transform/index.ts @@ -1,9 +1,9 @@ -import type {EnvType, OptionsType, OutputType} from './typings'; +import type {EnvType, OptionsType, OutputType, RootCollectorOptions} from './typings'; import {bold} from 'chalk'; import {log} from './log'; -import liquidSnippet from './liquid'; +import liquidDocument from './liquid'; import initMarkdownIt from './md'; function applyLiquid(input: string, options: OptionsType) { @@ -17,7 +17,7 @@ function applyLiquid(input: string, options: OptionsType) { return disableLiquid || isLiquided ? input - : liquidSnippet(input, vars, path, {conditionsInCode}); + : liquidDocument(input, vars, path, {conditionsInCode}); } function handleError(error: unknown, path?: string): never { @@ -33,8 +33,14 @@ function emitResult(html: string, env: EnvType): OutputType { }; } +type TransformFunction = { + (originInput: string, options?: OptionsType): OutputType; + // eslint-disable-next-line @typescript-eslint/no-explicit-any + collect: (input: string, options: RootCollectorOptions) => string; +}; + // eslint-disable-next-line consistent-return -function transform(originInput: string, options: OptionsType = {}) { +const transform: TransformFunction = (originInput: string, options: OptionsType = {}) => { const input = applyLiquid(originInput, options); const {parse, compile, env} = initMarkdownIt(options); @@ -43,11 +49,40 @@ function transform(originInput: string, options: OptionsType = {}) { } catch (error) { handleError(error, options.path); } -} +}; + +transform.collect = ( + input: string, + // eslint-disable-next-line @typescript-eslint/no-explicit-any + {mdItInitOptions, pluginCollectOptions, parserPluginsOverride}: RootCollectorOptions, +) => { + const maybeLiquidedInput = applyLiquid(input, mdItInitOptions); + const {parse} = initMarkdownIt({ + ...mdItInitOptions, + plugins: parserPluginsOverride ?? mdItInitOptions.plugins, + }); + + const plugins = mdItInitOptions.plugins ?? []; + + try { + const tokenStream = parse(maybeLiquidedInput); + + return plugins.reduce((collected, plugin) => { + const collectOutput = plugin.collect?.(collected, { + ...pluginCollectOptions, + tokenStream, + }); + + return collectOutput ?? collected; + }, input); + } catch (error) { + handleError(error, mdItInitOptions.path); + } +}; export = transform; -// eslint-disable-next-line @typescript-eslint/no-namespace, no-redeclare -- backward compatibility +// eslint-disable-next-line @typescript-eslint/no-namespace, @typescript-eslint/no-redeclare -- backward compatibility namespace transform { export type Options = OptionsType; export type Output = OutputType; diff --git a/src/transform/typings.ts b/src/transform/typings.ts index 0b67a801..cd5210d1 100644 --- a/src/transform/typings.ts +++ b/src/transform/typings.ts @@ -1,5 +1,5 @@ import {LanguageFn} from 'highlight.js'; -import DefaultMarkdownIt from 'markdown-it'; +import DefaultMarkdownIt, {Token} from 'markdown-it'; import DefaultStateCore from 'markdown-it/lib/rules_core/state_core'; import {SanitizeOptions} from './sanitize'; @@ -47,7 +47,7 @@ export interface OptionsType { sanitizeOptions?: SanitizeOptions; needFlatListHeadings?: boolean; // eslint-disable-next-line @typescript-eslint/no-explicit-any - plugins?: MarkdownItPluginCb[]; + plugins?: ExtendedPluginWithCollect[]; preprocessors?: MarkdownItPreprocessorCb[]; // Preprocessors should modify the input before passing it to MD highlightLangs?: HighlightLangMap; disableRules?: string[]; @@ -96,6 +96,27 @@ export type MarkdownItPluginCb = { (md: MarkdownIt, opts: T & MarkdownItPluginOpts): void; }; +export type IntrinsicCollectOptions = { + tokenStream: Token[]; +}; + +export type ExtendedPluginWithCollect< + PluginRegularOptions extends {} = {}, + PluginCollectOptions = {}, +> = MarkdownItPluginCb & { + collect?: ( + input: string, + options: PluginCollectOptions & IntrinsicCollectOptions, + ) => string | void; +}; + +export type RootCollectorOptions = { + mdItInitOptions: OptionsType; + pluginCollectOptions: PluginCollectOptions; + // eslint-disable-next-line @typescript-eslint/no-explicit-any + parserPluginsOverride?: ExtendedPluginWithCollect[]; +}; + export type MarkdownItPreprocessorCb = { (input: string, opts: T & Partial, md?: MarkdownIt): string; }; From 24f2bd5859d0d450a25fcefbd114d305ca3c1655 Mon Sep 17 00:00:00 2001 From: Daniel Kuznetsov Date: Sat, 28 Dec 2024 18:59:50 +0300 Subject: [PATCH 2/2] feat: implement tests for collect.transform --- src/transform/index.ts | 2 +- test/collect.test.ts | 218 +++++++++++++++++++++++++++++++++++++++++ 2 files changed, 219 insertions(+), 1 deletion(-) create mode 100644 test/collect.test.ts diff --git a/src/transform/index.ts b/src/transform/index.ts index 6588de75..427fd002 100644 --- a/src/transform/index.ts +++ b/src/transform/index.ts @@ -74,7 +74,7 @@ transform.collect = ( }); return collectOutput ?? collected; - }, input); + }, maybeLiquidedInput); } catch (error) { handleError(error, mdItInitOptions.path); } diff --git a/test/collect.test.ts b/test/collect.test.ts new file mode 100644 index 00000000..87f10ad8 --- /dev/null +++ b/test/collect.test.ts @@ -0,0 +1,218 @@ +import {Token} from 'markdown-it'; + +import transform from '../src/transform'; +import {ExtendedPluginWithCollect, IntrinsicCollectOptions} from '../src/transform/typings'; + +describe('transform.collect', () => { + it(`applies liquid if it wasn't disabled`, () => { + const input = `### Docs for {{ product }}`; + const vars = {product: 'Diplodoc Platform'}; + + const pluginWithCollect: ExtendedPluginWithCollect = jest.fn(); + pluginWithCollect.collect = jest.fn(); + + const result = transform.collect(input, { + mdItInitOptions: { + plugins: [pluginWithCollect], + vars: vars, + }, + pluginCollectOptions: {}, + }); + + expect(pluginWithCollect.collect).toHaveBeenCalledTimes(1); + expect(pluginWithCollect.collect).toHaveBeenCalledWith( + '### Docs for Diplodoc Platform', + expect.objectContaining({ + tokenStream: expect.arrayContaining([ + expect.objectContaining({ + children: expect.arrayContaining([ + expect.objectContaining({content: 'Docs for Diplodoc Platform'}), + ]), + }), + ]), + }), + ); + + expect(result).toBe('### Docs for Diplodoc Platform'); + }); + + it(`doesn't apply liquid if it was explicitly disabled by passing isLiquided = true`, () => { + const input = `### Docs for {{ product }}`; + const vars = {product: 'Diplodoc Platform'}; + + const pluginWithCollect: ExtendedPluginWithCollect = jest.fn(); + pluginWithCollect.collect = jest.fn(); + + const result = transform.collect(input, { + mdItInitOptions: { + plugins: [pluginWithCollect], + vars: vars, + isLiquided: true, + }, + pluginCollectOptions: {}, + }); + + expect(pluginWithCollect.collect).toHaveBeenCalledTimes(1); + expect(pluginWithCollect.collect).toHaveBeenCalledWith( + '### Docs for {{ product }}', + expect.objectContaining({ + tokenStream: expect.arrayContaining([ + expect.objectContaining({ + children: expect.arrayContaining([ + expect.objectContaining({content: 'Docs for {{ product }}'}), + ]), + }), + ]), + }), + ); + + expect(result).toBe('### Docs for {{ product }}'); + }); + + it(`doesn't apply liquid if it was explicitly disabled by passing disableLiquid = true`, () => { + const input = `### Docs for {{ product }}`; + const vars = {product: 'Diplodoc Platform'}; + + const pluginWithCollect: ExtendedPluginWithCollect = jest.fn(); + pluginWithCollect.collect = jest.fn(); + + const result = transform.collect(input, { + mdItInitOptions: { + plugins: [pluginWithCollect], + vars: vars, + disableLiquid: true, + }, + pluginCollectOptions: {}, + }); + + expect(pluginWithCollect.collect).toHaveBeenCalledTimes(1); + expect(pluginWithCollect.collect).toHaveBeenCalledWith( + '### Docs for {{ product }}', + expect.objectContaining({ + tokenStream: expect.arrayContaining([ + expect.objectContaining({ + children: expect.arrayContaining([ + expect.objectContaining({content: 'Docs for {{ product }}'}), + ]), + }), + ]), + }), + ); + + expect(result).toBe('### Docs for {{ product }}'); + }); + + it('applies collect function in order as specified by the plugin array', () => { + const pluginFactory = (textToAppend: string) => { + const plugin: ExtendedPluginWithCollect = jest.fn(); + + plugin.collect = (input: string) => `${input}${textToAppend}`; + + return plugin; + }; + + const result = transform.collect('', { + mdItInitOptions: { + plugins: [pluginFactory('foo'), pluginFactory('bar'), pluginFactory('baz')], + }, + pluginCollectOptions: {}, + }); + + expect(result).toBe('foobarbaz'); + }); + + it('provides valid token stream to collect functions after running the input through mdit plugins', () => { + const mockPlugin: ExtendedPluginWithCollect = (md) => { + md.core.ruler.push('mock', (state) => + state.tokens.push(new state.Token('mockToken', 'mock', 0)), + ); + }; + + mockPlugin.collect = jest.fn(); + + transform.collect('', { + mdItInitOptions: { + plugins: [mockPlugin], + }, + pluginCollectOptions: {}, + }); + + expect(mockPlugin.collect).toHaveBeenCalledWith( + '', + expect.objectContaining({ + tokenStream: expect.arrayContaining([ + expect.objectContaining({type: 'mockToken'}), + ]), + }), + ); + }); + + it('prioritizes PluginOverrides when evaluating the token stream to pass to collect functions', () => { + const mockPluginFactory = (tokenType: string) => { + const plugin: ExtendedPluginWithCollect = (md) => + md.core.ruler.push('mock', (state) => + state.tokens.push(new state.Token(tokenType, 'mock', 0)), + ); + + plugin.collect = jest.fn(); + + return plugin; + }; + + const originalPlugin = mockPluginFactory('original'); + const overriddenPlugin = mockPluginFactory('overridden'); + + transform.collect('', { + mdItInitOptions: { + plugins: [originalPlugin], + }, + pluginCollectOptions: {}, + parserPluginsOverride: [overriddenPlugin], + }); + + expect(originalPlugin.collect).toHaveBeenCalledWith( + '', + expect.objectContaining({ + tokenStream: expect.arrayContaining([ + expect.objectContaining({type: 'overridden'}), + ]), + }), + ); + + expect(originalPlugin.collect).not.toHaveBeenCalledWith( + '', + expect.objectContaining({ + tokenStream: expect.arrayContaining([ + expect.objectContaining({type: 'original'}), + ]), + }), + ); + }); + + it('never actually invokes collect functions that may exist in `parserPluginsOverride`', () => { + const mockPluginFactory = (tokenType: string) => { + const plugin: ExtendedPluginWithCollect = (md) => + md.core.ruler.push('mock', (state) => + state.tokens.push(new state.Token(tokenType, 'mock', 0)), + ); + + plugin.collect = jest.fn(() => tokenType); + + return plugin; + }; + + const originalPlugin = mockPluginFactory('original'); + const overriddenPlugin = mockPluginFactory('overridden'); + + const result = transform.collect('', { + mdItInitOptions: { + plugins: [originalPlugin], + }, + pluginCollectOptions: {}, + parserPluginsOverride: [overriddenPlugin], + }); + + expect(result).toBe('original'); + expect(overriddenPlugin.collect).not.toHaveBeenCalled(); + }); +});