From 22205d93bd414a9061a9609d4b1a03b38c61f415 Mon Sep 17 00:00:00 2001 From: Marvin Hagemeister Date: Mon, 16 Dec 2024 14:08:33 +0100 Subject: [PATCH] fix(router): exclude route groups in scoring for sorting --- src/plugins/fs_routes/mod.ts | 60 +++++++++++++++++++++++++----- src/plugins/fs_routes/mod_test.tsx | 53 ++++++++++++++++++++++++++ 2 files changed, 104 insertions(+), 9 deletions(-) diff --git a/src/plugins/fs_routes/mod.ts b/src/plugins/fs_routes/mod.ts index c85b0ace634..7e9cb6c5605 100644 --- a/src/plugins/fs_routes/mod.ts +++ b/src/plugins/fs_routes/mod.ts @@ -390,16 +390,48 @@ export function sortRoutePaths(a: string, b: string) { if (APP_REG.test(a)) return -1; else if (APP_REG.test(b)) return 1; - let segmentIdx = 0; const aLen = a.length; const bLen = b.length; - const maxLen = aLen > bLen ? aLen : bLen; - for (let i = 0; i < maxLen; i++) { - const charA = a.charAt(i); - const charB = b.charAt(i); + + let segment = false; + let aIdx = 0; + let bIdx = 0; + for (; aIdx < aLen && bIdx < bLen; aIdx++, bIdx++) { + let charA = a.charAt(aIdx); + let charB = b.charAt(bIdx); + + // When comparing a grouped route with a non-grouped one, we + // need to skip over the group name to effectively compare the + // actual route. + if (charA === "(" && charB !== "(") { + aIdx++; + + while (aIdx < aLen) { + charA = a.charAt(aIdx); + if (charA === ")") { + aIdx += 2; + charA = a.charAt(aIdx); + break; + } + + aIdx++; + } + } else if (charB === "(" && charA !== "(") { + bIdx++; + while (bIdx < bLen) { + charB = b.charAt(bIdx); + if (charB === ")") { + bIdx += 2; + charB = b.charAt(bIdx); + break; + } + + bIdx++; + } + } if (charA === "/" || charB === "/") { - segmentIdx = i; + segment = true; // If the other path doesn't close the segment // then we don't need to continue @@ -409,9 +441,11 @@ export function sortRoutePaths(a: string, b: string) { continue; } - if (i === segmentIdx + 1) { - const scoreA = getRoutePathScore(charA, a, i); - const scoreB = getRoutePathScore(charB, b, i); + if (segment) { + segment = false; + + const scoreA = getRoutePathScore(charA, a, aIdx); + const scoreB = getRoutePathScore(charB, b, bIdx); if (scoreA === scoreB) { if (charA !== charB) { // TODO: Do we need localeSort here or is this good enough? @@ -427,6 +461,14 @@ export function sortRoutePaths(a: string, b: string) { // TODO: Do we need localeSort here or is this good enough? return charA < charB ? 0 : 1; } + + // If we're at the end of A or B, then we assume that the longer + // path is more specific + if (aIdx === aLen - 1 && bIdx < bLen - 1) { + return 1; + } else if (bIdx === bLen - 1 && aIdx < aLen - 1) { + return -1; + } } return 0; diff --git a/src/plugins/fs_routes/mod_test.tsx b/src/plugins/fs_routes/mod_test.tsx index 41db660b96a..261a2dec2de 100644 --- a/src/plugins/fs_routes/mod_test.tsx +++ b/src/plugins/fs_routes/mod_test.tsx @@ -866,6 +866,36 @@ Deno.test("fsRoutes - route group specific templates", async () => { ); }); +Deno.test("fsRoutes - route group order #2", async () => { + const server = await createServer({ + "routes/(app)/[org]/[app]/index.tsx": { + default: () =>
fail
, + }, + "routes/auth/login.tsx": { + default: () =>
ok
, + }, + }); + + const res = await server.get("/auth/login"); + const doc = parseHtml(await res.text()); + expect(doc.body.firstChild?.textContent).toEqual("ok"); +}); + +Deno.test("fsRoutes - route group order #3", async () => { + const server = await createServer({ + "routes/(app)/(foo)/foo/index.tsx": { + default: () =>
fail
, + }, + "routes/(foo)/foo/bar.tsx": { + default: () =>
ok
, + }, + }); + + const res = await server.get("/foo/bar"); + const doc = parseHtml(await res.text()); + expect(doc.body.firstChild?.textContent).toEqual("ok"); +}); + Deno.test("fsRoutes - async route components", async () => { const server = await createServer<{ text: string }>({ "routes/_error.tsx": { @@ -1109,6 +1139,29 @@ Deno.test("fsRoutes - sortRoutePaths", () => { "/tsx/index.tsx", ]; expect(routes).toEqual(sorted); + + // Skip over groups + routes = [ + "/(app)/[org]/[app]/index.ts", + "/auth/login.ts", + ]; + routes.sort(sortRoutePaths); + sorted = [ + "/auth/login.ts", + "/(app)/[org]/[app]/index.ts", + ]; + expect(routes).toEqual(sorted); + + routes = [ + "/auth/login.ts", + "/(app)/[org]/[app]/index.ts", + ]; + routes.sort(sortRoutePaths); + sorted = [ + "/auth/login.ts", + "/(app)/[org]/[app]/index.ts", + ]; + expect(routes).toEqual(sorted); }); Deno.test("fsRoutes - registers default GET route for component without GET handler", async () => {