Skip to content

Commit 0e6b2d9

Browse files
amsalbrophdawg11
andauthored
fix(router): "hard" redirect to different paths on the same origin if redirect location does not contain basename (#10076)
Co-authored-by: Matt Brophy <[email protected]>
1 parent d6af011 commit 0e6b2d9

File tree

5 files changed

+76
-8
lines changed

5 files changed

+76
-8
lines changed

.changeset/wet-dogs-check.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"@remix-run/router": patch
3+
---
4+
5+
Correctly perform a "hard" redirect for same-origin absolute URLs outside of the router basename

contributors.yml

+1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
- alany411
1111
- alexlbr
1212
- AmRo045
13+
- amsal
1314
- andreiduca
1415
- arnassavickas
1516
- aroyan

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -105,7 +105,7 @@
105105
},
106106
"filesize": {
107107
"packages/router/dist/router.umd.min.js": {
108-
"none": "41.6 kB"
108+
"none": "41.7 kB"
109109
},
110110
"packages/react-router/dist/react-router.production.min.js": {
111111
"none": "13 kB"

packages/router/__tests__/router-test.ts

+58
Original file line numberDiff line numberDiff line change
@@ -630,6 +630,10 @@ function setup({
630630
let popHref = history.createHref(history.location);
631631
if (currentRouter?.basename) {
632632
popHref = stripBasename(popHref, currentRouter.basename) as string;
633+
invariant(
634+
popHref,
635+
"href passed to navigate should start with basename"
636+
);
633637
}
634638
helpers = getNavigationHelpers(popHref, navigationId);
635639
unsubscribe();
@@ -645,6 +649,10 @@ function setup({
645649
let navHref = href;
646650
if (currentRouter.basename) {
647651
navHref = stripBasename(navHref, currentRouter.basename) as string;
652+
invariant(
653+
navHref,
654+
"href passed to t.navigate() should start with basename"
655+
);
648656
}
649657
helpers = getNavigationHelpers(navHref, navigationId);
650658
shims?.forEach((routeId) =>
@@ -6293,6 +6301,56 @@ describe("a router", () => {
62936301
});
62946302
});
62956303

6304+
it("properly handles same-origin absolute URLs when using a basename", async () => {
6305+
let t = setup({ routes: REDIRECT_ROUTES, basename: "/base" });
6306+
6307+
let A = await t.navigate("/base/parent/child", {
6308+
formMethod: "post",
6309+
formData: createFormData({}),
6310+
});
6311+
6312+
let B = await A.actions.child.redirectReturn(
6313+
"http://localhost/base/parent",
6314+
undefined,
6315+
undefined,
6316+
["parent"]
6317+
);
6318+
await B.loaders.parent.resolve("PARENT");
6319+
expect(t.router.state.location).toMatchObject({
6320+
hash: "",
6321+
pathname: "/base/parent",
6322+
search: "",
6323+
state: {
6324+
_isRedirect: true,
6325+
},
6326+
});
6327+
});
6328+
6329+
it("treats same-origin absolute URLs as external if they don't match the basename", async () => {
6330+
// This is gross, don't blame me, blame SO :)
6331+
// https://stackoverflow.com/a/60697570
6332+
let oldLocation = window.location;
6333+
const location = new URL(window.location.href) as unknown as Location;
6334+
location.assign = jest.fn();
6335+
location.replace = jest.fn();
6336+
delete (window as any).location;
6337+
window.location = location as unknown as Location;
6338+
6339+
let t = setup({ routes: REDIRECT_ROUTES, basename: "/base" });
6340+
6341+
let A = await t.navigate("/base/parent/child", {
6342+
formMethod: "post",
6343+
formData: createFormData({}),
6344+
});
6345+
6346+
let url = "http://localhost/not/the/same/basename";
6347+
await A.actions.child.redirectReturn(url);
6348+
expect(window.location.assign).toHaveBeenCalledWith(url);
6349+
expect(window.location.replace).not.toHaveBeenCalled();
6350+
6351+
window.location = oldLocation;
6352+
});
6353+
62966354
describe("redirect status code handling", () => {
62976355
it("should not treat 300 as a redirect", async () => {
62986356
let t = setup({ routes: REDIRECT_ROUTES });

packages/router/router.ts

+11-7
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@ import {
3333
joinPaths,
3434
matchRoutes,
3535
resolveTo,
36+
stripBasename,
3637
warning,
3738
} from "./utils";
3839

@@ -1935,15 +1936,17 @@ export function createRouter(init: RouterInit): Router {
19351936
redirectLocation,
19361937
"Expected a location on the redirect navigation"
19371938
);
1938-
19391939
// Check if this an absolute external redirect that goes to a new origin
19401940
if (
19411941
ABSOLUTE_URL_REGEX.test(redirect.location) &&
19421942
isBrowser &&
19431943
typeof window?.location !== "undefined"
19441944
) {
1945-
let newOrigin = init.history.createURL(redirect.location).origin;
1946-
if (window.location.origin !== newOrigin) {
1945+
let url = init.history.createURL(redirect.location);
1946+
let isDifferentBasename =
1947+
stripBasename(url.pathname, init.basename || "/") == null;
1948+
1949+
if (window.location.origin !== url.origin || isDifferentBasename) {
19471950
if (replace) {
19481951
window.location.replace(redirect.location);
19491952
} else {
@@ -3173,14 +3176,15 @@ async function callLoaderOrAction(
31733176

31743177
location = createPath(resolvedLocation);
31753178
} else if (!isStaticRequest) {
3176-
// Strip off the protocol+origin for same-origin absolute redirects.
3177-
// If this is a static reques, we can let it go back to the browser
3178-
// as-is
3179+
// Strip off the protocol+origin for same-origin + same-basename absolute
3180+
// redirects. If this is a static request, we can let it go back to the
3181+
// browser as-is
31793182
let currentUrl = new URL(request.url);
31803183
let url = location.startsWith("//")
31813184
? new URL(currentUrl.protocol + location)
31823185
: new URL(location);
3183-
if (url.origin === currentUrl.origin) {
3186+
let isSameBasename = stripBasename(url.pathname, basename) != null;
3187+
if (url.origin === currentUrl.origin && isSameBasename) {
31843188
location = url.pathname + url.search + url.hash;
31853189
}
31863190
}

0 commit comments

Comments
 (0)