mirror of
https://github.com/outline/outline.git
synced 2026-06-13 11:25:03 +03:00
92 lines
2.8 KiB
TypeScript
92 lines
2.8 KiB
TypeScript
import { InternalError } from "@server/errors";
|
|
import Logger from "@server/logging/Logger";
|
|
import fetch from "@server/utils/fetch";
|
|
|
|
export interface OIDCConfiguration {
|
|
issuer: string;
|
|
authorization_endpoint: string;
|
|
token_endpoint: string;
|
|
userinfo_endpoint: string;
|
|
jwks_uri?: string;
|
|
end_session_endpoint?: string;
|
|
scopes_supported?: string[];
|
|
response_types_supported?: string[];
|
|
grant_types_supported?: string[];
|
|
code_challenge_methods_supported?: string[];
|
|
}
|
|
|
|
/**
|
|
* Fetches OIDC configuration from the well-known endpoint
|
|
* @param issuerUrl The OIDC issuer URL
|
|
* @returns Promise resolving to the OIDC configuration
|
|
*/
|
|
export async function fetchOIDCConfiguration(
|
|
issuerUrl: string
|
|
): Promise<OIDCConfiguration> {
|
|
try {
|
|
const wellKnownPath = "/.well-known/openid-configuration";
|
|
|
|
let wellKnownUrl: string;
|
|
|
|
// If the issuer URL already includes the well-known path, use it as-is
|
|
if (issuerUrl.includes(wellKnownPath)) {
|
|
wellKnownUrl = issuerUrl;
|
|
} else {
|
|
// Properly append well-known path to the issuer URL path
|
|
const url = new URL(issuerUrl);
|
|
// If pathname doesn't end with slash, append the full wellKnownPath (with leading slash)
|
|
if (!url.pathname.endsWith("/")) {
|
|
url.pathname += wellKnownPath;
|
|
} else {
|
|
// If pathname ends with slash, append wellKnownPath without leading slash
|
|
url.pathname += wellKnownPath.substring(1);
|
|
}
|
|
wellKnownUrl = url.toString();
|
|
}
|
|
|
|
Logger.info("plugins", `Fetching OIDC configuration from ${wellKnownUrl}`);
|
|
|
|
const response = await fetch(wellKnownUrl, {
|
|
method: "GET",
|
|
headers: {
|
|
Accept: "application/json",
|
|
},
|
|
timeout: 10000, // 10 second timeout
|
|
allowPrivateIPAddress: true,
|
|
});
|
|
|
|
if (!response.ok) {
|
|
throw new Error(
|
|
`Failed to fetch OIDC configuration: ${response.status} ${response.statusText}`
|
|
);
|
|
}
|
|
|
|
const config = (await response.json()) as OIDCConfiguration;
|
|
|
|
// Validate required endpoints are present
|
|
if (!config.authorization_endpoint) {
|
|
throw InternalError(
|
|
"Missing authorization_endpoint in OIDC configuration"
|
|
);
|
|
}
|
|
if (!config.token_endpoint) {
|
|
throw InternalError("Missing token_endpoint in OIDC configuration");
|
|
}
|
|
if (!config.userinfo_endpoint) {
|
|
throw InternalError("Missing userinfo_endpoint in OIDC configuration");
|
|
}
|
|
|
|
Logger.info("plugins", "Successfully fetched OIDC configuration", {
|
|
issuer: config.issuer,
|
|
authorization_endpoint: config.authorization_endpoint,
|
|
token_endpoint: config.token_endpoint,
|
|
userinfo_endpoint: config.userinfo_endpoint,
|
|
});
|
|
|
|
return config;
|
|
} catch (error) {
|
|
Logger.error("Failed to fetch OIDC configuration", error);
|
|
throw error;
|
|
}
|
|
}
|