@aura-stack/auth 0.4.0-rc.5 → 0.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/@types/index.d.ts +8 -3
- package/dist/@types/router.d.cjs +0 -17
- package/dist/@types/router.d.d.ts +7 -2
- package/dist/@types/router.d.js +0 -1
- package/dist/actions/callback/access-token.cjs +130 -71
- package/dist/actions/callback/access-token.d.ts +9 -4
- package/dist/actions/callback/access-token.js +3 -4
- package/dist/actions/callback/callback.cjs +428 -152
- package/dist/actions/callback/callback.d.ts +11 -3
- package/dist/actions/callback/callback.js +12 -10
- package/dist/actions/callback/userinfo.cjs +159 -65
- package/dist/actions/callback/userinfo.d.ts +8 -3
- package/dist/actions/callback/userinfo.js +7 -6
- package/dist/actions/csrfToken/csrfToken.cjs +70 -19
- package/dist/actions/csrfToken/csrfToken.js +8 -7
- package/dist/actions/index.cjs +780 -348
- package/dist/actions/index.d.ts +6 -2
- package/dist/actions/index.js +23 -18
- package/dist/actions/session/session.cjs +107 -26
- package/dist/actions/session/session.js +7 -5
- package/dist/actions/signIn/authorization-url.cjs +288 -0
- package/dist/actions/signIn/authorization-url.d.ts +31 -0
- package/dist/actions/signIn/authorization-url.js +16 -0
- package/dist/actions/signIn/authorization.cjs +209 -211
- package/dist/actions/signIn/authorization.d.ts +32 -21
- package/dist/actions/signIn/authorization.js +12 -9
- package/dist/actions/signIn/signIn.cjs +470 -235
- package/dist/actions/signIn/signIn.d.ts +12 -3
- package/dist/actions/signIn/signIn.js +11 -8
- package/dist/actions/signOut/signOut.cjs +376 -228
- package/dist/actions/signOut/signOut.d.ts +1 -1
- package/dist/actions/signOut/signOut.js +10 -9
- package/dist/api/createApi.cjs +750 -0
- package/dist/api/createApi.d.ts +12 -0
- package/dist/api/createApi.js +19 -0
- package/dist/api/getSession.cjs +141 -0
- package/dist/api/getSession.d.ts +16 -0
- package/dist/api/getSession.js +10 -0
- package/dist/api/signIn.cjs +549 -0
- package/dist/api/signIn.d.ts +26 -0
- package/dist/api/signIn.js +15 -0
- package/dist/api/signOut.cjs +279 -0
- package/dist/api/signOut.d.ts +16 -0
- package/dist/api/signOut.js +13 -0
- package/dist/assert.cjs +150 -5
- package/dist/assert.d.ts +26 -3
- package/dist/assert.js +17 -3
- package/dist/{chunk-YRCB5FLE.js → chunk-2A5B7GWR.js} +52 -6
- package/dist/chunk-2GQLSIJ2.js +40 -0
- package/dist/chunk-2IR674WX.js +44 -0
- package/dist/chunk-3J5TUH2I.js +50 -0
- package/dist/chunk-4RWSYUKX.js +98 -0
- package/dist/chunk-4YHJ4IEQ.js +25 -0
- package/dist/chunk-54CZPKR4.js +25 -0
- package/dist/chunk-5LZ7TOM3.js +25 -0
- package/dist/chunk-7BE46WWS.js +88 -0
- package/dist/chunk-7YYXFKLR.js +35 -0
- package/dist/chunk-C3A37LQC.js +33 -0
- package/dist/chunk-CITNGXDA.js +31 -0
- package/dist/chunk-CWX724AG.js +78 -0
- package/dist/chunk-D2CSIUKP.js +74 -0
- package/dist/chunk-E6G5YCI6.js +25 -0
- package/dist/chunk-EBAMFRB7.js +34 -0
- package/dist/chunk-EEE7UM5T.js +25 -0
- package/dist/{chunk-HT4YLL7N.js → chunk-FPCVZUVG.js} +10 -8
- package/dist/chunk-FW4W3REU.js +25 -0
- package/dist/chunk-GNNBM2WJ.js +83 -0
- package/dist/chunk-IPKO6UQN.js +25 -0
- package/dist/chunk-JOCGX3RP.js +59 -0
- package/dist/chunk-KBXWTD6E.js +94 -0
- package/dist/chunk-KMMAZFSJ.js +25 -0
- package/dist/chunk-LATR3NIV.js +117 -0
- package/dist/chunk-LAYPUDQF.js +39 -0
- package/dist/chunk-LDU7A2JE.js +25 -0
- package/dist/chunk-LX3TJ2TJ.js +294 -0
- package/dist/chunk-NHZBQNRR.js +143 -0
- package/dist/chunk-OVHNRULD.js +33 -0
- package/dist/chunk-PDP3PHB3.js +127 -0
- package/dist/chunk-PHYNROD4.js +47 -0
- package/dist/chunk-QQEKY4XP.js +29 -0
- package/dist/chunk-U4RK4LKJ.js +348 -0
- package/dist/{chunk-RRLIF4PQ.js → chunk-U5663F2U.js} +16 -1
- package/dist/chunk-UN7X6SU5.js +53 -0
- package/dist/chunk-UZQJJD6A.js +100 -0
- package/dist/chunk-V6LLEAR4.js +80 -0
- package/dist/chunk-WHNDRO3N.js +50 -0
- package/dist/{chunk-W6LG7BFW.js → chunk-XY5R3EHH.js} +30 -23
- package/dist/client/client.cjs +135 -0
- package/dist/client/client.d.ts +85 -0
- package/dist/client/client.js +9 -0
- package/dist/client/index.cjs +135 -0
- package/dist/client/index.d.ts +14 -0
- package/dist/client/index.js +10 -0
- package/dist/context.cjs +1237 -0
- package/dist/context.d.ts +16 -0
- package/dist/context.js +28 -0
- package/dist/cookie.cjs +57 -22
- package/dist/cookie.d.ts +11 -6
- package/dist/cookie.js +3 -2
- package/dist/createAuth.cjs +2320 -0
- package/dist/createAuth.d.ts +12 -0
- package/dist/createAuth.js +48 -0
- package/dist/env.cjs +78 -0
- package/dist/env.d.ts +10 -0
- package/dist/env.js +12 -0
- package/dist/errors.cjs +17 -0
- package/dist/errors.d.ts +15 -4
- package/dist/errors.js +5 -1
- package/dist/headers.cjs +28 -2
- package/dist/headers.d.ts +25 -1
- package/dist/headers.js +9 -3
- package/dist/index-_aXtxb_s.d.ts +1377 -0
- package/dist/index.cjs +1843 -610
- package/dist/index.d.ts +11 -92
- package/dist/index.js +53 -85
- package/dist/jose.cjs +113 -38
- package/dist/jose.d.ts +12 -23
- package/dist/jose.js +17 -7
- package/dist/logger.cjs +424 -0
- package/dist/logger.d.ts +12 -0
- package/dist/logger.js +17 -0
- package/dist/oauth/atlassian.cjs +57 -0
- package/dist/oauth/atlassian.d.ts +12 -0
- package/dist/oauth/atlassian.js +6 -0
- package/dist/oauth/bitbucket.cjs +19 -15
- package/dist/oauth/bitbucket.d.ts +7 -2
- package/dist/oauth/bitbucket.js +1 -1
- package/dist/oauth/discord.cjs +27 -24
- package/dist/oauth/discord.d.ts +7 -2
- package/dist/oauth/discord.js +1 -1
- package/dist/oauth/dropbox.cjs +53 -0
- package/dist/oauth/dropbox.d.ts +12 -0
- package/dist/oauth/dropbox.js +6 -0
- package/dist/oauth/figma.cjs +19 -16
- package/dist/oauth/figma.d.ts +7 -2
- package/dist/oauth/figma.js +1 -1
- package/dist/oauth/github.cjs +19 -8
- package/dist/oauth/github.d.ts +7 -2
- package/dist/oauth/github.js +1 -1
- package/dist/oauth/gitlab.cjs +19 -16
- package/dist/oauth/gitlab.d.ts +7 -2
- package/dist/oauth/gitlab.js +1 -1
- package/dist/oauth/index.cjs +529 -239
- package/dist/oauth/index.d.ts +7 -2
- package/dist/oauth/index.js +39 -22
- package/dist/oauth/mailchimp.cjs +19 -16
- package/dist/oauth/mailchimp.d.ts +7 -2
- package/dist/oauth/mailchimp.js +1 -1
- package/dist/oauth/notion.cjs +131 -0
- package/dist/oauth/notion.d.ts +12 -0
- package/dist/oauth/notion.js +9 -0
- package/dist/oauth/pinterest.cjs +19 -16
- package/dist/oauth/pinterest.d.ts +7 -2
- package/dist/oauth/pinterest.js +1 -1
- package/dist/oauth/spotify.cjs +19 -16
- package/dist/oauth/spotify.d.ts +7 -2
- package/dist/oauth/spotify.js +1 -1
- package/dist/oauth/strava.cjs +19 -16
- package/dist/oauth/strava.d.ts +7 -2
- package/dist/oauth/strava.js +1 -1
- package/dist/oauth/twitch.cjs +95 -0
- package/dist/oauth/twitch.d.ts +12 -0
- package/dist/oauth/twitch.js +7 -0
- package/dist/oauth/x.cjs +19 -16
- package/dist/oauth/x.d.ts +7 -2
- package/dist/oauth/x.js +1 -1
- package/dist/schemas.cjs +89 -42
- package/dist/schemas.d.ts +114 -18
- package/dist/schemas.js +5 -3
- package/dist/secure.cjs +73 -31
- package/dist/secure.d.ts +11 -11
- package/dist/secure.js +7 -6
- package/dist/utils.cjs +203 -90
- package/dist/utils.d.ts +21 -40
- package/dist/utils.js +21 -12
- package/package.json +9 -6
- package/dist/chunk-3EUWD5BB.js +0 -63
- package/dist/chunk-42XB3YCW.js +0 -22
- package/dist/chunk-6R2YZ4AC.js +0 -22
- package/dist/chunk-A3N4PVAT.js +0 -70
- package/dist/chunk-B737EUJV.js +0 -22
- package/dist/chunk-CXLATHS5.js +0 -143
- package/dist/chunk-E3OXBRYF.js +0 -22
- package/dist/chunk-EIL2FPSS.js +0 -22
- package/dist/chunk-EMKJA2GJ.js +0 -89
- package/dist/chunk-FIPU4MLT.js +0 -21
- package/dist/chunk-FKRDCWBF.js +0 -22
- package/dist/chunk-GA2SMTJO.js +0 -58
- package/dist/chunk-HP34YGGJ.js +0 -22
- package/dist/chunk-IKHPGFCW.js +0 -14
- package/dist/chunk-IUYZQTJV.js +0 -30
- package/dist/chunk-IVET23KF.js +0 -58
- package/dist/chunk-JVFTCTTE.js +0 -33
- package/dist/chunk-KRNOMBXQ.js +0 -22
- package/dist/chunk-KSWLO5ZU.js +0 -102
- package/dist/chunk-N2APGLXA.js +0 -71
- package/dist/chunk-N4SX7TZT.js +0 -96
- package/dist/chunk-STHEPPUZ.js +0 -11
- package/dist/chunk-TLE4PXY3.js +0 -39
- package/dist/index-B8jeIElf.d.ts +0 -679
- /package/dist/{chunk-DIVDFNAP.js → chunk-5X7JZMEF.js} +0 -0
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
// src/oauth/x.ts
|
|
2
|
+
var x = (options) => {
|
|
3
|
+
return {
|
|
4
|
+
id: "x",
|
|
5
|
+
name: "X",
|
|
6
|
+
authorizeURL: "https://twitter.com/i/oauth2/authorize",
|
|
7
|
+
accessToken: "https://api.twitter.com/2/oauth2/token",
|
|
8
|
+
userInfo: "https://api.twitter.com/2/users/me?user.fields=profile_image_url",
|
|
9
|
+
scope: "tweet.read users.read offline.access",
|
|
10
|
+
responseType: "code",
|
|
11
|
+
profile(profile) {
|
|
12
|
+
return {
|
|
13
|
+
sub: profile.data.id,
|
|
14
|
+
name: profile.data.name,
|
|
15
|
+
image: profile.data.profile_image_url,
|
|
16
|
+
email: void 0
|
|
17
|
+
};
|
|
18
|
+
},
|
|
19
|
+
...options
|
|
20
|
+
};
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
export {
|
|
24
|
+
x
|
|
25
|
+
};
|
|
@@ -1,13 +1,13 @@
|
|
|
1
|
+
import {
|
|
2
|
+
createCSRF
|
|
3
|
+
} from "./chunk-V6LLEAR4.js";
|
|
1
4
|
import {
|
|
2
5
|
getCookie,
|
|
3
6
|
setCookie
|
|
4
|
-
} from "./chunk-
|
|
7
|
+
} from "./chunk-XY5R3EHH.js";
|
|
5
8
|
import {
|
|
6
|
-
|
|
7
|
-
} from "./chunk-
|
|
8
|
-
import {
|
|
9
|
-
createCSRF
|
|
10
|
-
} from "./chunk-N2APGLXA.js";
|
|
9
|
+
secureApiHeaders
|
|
10
|
+
} from "./chunk-EBAMFRB7.js";
|
|
11
11
|
|
|
12
12
|
// src/actions/csrfToken/csrfToken.ts
|
|
13
13
|
import { createEndpoint } from "@aura-stack/router";
|
|
@@ -21,11 +21,13 @@ var getCSRFToken = (request, cookieName) => {
|
|
|
21
21
|
var csrfTokenAction = createEndpoint("GET", "/csrfToken", async (ctx) => {
|
|
22
22
|
const {
|
|
23
23
|
request,
|
|
24
|
-
context: { jose, cookies }
|
|
24
|
+
context: { jose, cookies, logger }
|
|
25
25
|
} = ctx;
|
|
26
26
|
const token = getCSRFToken(request, cookies.csrfToken.name);
|
|
27
|
+
logger?.log("CSRF_TOKEN_REQUESTED", { structuredData: { has_token: Boolean(token) } });
|
|
27
28
|
const csrfToken = await createCSRF(jose, token);
|
|
28
|
-
|
|
29
|
+
logger?.log("CSRF_TOKEN_ISSUED", { structuredData: { issued: Boolean(csrfToken) } });
|
|
30
|
+
const headers = new Headers(secureApiHeaders);
|
|
29
31
|
headers.append("Set-Cookie", setCookie(cookies.csrfToken.name, csrfToken, cookies.csrfToken.attributes));
|
|
30
32
|
return Response.json({ csrfToken }, { headers });
|
|
31
33
|
});
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
// src/oauth/github.ts
|
|
2
|
+
var github = (options) => {
|
|
3
|
+
return {
|
|
4
|
+
id: "github",
|
|
5
|
+
name: "GitHub",
|
|
6
|
+
authorizeURL: "https://github.com/login/oauth/authorize",
|
|
7
|
+
accessToken: "https://github.com/login/oauth/access_token",
|
|
8
|
+
userInfo: "https://api.github.com/user",
|
|
9
|
+
scope: "read:user user:email",
|
|
10
|
+
responseType: "code",
|
|
11
|
+
profile: (profile) => {
|
|
12
|
+
return {
|
|
13
|
+
sub: profile.id.toString(),
|
|
14
|
+
name: profile.name ?? profile.login,
|
|
15
|
+
email: profile.email ?? void 0,
|
|
16
|
+
image: profile.avatar_url
|
|
17
|
+
};
|
|
18
|
+
},
|
|
19
|
+
...options
|
|
20
|
+
};
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
export {
|
|
24
|
+
github
|
|
25
|
+
};
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import {
|
|
2
|
+
fetchAsync
|
|
3
|
+
} from "./chunk-ZNCZVF6U.js";
|
|
4
|
+
import {
|
|
5
|
+
generateSecure
|
|
6
|
+
} from "./chunk-V6LLEAR4.js";
|
|
7
|
+
import {
|
|
8
|
+
OAuthErrorResponse
|
|
9
|
+
} from "./chunk-2A5B7GWR.js";
|
|
10
|
+
import {
|
|
11
|
+
AURA_AUTH_VERSION
|
|
12
|
+
} from "./chunk-LX3TJ2TJ.js";
|
|
13
|
+
import {
|
|
14
|
+
OAuthProtocolError,
|
|
15
|
+
isNativeError,
|
|
16
|
+
isOAuthProtocolError
|
|
17
|
+
} from "./chunk-U5663F2U.js";
|
|
18
|
+
|
|
19
|
+
// src/actions/callback/userinfo.ts
|
|
20
|
+
var getDefaultUserInfo = (profile) => {
|
|
21
|
+
const sub = generateSecure(16);
|
|
22
|
+
return {
|
|
23
|
+
sub: profile?.id ?? profile?.sub ?? sub,
|
|
24
|
+
email: profile?.email,
|
|
25
|
+
name: profile?.name ?? profile?.username ?? profile?.nickname,
|
|
26
|
+
image: profile?.image ?? profile?.picture
|
|
27
|
+
};
|
|
28
|
+
};
|
|
29
|
+
var getUserInfo = async (oauthConfig, accessToken, logger) => {
|
|
30
|
+
const userInfoConfig = oauthConfig.userInfo;
|
|
31
|
+
const userinfoURL = typeof userInfoConfig === "string" ? userInfoConfig : userInfoConfig.url;
|
|
32
|
+
const extraHeaders = typeof userInfoConfig === "string" ? void 0 : userInfoConfig.headers;
|
|
33
|
+
const method = typeof userInfoConfig === "string" ? "GET" : (userInfoConfig.method ?? "GET").toUpperCase();
|
|
34
|
+
try {
|
|
35
|
+
logger?.log("OAUTH_USERINFO_REQUEST_INITIATED", {
|
|
36
|
+
structuredData: {
|
|
37
|
+
endpoint: userinfoURL
|
|
38
|
+
}
|
|
39
|
+
});
|
|
40
|
+
const response = await fetchAsync(userinfoURL, {
|
|
41
|
+
method,
|
|
42
|
+
headers: {
|
|
43
|
+
"User-Agent": `Aura Auth/${AURA_AUTH_VERSION}`,
|
|
44
|
+
Accept: "application/json",
|
|
45
|
+
Authorization: `Bearer ${accessToken}`,
|
|
46
|
+
...extraHeaders ?? {}
|
|
47
|
+
}
|
|
48
|
+
});
|
|
49
|
+
if (!response.ok) {
|
|
50
|
+
logger?.log("OAUTH_USERINFO_INVALID_RESPONSE");
|
|
51
|
+
throw new OAuthProtocolError("INVALID_REQUEST", "Invalid userinfo response format");
|
|
52
|
+
}
|
|
53
|
+
const json = await response.json();
|
|
54
|
+
const { success, data } = OAuthErrorResponse.safeParse(json);
|
|
55
|
+
if (success) {
|
|
56
|
+
logger?.log("OAUTH_USERINFO_ERROR", {
|
|
57
|
+
message: "Error response received from OAuth userinfo endpoint",
|
|
58
|
+
structuredData: {
|
|
59
|
+
error: data.error,
|
|
60
|
+
error_description: data.error_description ?? ""
|
|
61
|
+
}
|
|
62
|
+
});
|
|
63
|
+
throw new OAuthProtocolError("INVALID_REQUEST", "An error was received from the OAuth userinfo endpoint.");
|
|
64
|
+
}
|
|
65
|
+
logger?.log("OAUTH_USERINFO_SUCCESS");
|
|
66
|
+
return oauthConfig?.profile ? oauthConfig.profile(json) : getDefaultUserInfo(json);
|
|
67
|
+
} catch (error) {
|
|
68
|
+
if (isOAuthProtocolError(error)) {
|
|
69
|
+
throw error;
|
|
70
|
+
}
|
|
71
|
+
logger?.log("OAUTH_USERINFO_REQUEST_FAILED");
|
|
72
|
+
if (isNativeError(error)) {
|
|
73
|
+
throw new OAuthProtocolError("SERVER_ERROR", "Failed to fetch user information from OAuth provider", "", {
|
|
74
|
+
cause: error
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
throw new OAuthProtocolError("SERVER_ERROR", "Failed to fetch user information", "", { cause: error });
|
|
78
|
+
}
|
|
79
|
+
};
|
|
80
|
+
|
|
81
|
+
export {
|
|
82
|
+
getUserInfo
|
|
83
|
+
};
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
// src/oauth/spotify.ts
|
|
2
|
+
var spotify = (options) => {
|
|
3
|
+
return {
|
|
4
|
+
id: "spotify",
|
|
5
|
+
name: "Spotify",
|
|
6
|
+
authorizeURL: "https://accounts.spotify.com/authorize",
|
|
7
|
+
accessToken: "https://accounts.spotify.com/api/token",
|
|
8
|
+
userInfo: "https://api.spotify.com/v1/me",
|
|
9
|
+
scope: "user-read-private user-read-email",
|
|
10
|
+
responseType: "code",
|
|
11
|
+
profile(profile) {
|
|
12
|
+
return {
|
|
13
|
+
sub: profile.id,
|
|
14
|
+
name: profile.display_name,
|
|
15
|
+
email: profile.email,
|
|
16
|
+
image: profile.images[0]?.url ?? void 0
|
|
17
|
+
};
|
|
18
|
+
},
|
|
19
|
+
...options
|
|
20
|
+
};
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
export {
|
|
24
|
+
spotify
|
|
25
|
+
};
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import {
|
|
2
|
+
createAuthorizationURL
|
|
3
|
+
} from "./chunk-D2CSIUKP.js";
|
|
4
|
+
import {
|
|
5
|
+
createRedirectTo,
|
|
6
|
+
createRedirectURI,
|
|
7
|
+
createSignInURL,
|
|
8
|
+
getBaseURL
|
|
9
|
+
} from "./chunk-LATR3NIV.js";
|
|
10
|
+
import {
|
|
11
|
+
AuthInternalError
|
|
12
|
+
} from "./chunk-U5663F2U.js";
|
|
13
|
+
import {
|
|
14
|
+
cacheControl
|
|
15
|
+
} from "./chunk-EBAMFRB7.js";
|
|
16
|
+
|
|
17
|
+
// src/api/signIn.ts
|
|
18
|
+
import { HeadersBuilder } from "@aura-stack/router";
|
|
19
|
+
var signIn = async (oauth, {
|
|
20
|
+
ctx,
|
|
21
|
+
headers: headersInit,
|
|
22
|
+
redirectTo = "/",
|
|
23
|
+
redirect,
|
|
24
|
+
request: requestInit
|
|
25
|
+
}) => {
|
|
26
|
+
const headers = new Headers(headersInit);
|
|
27
|
+
const provider = ctx.oauth[oauth];
|
|
28
|
+
if (!provider) {
|
|
29
|
+
throw new AuthInternalError("INVALID_OAUTH_CONFIGURATION", `The OAuth provider "${oauth}" is not configured.`);
|
|
30
|
+
}
|
|
31
|
+
let request = requestInit;
|
|
32
|
+
if (!request) {
|
|
33
|
+
const origin = await getBaseURL({ ctx, headers });
|
|
34
|
+
const url = `${origin}${ctx.basePath}/signIn/${oauth}`;
|
|
35
|
+
request = new Request(url, { headers });
|
|
36
|
+
}
|
|
37
|
+
if (redirect === false) {
|
|
38
|
+
const signInURL = await createSignInURL({ request, oauth, ctx, redirectTo });
|
|
39
|
+
return { redirect: false, signInURL };
|
|
40
|
+
}
|
|
41
|
+
const redirectURI = await createRedirectURI(request, oauth, ctx);
|
|
42
|
+
const redirectToValue = await createRedirectTo(request, redirectTo, ctx);
|
|
43
|
+
const { authorization, state, codeVerifier } = await createAuthorizationURL(provider, redirectURI, ctx);
|
|
44
|
+
ctx?.logger?.log("SIGN_IN_INITIATED", {
|
|
45
|
+
structuredData: { oauth_provider: oauth }
|
|
46
|
+
});
|
|
47
|
+
const headersList = new HeadersBuilder(cacheControl).setHeader("Location", authorization).setCookie(ctx.cookies.state.name, state, ctx.cookies.state.attributes).setCookie(ctx.cookies.redirectURI.name, redirectURI, ctx.cookies.redirectURI.attributes).setCookie(ctx.cookies.redirectTo.name, redirectToValue, ctx.cookies.redirectTo.attributes).setCookie(ctx.cookies.codeVerifier.name, codeVerifier, ctx.cookies.codeVerifier.attributes).toHeaders();
|
|
48
|
+
return Response.json(
|
|
49
|
+
{ redirect: redirect ?? true, signInURL: authorization },
|
|
50
|
+
{
|
|
51
|
+
status: redirect ?? true ? 302 : 200,
|
|
52
|
+
headers: headersList
|
|
53
|
+
}
|
|
54
|
+
);
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
export {
|
|
58
|
+
signIn
|
|
59
|
+
};
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
import {
|
|
2
|
+
verifyCSRF
|
|
3
|
+
} from "./chunk-V6LLEAR4.js";
|
|
4
|
+
import {
|
|
5
|
+
getErrorName
|
|
6
|
+
} from "./chunk-LX3TJ2TJ.js";
|
|
7
|
+
import {
|
|
8
|
+
expiredCookieAttributes,
|
|
9
|
+
getCookie
|
|
10
|
+
} from "./chunk-XY5R3EHH.js";
|
|
11
|
+
import {
|
|
12
|
+
AuthSecurityError
|
|
13
|
+
} from "./chunk-U5663F2U.js";
|
|
14
|
+
import {
|
|
15
|
+
secureApiHeaders
|
|
16
|
+
} from "./chunk-EBAMFRB7.js";
|
|
17
|
+
|
|
18
|
+
// src/api/signOut.ts
|
|
19
|
+
import { HeadersBuilder } from "@aura-stack/router";
|
|
20
|
+
var signOut = async ({
|
|
21
|
+
ctx,
|
|
22
|
+
headers: headersInit,
|
|
23
|
+
redirectTo = "/",
|
|
24
|
+
skipCSRFCheck = false
|
|
25
|
+
}) => {
|
|
26
|
+
const headers = new Headers(headersInit);
|
|
27
|
+
const header = headers.get("X-CSRF-Token");
|
|
28
|
+
let session = null;
|
|
29
|
+
let csrfToken = null;
|
|
30
|
+
try {
|
|
31
|
+
session = getCookie(headers, ctx.cookies.sessionToken.name);
|
|
32
|
+
} catch {
|
|
33
|
+
throw new AuthSecurityError("SESSION_TOKEN_MISSING", "The sessionToken is missing.");
|
|
34
|
+
}
|
|
35
|
+
try {
|
|
36
|
+
csrfToken = getCookie(headers, ctx.cookies.csrfToken.name);
|
|
37
|
+
} catch {
|
|
38
|
+
throw new AuthSecurityError("CSRF_TOKEN_MISSING", "The CSRF token is missing.");
|
|
39
|
+
}
|
|
40
|
+
ctx?.logger?.log("SIGN_OUT_ATTEMPT", {
|
|
41
|
+
structuredData: {
|
|
42
|
+
has_session: Boolean(session),
|
|
43
|
+
has_csrf_token: Boolean(csrfToken),
|
|
44
|
+
has_csrf_header: Boolean(header),
|
|
45
|
+
skip_csrf_check: skipCSRFCheck
|
|
46
|
+
}
|
|
47
|
+
});
|
|
48
|
+
if (!session) {
|
|
49
|
+
ctx?.logger?.log("SESSION_TOKEN_MISSING");
|
|
50
|
+
throw new AuthSecurityError("SESSION_TOKEN_MISSING", "The sessionToken is missing.");
|
|
51
|
+
}
|
|
52
|
+
if (!skipCSRFCheck) {
|
|
53
|
+
if (!csrfToken) {
|
|
54
|
+
ctx?.logger?.log("CSRF_TOKEN_MISSING");
|
|
55
|
+
throw new AuthSecurityError("CSRF_TOKEN_MISSING", "The CSRF token is missing.");
|
|
56
|
+
}
|
|
57
|
+
if (!header) {
|
|
58
|
+
ctx?.logger?.log("CSRF_HEADER_MISSING");
|
|
59
|
+
throw new AuthSecurityError("CSRF_HEADER_MISSING", "The CSRF header is missing.");
|
|
60
|
+
}
|
|
61
|
+
try {
|
|
62
|
+
await verifyCSRF(ctx.jose, csrfToken, header);
|
|
63
|
+
} catch (error) {
|
|
64
|
+
ctx?.logger?.log("CSRF_TOKEN_INVALID", { structuredData: { error_type: getErrorName(error) } });
|
|
65
|
+
throw new AuthSecurityError("CSRF_TOKEN_INVALID", "CSRF token verification failed");
|
|
66
|
+
}
|
|
67
|
+
ctx?.logger?.log("SIGN_OUT_CSRF_VERIFIED");
|
|
68
|
+
} else {
|
|
69
|
+
try {
|
|
70
|
+
await ctx.jose.verifyJWS(csrfToken);
|
|
71
|
+
} catch (error) {
|
|
72
|
+
ctx?.logger?.log("CSRF_TOKEN_INVALID", { structuredData: { error_type: getErrorName(error) } });
|
|
73
|
+
throw new AuthSecurityError("CSRF_TOKEN_INVALID", "CSRF token verification failed");
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
try {
|
|
77
|
+
await ctx.jose.decodeJWT(session);
|
|
78
|
+
ctx?.logger?.log("SIGN_OUT_SUCCESS");
|
|
79
|
+
} catch (error) {
|
|
80
|
+
ctx?.logger?.log("INVALID_JWT_TOKEN", { structuredData: { error_type: getErrorName(error) } });
|
|
81
|
+
}
|
|
82
|
+
const headersList = new HeadersBuilder(secureApiHeaders).setHeader("Location", redirectTo).setCookie(ctx.cookies.csrfToken.name, "", expiredCookieAttributes).setCookie(ctx.cookies.sessionToken.name, "", expiredCookieAttributes).toHeaders();
|
|
83
|
+
return Response.json(
|
|
84
|
+
{ redirect: Boolean(redirectTo), url: redirectTo },
|
|
85
|
+
{
|
|
86
|
+
status: 202,
|
|
87
|
+
headers: headersList
|
|
88
|
+
}
|
|
89
|
+
);
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
export {
|
|
93
|
+
signOut
|
|
94
|
+
};
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
// src/oauth/figma.ts
|
|
2
|
+
var figma = (options) => {
|
|
3
|
+
return {
|
|
4
|
+
id: "figma",
|
|
5
|
+
name: "Figma",
|
|
6
|
+
authorizeURL: "https://www.figma.com/oauth",
|
|
7
|
+
accessToken: "https://api.figma.com/v1/oauth/token",
|
|
8
|
+
userInfo: "https://api.figma.com/v1/me",
|
|
9
|
+
scope: "current_user:read",
|
|
10
|
+
responseType: "code",
|
|
11
|
+
profile(profile) {
|
|
12
|
+
return {
|
|
13
|
+
sub: profile.id,
|
|
14
|
+
name: profile.handle,
|
|
15
|
+
email: profile.email,
|
|
16
|
+
image: profile.img_url
|
|
17
|
+
};
|
|
18
|
+
},
|
|
19
|
+
...options
|
|
20
|
+
};
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
export {
|
|
24
|
+
figma
|
|
25
|
+
};
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
import {
|
|
2
|
+
equals,
|
|
3
|
+
extractPath,
|
|
4
|
+
isRelativeURL,
|
|
5
|
+
isSameOrigin,
|
|
6
|
+
isTrustedOrigin,
|
|
7
|
+
isValidURL,
|
|
8
|
+
patternToRegex
|
|
9
|
+
} from "./chunk-LX3TJ2TJ.js";
|
|
10
|
+
import {
|
|
11
|
+
getEnv
|
|
12
|
+
} from "./chunk-WHNDRO3N.js";
|
|
13
|
+
import {
|
|
14
|
+
AuthInternalError
|
|
15
|
+
} from "./chunk-U5663F2U.js";
|
|
16
|
+
|
|
17
|
+
// src/actions/signIn/authorization.ts
|
|
18
|
+
var getTrustedOrigins = async (request, trustedOrigins) => {
|
|
19
|
+
if (!trustedOrigins) return [];
|
|
20
|
+
const raw = typeof trustedOrigins === "function" ? await trustedOrigins(request) : trustedOrigins;
|
|
21
|
+
return Array.isArray(raw) ? raw : typeof raw === "string" ? [raw] : [];
|
|
22
|
+
};
|
|
23
|
+
var getBaseURL = async ({
|
|
24
|
+
ctx,
|
|
25
|
+
request,
|
|
26
|
+
headers: headersInit
|
|
27
|
+
}) => {
|
|
28
|
+
const origin = getEnv("BASE_URL") || ctx?.baseURL;
|
|
29
|
+
if (origin && origin !== "/") return origin;
|
|
30
|
+
if (ctx?.trustedProxyHeaders) {
|
|
31
|
+
const headers = headersInit && new Headers(headersInit) || request?.headers;
|
|
32
|
+
const protocol = headers?.get("Forwarded")?.match(/proto=([^;]+)/i)?.[1] ?? headers?.get("X-Forwarded-Proto") ?? "http";
|
|
33
|
+
const host = headers?.get("Host") ?? headers?.get("Forwarded")?.match(/host=([^;]+)/i)?.[1] ?? headers?.get("X-Forwarded-Host") ?? null;
|
|
34
|
+
if (host) return `${protocol}://${host}`;
|
|
35
|
+
throw new AuthInternalError(
|
|
36
|
+
"INVALID_OAUTH_CONFIGURATION",
|
|
37
|
+
"The URL cannot be constructed. Please set the BASE_URL environment variable or provide trusted proxy host headers."
|
|
38
|
+
);
|
|
39
|
+
}
|
|
40
|
+
try {
|
|
41
|
+
return new URL(request?.url ?? "not-found").origin;
|
|
42
|
+
} catch (error) {
|
|
43
|
+
throw new AuthInternalError(
|
|
44
|
+
"INVALID_OAUTH_CONFIGURATION",
|
|
45
|
+
"The URL cannot be constructed. Please set the BASE_URL environment variable or enable trustedProxyHeaders.",
|
|
46
|
+
{ cause: error }
|
|
47
|
+
);
|
|
48
|
+
}
|
|
49
|
+
};
|
|
50
|
+
var getOriginURL = async (request, context) => {
|
|
51
|
+
const trustedOrigins = await getTrustedOrigins(request, context?.trustedOrigins);
|
|
52
|
+
trustedOrigins.push(new URL(request.url).origin);
|
|
53
|
+
const origin = await getBaseURL({ request, ctx: context });
|
|
54
|
+
if (!isTrustedOrigin(origin, trustedOrigins)) {
|
|
55
|
+
context?.logger?.log("UNTRUSTED_ORIGIN", { structuredData: { origin } });
|
|
56
|
+
throw new AuthInternalError("UNTRUSTED_ORIGIN", "The constructed origin URL is not trusted.");
|
|
57
|
+
}
|
|
58
|
+
return origin;
|
|
59
|
+
};
|
|
60
|
+
var createRedirectURI = async (request, oauth, context) => {
|
|
61
|
+
const origin = await getOriginURL(request, context);
|
|
62
|
+
return `${origin}${context.basePath}/callback/${oauth}`;
|
|
63
|
+
};
|
|
64
|
+
var createSignInURL = async ({
|
|
65
|
+
request,
|
|
66
|
+
oauth,
|
|
67
|
+
ctx,
|
|
68
|
+
redirectTo
|
|
69
|
+
}) => {
|
|
70
|
+
const origin = await getOriginURL(request, ctx);
|
|
71
|
+
const searchParams = new URLSearchParams();
|
|
72
|
+
if (redirectTo !== void 0) searchParams.set("redirectTo", String(redirectTo));
|
|
73
|
+
return `${origin}${ctx.basePath}/signIn/${oauth}?${searchParams.toString()}`;
|
|
74
|
+
};
|
|
75
|
+
var createRedirectTo = async (request, redirectTo, context) => {
|
|
76
|
+
try {
|
|
77
|
+
const headers = request.headers;
|
|
78
|
+
const requestOrigin = await getOriginURL(request, context);
|
|
79
|
+
const origins = await getTrustedOrigins(request, context?.trustedOrigins);
|
|
80
|
+
const validateURL = (url) => {
|
|
81
|
+
if (!isRelativeURL(url) && !isValidURL(url)) return "/";
|
|
82
|
+
if (isRelativeURL(url)) return url;
|
|
83
|
+
if (origins.length > 0) {
|
|
84
|
+
if (isTrustedOrigin(url, origins)) {
|
|
85
|
+
const urlOrigin = new URL(url).origin;
|
|
86
|
+
for (const pattern of origins) {
|
|
87
|
+
const regex = patternToRegex(pattern);
|
|
88
|
+
if (regex?.test(urlOrigin)) {
|
|
89
|
+
return isSameOrigin(url, request.url) ? extractPath(url) : url;
|
|
90
|
+
}
|
|
91
|
+
if (isValidURL(pattern) && equals(new URL(pattern).origin, urlOrigin)) return url;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
context?.logger?.log("OPEN_REDIRECT_ATTACK");
|
|
95
|
+
return "/";
|
|
96
|
+
}
|
|
97
|
+
if (isSameOrigin(url, requestOrigin)) {
|
|
98
|
+
return extractPath(url);
|
|
99
|
+
}
|
|
100
|
+
context?.logger?.log("OPEN_REDIRECT_ATTACK");
|
|
101
|
+
return "/";
|
|
102
|
+
};
|
|
103
|
+
return validateURL(redirectTo ?? headers.get("Referer") ?? headers.get("Origin") ?? "/");
|
|
104
|
+
} catch (error) {
|
|
105
|
+
context?.logger?.log("POTENTIAL_OPEN_REDIRECT_ATTACK_DETECTED");
|
|
106
|
+
return "/";
|
|
107
|
+
}
|
|
108
|
+
};
|
|
109
|
+
|
|
110
|
+
export {
|
|
111
|
+
getTrustedOrigins,
|
|
112
|
+
getBaseURL,
|
|
113
|
+
getOriginURL,
|
|
114
|
+
createRedirectURI,
|
|
115
|
+
createSignInURL,
|
|
116
|
+
createRedirectTo
|
|
117
|
+
};
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import {
|
|
2
|
+
getSession
|
|
3
|
+
} from "./chunk-CITNGXDA.js";
|
|
4
|
+
import {
|
|
5
|
+
signIn
|
|
6
|
+
} from "./chunk-JOCGX3RP.js";
|
|
7
|
+
import {
|
|
8
|
+
signOut
|
|
9
|
+
} from "./chunk-KBXWTD6E.js";
|
|
10
|
+
import {
|
|
11
|
+
validateRedirectTo
|
|
12
|
+
} from "./chunk-LX3TJ2TJ.js";
|
|
13
|
+
|
|
14
|
+
// src/api/createApi.ts
|
|
15
|
+
var createAuthAPI = (ctx) => {
|
|
16
|
+
return {
|
|
17
|
+
getSession: async (options) => {
|
|
18
|
+
const session = await getSession({ ctx, headers: options.headers });
|
|
19
|
+
return session;
|
|
20
|
+
},
|
|
21
|
+
signIn: async (oauth, options) => {
|
|
22
|
+
return signIn(oauth, {
|
|
23
|
+
ctx,
|
|
24
|
+
headers: options?.headers,
|
|
25
|
+
request: options?.request,
|
|
26
|
+
redirect: options?.redirect,
|
|
27
|
+
redirectTo: options?.redirectTo
|
|
28
|
+
});
|
|
29
|
+
},
|
|
30
|
+
signOut: async (options) => {
|
|
31
|
+
const redirectTo = validateRedirectTo(options?.redirectTo ?? "/");
|
|
32
|
+
return signOut({ ctx, headers: options.headers, redirectTo, skipCSRFCheck: true });
|
|
33
|
+
}
|
|
34
|
+
};
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
export {
|
|
38
|
+
createAuthAPI
|
|
39
|
+
};
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
// src/oauth/mailchimp.ts
|
|
2
|
+
var mailchimp = (options) => {
|
|
3
|
+
return {
|
|
4
|
+
id: "mailchimp",
|
|
5
|
+
name: "Mailchimp",
|
|
6
|
+
authorizeURL: "https://login.mailchimp.com/oauth2/authorize",
|
|
7
|
+
accessToken: "https://login.mailchimp.com/oauth2/token",
|
|
8
|
+
userInfo: "https://login.mailchimp.com/oauth2/metadata",
|
|
9
|
+
scope: "",
|
|
10
|
+
responseType: "code",
|
|
11
|
+
profile(profile) {
|
|
12
|
+
return {
|
|
13
|
+
sub: profile.user_id,
|
|
14
|
+
name: profile.accountname,
|
|
15
|
+
email: profile.login.email,
|
|
16
|
+
image: profile.login.avatar
|
|
17
|
+
};
|
|
18
|
+
},
|
|
19
|
+
...options
|
|
20
|
+
};
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
export {
|
|
24
|
+
mailchimp
|
|
25
|
+
};
|