From d9607ab93bb8d1dade1543f51276cb39b6f033e0 Mon Sep 17 00:00:00 2001 From: skirtle <65301168+skirtles-code@users.noreply.github.com> Date: Tue, 20 Feb 2024 19:56:24 +0000 Subject: [PATCH 1/4] perf: faster handling of static paths --- packages/router/src/matcher/index.ts | 47 +--- packages/router/src/matcher/matcherTree.ts | 208 ++++++++++++++++++ packages/router/src/matcher/pathMatcher.ts | 29 ++- .../router/src/matcher/pathParserRanker.ts | 6 +- .../router/src/matcher/staticPathParser.ts | 71 ++++++ 5 files changed, 322 insertions(+), 39 deletions(-) create mode 100644 packages/router/src/matcher/matcherTree.ts create mode 100644 packages/router/src/matcher/staticPathParser.ts diff --git a/packages/router/src/matcher/index.ts b/packages/router/src/matcher/index.ts index bdc292c86..a6c593d9b 100644 --- a/packages/router/src/matcher/index.ts +++ b/packages/router/src/matcher/index.ts @@ -7,6 +7,7 @@ import { _RouteRecordProps, } from '../types' import { createRouterError, ErrorTypes, MatcherError } from '../errors' +import { createMatcherTree } from './matcherTree' import { createRouteRecordMatcher, RouteRecordMatcher } from './pathMatcher' import { RouteRecordNormalized } from './types' @@ -16,8 +17,6 @@ import type { _PathParserOptions, } from './pathParserRanker' -import { comparePathParserScore } from './pathParserRanker' - import { warn } from '../warning' import { assign, noop } from '../utils' @@ -58,8 +57,8 @@ export function createRouterMatcher( routes: Readonly, globalOptions: PathParserOptions ): RouterMatcher { - // normalized ordered array of matchers - const matchers: RouteRecordMatcher[] = [] + // normalized ordered tree of matchers + const matcherTree = createMatcherTree() const matcherMap = new Map() globalOptions = mergeOptions( { strict: false, end: true, sensitive: false } as PathParserOptions, @@ -203,37 +202,24 @@ export function createRouterMatcher( const matcher = matcherMap.get(matcherRef) if (matcher) { matcherMap.delete(matcherRef) - matchers.splice(matchers.indexOf(matcher), 1) + matcherTree.remove(matcher) matcher.children.forEach(removeRoute) matcher.alias.forEach(removeRoute) } } else { - const index = matchers.indexOf(matcherRef) - if (index > -1) { - matchers.splice(index, 1) - if (matcherRef.record.name) matcherMap.delete(matcherRef.record.name) - matcherRef.children.forEach(removeRoute) - matcherRef.alias.forEach(removeRoute) - } + matcherTree.remove(matcherRef) + if (matcherRef.record.name) matcherMap.delete(matcherRef.record.name) + matcherRef.children.forEach(removeRoute) + matcherRef.alias.forEach(removeRoute) } } function getRoutes() { - return matchers + return matcherTree.toArray() } function insertMatcher(matcher: RouteRecordMatcher) { - let i = 0 - while ( - i < matchers.length && - comparePathParserScore(matcher, matchers[i]) >= 0 && - // Adding children with empty path should still appear before the parent - // https://github.com/vuejs/router/issues/1124 - (matcher.record.path !== matchers[i].record.path || - !isRecordChildOf(matcher, matchers[i])) - ) - i++ - matchers.splice(i, 0, matcher) + matcherTree.add(matcher) // only add the original record to the name map if (matcher.record.name && !isAliasRecord(matcher)) matcherMap.set(matcher.record.name, matcher) @@ -306,7 +292,7 @@ export function createRouterMatcher( ) } - matcher = matchers.find(m => m.re.test(path)) + matcher = matcherTree.find(path) // matcher should have a value after the loop if (matcher) { @@ -319,7 +305,7 @@ export function createRouterMatcher( // match by name or path of current route matcher = currentLocation.name ? matcherMap.get(currentLocation.name) - : matchers.find(m => m.re.test(currentLocation.path)) + : matcherTree.find(currentLocation.path) if (!matcher) throw createRouterError(ErrorTypes.MATCHER_NOT_FOUND, { location, @@ -525,13 +511,4 @@ function checkMissingParamsInAbsolutePath( } } -function isRecordChildOf( - record: RouteRecordMatcher, - parent: RouteRecordMatcher -): boolean { - return parent.children.some( - child => child === record || isRecordChildOf(record, child) - ) -} - export type { PathParserOptions, _PathParserOptions } diff --git a/packages/router/src/matcher/matcherTree.ts b/packages/router/src/matcher/matcherTree.ts new file mode 100644 index 000000000..e9c860661 --- /dev/null +++ b/packages/router/src/matcher/matcherTree.ts @@ -0,0 +1,208 @@ +import { RouteRecordMatcher } from './pathMatcher' +import { comparePathParserScore } from './pathParserRanker' + +type MatcherTree = { + add: (matcher: RouteRecordMatcher) => void + remove: (matcher: RouteRecordMatcher) => void + find: (path: string) => RouteRecordMatcher | undefined + toArray: () => RouteRecordMatcher[] +} + +function normalizePath(path: string) { + // We match case-insensitively initially, then let the matcher check more rigorously + path = path.toUpperCase() + + // TODO: Check more thoroughly whether this is really necessary + while (path.endsWith('/')) { + path = path.slice(0, -1) + } + + return path +} + +export function createMatcherTree(): MatcherTree { + const root = createMatcherNode() + const exactMatchers: Record = + Object.create(null) + + return { + add(matcher) { + if (matcher.staticPath) { + const path = normalizePath(matcher.record.path) + + exactMatchers[path] = exactMatchers[path] || [] + insertMatcher(matcher, exactMatchers[path]) + } else { + root.add(matcher) + } + }, + + remove(matcher) { + if (matcher.staticPath) { + const path = normalizePath(matcher.record.path) + + if (exactMatchers[path]) { + // TODO: Remove array if length is zero + remove(matcher, exactMatchers[path]) + } + } else { + root.remove(matcher) + } + }, + + find(path) { + const matchers = exactMatchers[normalizePath(path)] + + if (matchers) { + for (const matcher of matchers) { + if (matcher.re.test(path)) { + return matcher + } + } + } + + return root.find(path) + }, + + toArray() { + const arr = root.toArray() + + for (const key in exactMatchers) { + arr.unshift(...exactMatchers[key]) + } + + return arr + }, + } +} + +function createMatcherNode(depth = 1): MatcherTree { + let segments: Record | null = null + let wildcards: RouteRecordMatcher[] | null = null + + return { + add(matcher) { + const { staticTokens } = matcher + const myToken = staticTokens[depth - 1]?.toUpperCase() + + if (myToken != null) { + if (!segments) { + segments = Object.create(null) + } + + if (!segments![myToken]) { + segments![myToken] = createMatcherNode(depth + 1) + } + + segments![myToken].add(matcher) + + return + } + + if (!wildcards) { + wildcards = [] + } + + insertMatcher(matcher, wildcards) + }, + + remove(matcher) { + // TODO: Remove any empty data structures + if (segments) { + const myToken = matcher.staticTokens[depth - 1]?.toUpperCase() + + if (myToken != null) { + if (segments[myToken]) { + segments[myToken].remove(matcher) + return + } + } + } + + if (wildcards) { + remove(matcher, wildcards) + } + }, + + find(path) { + const tokens = path.split('/') + const myToken = tokens[depth] + + if (segments && myToken != null) { + const segmentMatcher = segments[myToken.toUpperCase()] + + if (segmentMatcher) { + const match = segmentMatcher.find(path) + + if (match) { + return match + } + } + } + + if (wildcards) { + return wildcards.find(matcher => matcher.re.test(path)) + } + + return + }, + + toArray() { + const matchers: RouteRecordMatcher[] = [] + + for (const key in segments) { + // TODO: push may not scale well enough + matchers.push(...segments[key].toArray()) + } + + if (wildcards) { + matchers.push(...wildcards) + } + + return matchers + }, + } +} + +function remove(item: T, items: T[]) { + const index = items.indexOf(item) + + if (index > -1) { + items.splice(index, 1) + } +} + +function insertMatcher( + matcher: RouteRecordMatcher, + matchers: RouteRecordMatcher[] +) { + const index = findInsertionIndex(matcher, matchers) + matchers.splice(index, 0, matcher) +} + +function findInsertionIndex( + matcher: RouteRecordMatcher, + matchers: RouteRecordMatcher[] +) { + let i = 0 + while ( + i < matchers.length && + comparePathParserScore(matcher, matchers[i]) >= 0 && + // Adding children with empty path should still appear before the parent + // https://github.com/vuejs/router/issues/1124 + (matcher.record.path !== matchers[i].record.path || + !isRecordChildOf(matcher, matchers[i])) + ) + i++ + + return i +} + +function isRecordChildOf( + record: RouteRecordMatcher, + parent: RouteRecordMatcher +): boolean { + return parent.children.some( + child => child === record || isRecordChildOf(record, child) + ) +} diff --git a/packages/router/src/matcher/pathMatcher.ts b/packages/router/src/matcher/pathMatcher.ts index aae2b7826..7ceef0671 100644 --- a/packages/router/src/matcher/pathMatcher.ts +++ b/packages/router/src/matcher/pathMatcher.ts @@ -4,11 +4,14 @@ import { PathParser, PathParserOptions, } from './pathParserRanker' +import { staticPathToParser } from './staticPathParser' import { tokenizePath } from './pathTokenizer' import { warn } from '../warning' import { assign } from '../utils' export interface RouteRecordMatcher extends PathParser { + staticPath: boolean + staticTokens: string[] record: RouteRecord parent: RouteRecordMatcher | undefined children: RouteRecordMatcher[] @@ -21,7 +24,29 @@ export function createRouteRecordMatcher( parent: RouteRecordMatcher | undefined, options?: PathParserOptions ): RouteRecordMatcher { - const parser = tokensToParser(tokenizePath(record.path), options) + const tokens = tokenizePath(record.path) + + // TODO: Merge options properly + const staticPath = + options?.end !== false && + tokens.every( + segment => + segment.length === 0 || (segment.length === 1 && segment[0].type === 0) + ) + + const staticTokens: string[] = [] + + for (const token of tokens) { + if (token.length === 1 && token[0].type === 0) { + staticTokens.push(token[0].value) + } else { + break + } + } + + const parser = staticPath + ? staticPathToParser(record.path, tokens, options) + : tokensToParser(tokens, options) // warn against params with the same name if (__DEV__) { @@ -36,6 +61,8 @@ export function createRouteRecordMatcher( } const matcher: RouteRecordMatcher = assign(parser, { + staticPath, + staticTokens, record, parent, // these needs to be populated by the parent diff --git a/packages/router/src/matcher/pathParserRanker.ts b/packages/router/src/matcher/pathParserRanker.ts index 670013794..dfc4cde03 100644 --- a/packages/router/src/matcher/pathParserRanker.ts +++ b/packages/router/src/matcher/pathParserRanker.ts @@ -16,7 +16,7 @@ export interface PathParser { /** * The regexp used to match a url */ - re: RegExp + re: { test: (str: string) => boolean } /** * The score of the parser @@ -89,7 +89,7 @@ export type PathParserOptions = Pick< // default pattern for a param: non-greedy everything but / const BASE_PARAM_PATTERN = '[^/]+?' -const BASE_PATH_PARSER_OPTIONS: Required<_PathParserOptions> = { +export const BASE_PATH_PARSER_OPTIONS: Required<_PathParserOptions> = { sensitive: false, strict: false, start: true, @@ -97,7 +97,7 @@ const BASE_PATH_PARSER_OPTIONS: Required<_PathParserOptions> = { } // Scoring values used in tokensToParser -const enum PathScore { +export const enum PathScore { _multiplier = 10, Root = 9 * _multiplier, // just / Segment = 4 * _multiplier, // /a-segment diff --git a/packages/router/src/matcher/staticPathParser.ts b/packages/router/src/matcher/staticPathParser.ts new file mode 100644 index 000000000..c4d4af8e2 --- /dev/null +++ b/packages/router/src/matcher/staticPathParser.ts @@ -0,0 +1,71 @@ +import { + PathParser, + PathParserOptions, + PathScore, + BASE_PATH_PARSER_OPTIONS, +} from './pathParserRanker' +import { Token } from './pathTokenizer' +import { assign } from '../utils' + +export function staticPathToParser( + path: string, + tokens: Array, + extraOptions?: PathParserOptions +): PathParser { + const options = assign({}, BASE_PATH_PARSER_OPTIONS, extraOptions) + + const matchPath = options.sensitive ? path : path.toUpperCase() + + let test: (p: string) => boolean + + if (options.strict) { + if (options.sensitive) { + test = p => p === matchPath + } else { + test = p => p.toUpperCase() === matchPath + } + } else { + const withSlash = matchPath.endsWith('/') ? matchPath : matchPath + '/' + const withoutSlash = withSlash.slice(0, -1) + + if (options.sensitive) { + test = p => p === withSlash || p === withoutSlash + } else { + test = p => { + p = p.toUpperCase() + return p === withSlash || p === withoutSlash + } + } + } + + const score: Array = tokens.map(segment => { + if (segment.length === 1) { + return [ + PathScore.Static + + PathScore.Segment + + (options.sensitive ? PathScore.BonusCaseSensitive : 0), + ] + } else { + return [PathScore.Root] + } + }) + + if (options.strict && options.end) { + const i = score.length - 1 + score[i][score[i].length - 1] += PathScore.BonusStrict + } + + return { + re: { + test, + }, + score, + keys: [], + parse() { + return {} + }, + stringify() { + return path || '/' + }, + } +} From 07e86a9f9dc4d9b8aa571358b565ed2b7ff63469 Mon Sep 17 00:00:00 2001 From: skirtle <65301168+skirtles-code@users.noreply.github.com> Date: Mon, 4 Mar 2024 10:49:24 +0000 Subject: [PATCH 2/4] Fix problems combining `end: false` and `strict: false` --- packages/router/src/matcher/pathMatcher.ts | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/packages/router/src/matcher/pathMatcher.ts b/packages/router/src/matcher/pathMatcher.ts index 7ceef0671..78fcc466b 100644 --- a/packages/router/src/matcher/pathMatcher.ts +++ b/packages/router/src/matcher/pathMatcher.ts @@ -44,6 +44,10 @@ export function createRouteRecordMatcher( } } + if (options?.end === false && !options?.strict) { + staticTokens.pop() + } + const parser = staticPath ? staticPathToParser(record.path, tokens, options) : tokensToParser(tokens, options) From 987ea0971356b7667f1bab0a04831b47475ab7fa Mon Sep 17 00:00:00 2001 From: skirtle <65301168+skirtles-code@users.noreply.github.com> Date: Mon, 4 Mar 2024 11:45:04 +0000 Subject: [PATCH 3/4] Fix problems with resolving paths for `sensitive` and `end` options --- packages/router/src/matcher/matcherTree.ts | 43 ++++++++++++++-------- 1 file changed, 27 insertions(+), 16 deletions(-) diff --git a/packages/router/src/matcher/matcherTree.ts b/packages/router/src/matcher/matcherTree.ts index e9c860661..aaf889762 100644 --- a/packages/router/src/matcher/matcherTree.ts +++ b/packages/router/src/matcher/matcherTree.ts @@ -20,6 +20,22 @@ function normalizePath(path: string) { return path } +function chooseBestMatcher( + firstMatcher: RouteRecordMatcher | undefined, + secondMatcher: RouteRecordMatcher | undefined +) { + if (secondMatcher) { + if ( + !firstMatcher || + comparePathParserScore(firstMatcher, secondMatcher) > 0 + ) { + firstMatcher = secondMatcher + } + } + + return firstMatcher +} + export function createMatcherTree(): MatcherTree { const root = createMatcherNode() const exactMatchers: Record = @@ -53,15 +69,10 @@ export function createMatcherTree(): MatcherTree { find(path) { const matchers = exactMatchers[normalizePath(path)] - if (matchers) { - for (const matcher of matchers) { - if (matcher.re.test(path)) { - return matcher - } - } - } - - return root.find(path) + return chooseBestMatcher( + matchers && matchers.find(matcher => matcher.re.test(path)), + root.find(path) + ) }, toArray() { @@ -127,24 +138,24 @@ function createMatcherNode(depth = 1): MatcherTree { find(path) { const tokens = path.split('/') const myToken = tokens[depth] + let matcher: RouteRecordMatcher | undefined if (segments && myToken != null) { const segmentMatcher = segments[myToken.toUpperCase()] if (segmentMatcher) { - const match = segmentMatcher.find(path) - - if (match) { - return match - } + matcher = segmentMatcher.find(path) } } if (wildcards) { - return wildcards.find(matcher => matcher.re.test(path)) + matcher = chooseBestMatcher( + matcher, + wildcards.find(matcher => matcher.re.test(path)) + ) } - return + return matcher }, toArray() { From 454ecf8104dbf5ed9016b8616a195a9d92caf4ac Mon Sep 17 00:00:00 2001 From: skirtle <65301168+skirtles-code@users.noreply.github.com> Date: Mon, 26 Aug 2024 23:03:42 +0100 Subject: [PATCH 4/4] Fix problems caused by merging 'main' --- packages/router/src/matcher/index.ts | 79 +-------------- packages/router/src/matcher/matcherTree.ts | 109 ++++++++++++++++----- 2 files changed, 88 insertions(+), 100 deletions(-) diff --git a/packages/router/src/matcher/index.ts b/packages/router/src/matcher/index.ts index 01edf992c..0b4ad1221 100644 --- a/packages/router/src/matcher/index.ts +++ b/packages/router/src/matcher/index.ts @@ -5,7 +5,7 @@ import { isRouteName, } from '../types' import { createRouterError, ErrorTypes, MatcherError } from '../errors' -import { createMatcherTree } from './matcherTree' +import { createMatcherTree, isMatchable } from './matcherTree' import { createRouteRecordMatcher, RouteRecordMatcher } from './pathMatcher' import { RouteRecordNormalized } from './types' @@ -340,7 +340,7 @@ export function createRouterMatcher( routes.forEach(route => addRoute(route)) function clearRoutes() { - matchers.length = 0 + matcherTree.clear() matcherMap.clear() } @@ -523,79 +523,4 @@ function checkMissingParamsInAbsolutePath( } } -/** - * Performs a binary search to find the correct insertion index for a new matcher. - * - * Matchers are primarily sorted by their score. If scores are tied then we also consider parent/child relationships, - * with descendants coming before ancestors. If there's still a tie, new routes are inserted after existing routes. - * - * @param matcher - new matcher to be inserted - * @param matchers - existing matchers - */ -function findInsertionIndex( - matcher: RouteRecordMatcher, - matchers: RouteRecordMatcher[] -) { - // First phase: binary search based on score - let lower = 0 - let upper = matchers.length - - while (lower !== upper) { - const mid = (lower + upper) >> 1 - const sortOrder = comparePathParserScore(matcher, matchers[mid]) - - if (sortOrder < 0) { - upper = mid - } else { - lower = mid + 1 - } - } - - // Second phase: check for an ancestor with the same score - const insertionAncestor = getInsertionAncestor(matcher) - - if (insertionAncestor) { - upper = matchers.lastIndexOf(insertionAncestor, upper - 1) - - if (__DEV__ && upper < 0) { - // This should never happen - warn( - `Finding ancestor route "${insertionAncestor.record.path}" failed for "${matcher.record.path}"` - ) - } - } - - return upper -} - -function getInsertionAncestor(matcher: RouteRecordMatcher) { - let ancestor: RouteRecordMatcher | undefined = matcher - - while ((ancestor = ancestor.parent)) { - if ( - isMatchable(ancestor) && - comparePathParserScore(matcher, ancestor) === 0 - ) { - return ancestor - } - } - - return -} - -/** - * Checks if a matcher can be reachable. This means if it's possible to reach it as a route. For example, routes without - * a component, or name, or redirect, are just used to group other routes. - * @param matcher - * @param matcher.record record of the matcher - * @returns - */ -function isMatchable({ record }: RouteRecordMatcher): boolean { - return !!( - record.name || - (record.components && Object.keys(record.components).length) || - record.redirect - ) -} - export type { PathParserOptions, _PathParserOptions } diff --git a/packages/router/src/matcher/matcherTree.ts b/packages/router/src/matcher/matcherTree.ts index aaf889762..e00185b32 100644 --- a/packages/router/src/matcher/matcherTree.ts +++ b/packages/router/src/matcher/matcherTree.ts @@ -1,13 +1,18 @@ import { RouteRecordMatcher } from './pathMatcher' import { comparePathParserScore } from './pathParserRanker' +import { warn } from '../warning' -type MatcherTree = { +type MatcherNode = { add: (matcher: RouteRecordMatcher) => void remove: (matcher: RouteRecordMatcher) => void find: (path: string) => RouteRecordMatcher | undefined toArray: () => RouteRecordMatcher[] } +type MatcherTree = MatcherNode & { + clear: () => void +} + function normalizePath(path: string) { // We match case-insensitively initially, then let the matcher check more rigorously path = path.toUpperCase() @@ -37,9 +42,8 @@ function chooseBestMatcher( } export function createMatcherTree(): MatcherTree { - const root = createMatcherNode() - const exactMatchers: Record = - Object.create(null) + let root = createMatcherNode() + let exactMatchers: Record = Object.create(null) return { add(matcher) { @@ -66,6 +70,11 @@ export function createMatcherTree(): MatcherTree { } }, + clear() { + root = createMatcherNode() + exactMatchers = Object.create(null) + }, + find(path) { const matchers = exactMatchers[normalizePath(path)] @@ -87,8 +96,8 @@ export function createMatcherTree(): MatcherTree { } } -function createMatcherNode(depth = 1): MatcherTree { - let segments: Record | null = null +function createMatcherNode(depth = 1): MatcherNode { + let segments: Record | null = null let wildcards: RouteRecordMatcher[] | null = null return { @@ -191,29 +200,83 @@ function insertMatcher( matchers.splice(index, 0, matcher) } +/** + * Performs a binary search to find the correct insertion index for a new matcher. + * + * Matchers are primarily sorted by their score. If scores are tied then we also consider parent/child relationships, + * with descendants coming before ancestors. If there's still a tie, new routes are inserted after existing routes. + * + * @param matcher - new matcher to be inserted + * @param matchers - existing matchers + */ function findInsertionIndex( matcher: RouteRecordMatcher, matchers: RouteRecordMatcher[] ) { - let i = 0 - while ( - i < matchers.length && - comparePathParserScore(matcher, matchers[i]) >= 0 && - // Adding children with empty path should still appear before the parent - // https://github.com/vuejs/router/issues/1124 - (matcher.record.path !== matchers[i].record.path || - !isRecordChildOf(matcher, matchers[i])) - ) - i++ + // First phase: binary search based on score + let lower = 0 + let upper = matchers.length + + while (lower !== upper) { + const mid = (lower + upper) >> 1 + const sortOrder = comparePathParserScore(matcher, matchers[mid]) + + if (sortOrder < 0) { + upper = mid + } else { + lower = mid + 1 + } + } + + // Second phase: check for an ancestor with the same score + const insertionAncestor = getInsertionAncestor(matcher) + + if (insertionAncestor) { + upper = matchers.lastIndexOf(insertionAncestor, upper - 1) + + if (__DEV__ && upper < 0) { + // This should never happen + warn( + `Finding ancestor route "${insertionAncestor.record.path}" failed for "${matcher.record.path}"` + ) + } + } + + return upper +} + +function getInsertionAncestor(matcher: RouteRecordMatcher) { + let ancestor: RouteRecordMatcher | undefined = matcher + + while ((ancestor = ancestor.parent)) { + if ( + isMatchable(ancestor) && + matcher.staticTokens.length === ancestor.staticTokens.length && + comparePathParserScore(matcher, ancestor) === 0 && + ancestor.staticTokens.every( + (token, index) => + matcher.staticTokens[index].toUpperCase() === token.toUpperCase() + ) + ) { + return ancestor + } + } - return i + return } -function isRecordChildOf( - record: RouteRecordMatcher, - parent: RouteRecordMatcher -): boolean { - return parent.children.some( - child => child === record || isRecordChildOf(record, child) +/** + * Checks if a matcher can be reachable. This means if it's possible to reach it as a route. For example, routes without + * a component, or name, or redirect, are just used to group other routes. + * @param matcher + * @param matcher.record record of the matcher + * @returns + */ +// TODO: This should probably live elsewhere +export function isMatchable({ record }: RouteRecordMatcher): boolean { + return !!( + record.name || + (record.components && Object.keys(record.components).length) || + record.redirect ) }