Files
zerobyte/app/server/modules/sso/middlewares/validate-callback-urls.ts

44 lines
1.2 KiB
TypeScript

import { APIError } from "better-auth/api";
import type { AuthMiddlewareContext } from "~/server/lib/auth";
function isValidCallbackPath(value: string): boolean {
let decoded: string;
try {
decoded = decodeURIComponent(value);
} catch {
return false;
}
if (!decoded.startsWith("/") || decoded.startsWith("//") || decoded.includes("\\")) {
return false;
}
if (decoded.startsWith("/sso/callback/") || decoded.startsWith("/sso/saml2/")) {
return false;
}
return true;
}
export const validateSsoCallbackUrls = async (ctx: AuthMiddlewareContext) => {
if (ctx.path !== "/sign-in/sso") {
return;
}
const sources = [ctx.body, ctx.query].filter((s) => s && typeof s === "object");
for (const source of sources) {
const payload = source as Record<string, unknown>;
for (const field of ["callbackURL", "errorCallbackURL", "newUserCallbackURL"]) {
const value = payload[field];
if (value !== undefined && (typeof value !== "string" || !isValidCallbackPath(value))) {
throw new APIError("BAD_REQUEST", {
message: `Invalid ${field}. Only relative paths like /login are allowed.`,
code: `INVALID_${field.toUpperCase()}`,
});
}
}
}
};