diff --git a/packages/core/src/code-to-tokens-base.ts b/packages/core/src/code-to-tokens-base.ts index 520a64d35..405c3ad7d 100644 --- a/packages/core/src/code-to-tokens-base.ts +++ b/packages/core/src/code-to-tokens-base.ts @@ -45,6 +45,11 @@ export function tokenizeWithTheme( ...options?.colorReplacements, } + const { + tokenizeMaxLineLength = 0, + tokenizeTimeLimit = 500, + } = options + const lines = splitLines(code) let ruleStack = INITIAL @@ -59,6 +64,18 @@ export function tokenizeWithTheme( continue } + // Do not attempt to tokenize if the line length is longer than the `tokenizationMaxLineLength` + if (tokenizeMaxLineLength > 0 && line.length >= tokenizeMaxLineLength) { + actual = [] + final.push([{ + content: line, + offset: lineOffset, + color: '', + fontStyle: 0, + }]) + continue + } + let resultWithScopes let tokensWithScopes let tokensWithScopesIndex @@ -69,7 +86,7 @@ export function tokenizeWithTheme( tokensWithScopesIndex = 0 } - const result = grammar.tokenizeLine2(line, ruleStack) + const result = grammar.tokenizeLine2(line, ruleStack, tokenizeTimeLimit) const tokensLength = result.tokens.length / 2 for (let j = 0; j < tokensLength; j++) { diff --git a/packages/core/src/types/options.ts b/packages/core/src/types/options.ts index 0688c72e2..25ee33958 100644 --- a/packages/core/src/types/options.ts +++ b/packages/core/src/types/options.ts @@ -118,7 +118,7 @@ export interface CodeToHastOptionsCommon extends TransformerOptions, DecorationOptions, - Pick { + Pick { lang: StringLiteralUnion diff --git a/packages/core/src/types/tokens.ts b/packages/core/src/types/tokens.ts index 50bfe1d5a..40b225d51 100644 --- a/packages/core/src/types/tokens.ts +++ b/packages/core/src/types/tokens.ts @@ -158,6 +158,20 @@ export interface TokenizeWithThemeOptions { * This will be merged with theme's `colorReplacements` if any. */ colorReplacements?: Record + + /** + * Lines above this length will not be tokenized for performance reasons. + * + * @default 0 (no limit) + */ + tokenizeMaxLineLength?: number + + /** + * Time limit in milliseconds for tokenizing a single line. + * + * @default 500 (0.5s) + */ + tokenizeTimeLimit?: number } /** diff --git a/packages/monaco/src/index.ts b/packages/monaco/src/index.ts index f429b0602..fa54e4653 100644 --- a/packages/monaco/src/index.ts +++ b/packages/monaco/src/index.ts @@ -77,10 +77,25 @@ export function shikiToMonaco( return new TokenizerState(INITIAL, highlighter) }, tokenize(line, state: TokenizerState) { + // Do not attempt to tokenize if a line is too long + // default to 20000 (as in monaco-editor-core defaults) + const tokenizeMaxLineLength = 20000 + const tokenizeTimeLimit = 500 + + if (line.length >= tokenizeMaxLineLength) { + return { + endState: state, + tokens: [{ startIndex: 0, scopes: '' }], + } + } + const grammar = state.highlighter.getLanguage(lang) const { colorMap } = state.highlighter.setTheme(currentTheme) const theme = themeMap.get(currentTheme) - const result = grammar.tokenizeLine2(line, state.ruleStack) + const result = grammar.tokenizeLine2(line, state.ruleStack, tokenizeTimeLimit) + + if (result.stoppedEarly) + console.warn(`Time limit reached when tokenizing line: ${line.substring(0, 100)}`) const colorToScopeMap = new Map() diff --git a/packages/shiki/test/general.test.ts b/packages/shiki/test/general.test.ts index 5840bccc3..d1ed9bac7 100644 --- a/packages/shiki/test/general.test.ts +++ b/packages/shiki/test/general.test.ts @@ -179,6 +179,21 @@ describe('should', () => { } } }) + + it('skip line tokenizing', async () => { + const longText = 'foo'.repeat(50) + + expect(await codeToHtml(`const long = ${longText}`, { + theme: 'vitesse-light', + lang: 'javascript', + })).toMatchInlineSnapshot(`"
const long = ${longText}
"`) + + expect(await codeToHtml(`const short = ""\nconst long = ${longText}`, { + theme: 'vitesse-light', + lang: 'javascript', + tokenizeMaxLineLength: 100, + })).toMatchInlineSnapshot(`"
const short = ""\nconst long = ${longText}
"`) + }) }) describe('errors', () => {