@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.
Files changed (201) hide show
  1. package/dist/@types/index.d.ts +8 -3
  2. package/dist/@types/router.d.cjs +0 -17
  3. package/dist/@types/router.d.d.ts +7 -2
  4. package/dist/@types/router.d.js +0 -1
  5. package/dist/actions/callback/access-token.cjs +130 -71
  6. package/dist/actions/callback/access-token.d.ts +9 -4
  7. package/dist/actions/callback/access-token.js +3 -4
  8. package/dist/actions/callback/callback.cjs +428 -152
  9. package/dist/actions/callback/callback.d.ts +11 -3
  10. package/dist/actions/callback/callback.js +12 -10
  11. package/dist/actions/callback/userinfo.cjs +159 -65
  12. package/dist/actions/callback/userinfo.d.ts +8 -3
  13. package/dist/actions/callback/userinfo.js +7 -6
  14. package/dist/actions/csrfToken/csrfToken.cjs +70 -19
  15. package/dist/actions/csrfToken/csrfToken.js +8 -7
  16. package/dist/actions/index.cjs +780 -348
  17. package/dist/actions/index.d.ts +6 -2
  18. package/dist/actions/index.js +23 -18
  19. package/dist/actions/session/session.cjs +107 -26
  20. package/dist/actions/session/session.js +7 -5
  21. package/dist/actions/signIn/authorization-url.cjs +288 -0
  22. package/dist/actions/signIn/authorization-url.d.ts +31 -0
  23. package/dist/actions/signIn/authorization-url.js +16 -0
  24. package/dist/actions/signIn/authorization.cjs +209 -211
  25. package/dist/actions/signIn/authorization.d.ts +32 -21
  26. package/dist/actions/signIn/authorization.js +12 -9
  27. package/dist/actions/signIn/signIn.cjs +470 -235
  28. package/dist/actions/signIn/signIn.d.ts +12 -3
  29. package/dist/actions/signIn/signIn.js +11 -8
  30. package/dist/actions/signOut/signOut.cjs +376 -228
  31. package/dist/actions/signOut/signOut.d.ts +1 -1
  32. package/dist/actions/signOut/signOut.js +10 -9
  33. package/dist/api/createApi.cjs +750 -0
  34. package/dist/api/createApi.d.ts +12 -0
  35. package/dist/api/createApi.js +19 -0
  36. package/dist/api/getSession.cjs +141 -0
  37. package/dist/api/getSession.d.ts +16 -0
  38. package/dist/api/getSession.js +10 -0
  39. package/dist/api/signIn.cjs +549 -0
  40. package/dist/api/signIn.d.ts +26 -0
  41. package/dist/api/signIn.js +15 -0
  42. package/dist/api/signOut.cjs +279 -0
  43. package/dist/api/signOut.d.ts +16 -0
  44. package/dist/api/signOut.js +13 -0
  45. package/dist/assert.cjs +150 -5
  46. package/dist/assert.d.ts +26 -3
  47. package/dist/assert.js +17 -3
  48. package/dist/{chunk-YRCB5FLE.js → chunk-2A5B7GWR.js} +52 -6
  49. package/dist/chunk-2GQLSIJ2.js +40 -0
  50. package/dist/chunk-2IR674WX.js +44 -0
  51. package/dist/chunk-3J5TUH2I.js +50 -0
  52. package/dist/chunk-4RWSYUKX.js +98 -0
  53. package/dist/chunk-4YHJ4IEQ.js +25 -0
  54. package/dist/chunk-54CZPKR4.js +25 -0
  55. package/dist/chunk-5LZ7TOM3.js +25 -0
  56. package/dist/chunk-7BE46WWS.js +88 -0
  57. package/dist/chunk-7YYXFKLR.js +35 -0
  58. package/dist/chunk-C3A37LQC.js +33 -0
  59. package/dist/chunk-CITNGXDA.js +31 -0
  60. package/dist/chunk-CWX724AG.js +78 -0
  61. package/dist/chunk-D2CSIUKP.js +74 -0
  62. package/dist/chunk-E6G5YCI6.js +25 -0
  63. package/dist/chunk-EBAMFRB7.js +34 -0
  64. package/dist/chunk-EEE7UM5T.js +25 -0
  65. package/dist/{chunk-HT4YLL7N.js → chunk-FPCVZUVG.js} +10 -8
  66. package/dist/chunk-FW4W3REU.js +25 -0
  67. package/dist/chunk-GNNBM2WJ.js +83 -0
  68. package/dist/chunk-IPKO6UQN.js +25 -0
  69. package/dist/chunk-JOCGX3RP.js +59 -0
  70. package/dist/chunk-KBXWTD6E.js +94 -0
  71. package/dist/chunk-KMMAZFSJ.js +25 -0
  72. package/dist/chunk-LATR3NIV.js +117 -0
  73. package/dist/chunk-LAYPUDQF.js +39 -0
  74. package/dist/chunk-LDU7A2JE.js +25 -0
  75. package/dist/chunk-LX3TJ2TJ.js +294 -0
  76. package/dist/chunk-NHZBQNRR.js +143 -0
  77. package/dist/chunk-OVHNRULD.js +33 -0
  78. package/dist/chunk-PDP3PHB3.js +127 -0
  79. package/dist/chunk-PHYNROD4.js +47 -0
  80. package/dist/chunk-QQEKY4XP.js +29 -0
  81. package/dist/chunk-U4RK4LKJ.js +348 -0
  82. package/dist/{chunk-RRLIF4PQ.js → chunk-U5663F2U.js} +16 -1
  83. package/dist/chunk-UN7X6SU5.js +53 -0
  84. package/dist/chunk-UZQJJD6A.js +100 -0
  85. package/dist/chunk-V6LLEAR4.js +80 -0
  86. package/dist/chunk-WHNDRO3N.js +50 -0
  87. package/dist/{chunk-W6LG7BFW.js → chunk-XY5R3EHH.js} +30 -23
  88. package/dist/client/client.cjs +135 -0
  89. package/dist/client/client.d.ts +85 -0
  90. package/dist/client/client.js +9 -0
  91. package/dist/client/index.cjs +135 -0
  92. package/dist/client/index.d.ts +14 -0
  93. package/dist/client/index.js +10 -0
  94. package/dist/context.cjs +1237 -0
  95. package/dist/context.d.ts +16 -0
  96. package/dist/context.js +28 -0
  97. package/dist/cookie.cjs +57 -22
  98. package/dist/cookie.d.ts +11 -6
  99. package/dist/cookie.js +3 -2
  100. package/dist/createAuth.cjs +2320 -0
  101. package/dist/createAuth.d.ts +12 -0
  102. package/dist/createAuth.js +48 -0
  103. package/dist/env.cjs +78 -0
  104. package/dist/env.d.ts +10 -0
  105. package/dist/env.js +12 -0
  106. package/dist/errors.cjs +17 -0
  107. package/dist/errors.d.ts +15 -4
  108. package/dist/errors.js +5 -1
  109. package/dist/headers.cjs +28 -2
  110. package/dist/headers.d.ts +25 -1
  111. package/dist/headers.js +9 -3
  112. package/dist/index-_aXtxb_s.d.ts +1377 -0
  113. package/dist/index.cjs +1843 -610
  114. package/dist/index.d.ts +11 -92
  115. package/dist/index.js +53 -85
  116. package/dist/jose.cjs +113 -38
  117. package/dist/jose.d.ts +12 -23
  118. package/dist/jose.js +17 -7
  119. package/dist/logger.cjs +424 -0
  120. package/dist/logger.d.ts +12 -0
  121. package/dist/logger.js +17 -0
  122. package/dist/oauth/atlassian.cjs +57 -0
  123. package/dist/oauth/atlassian.d.ts +12 -0
  124. package/dist/oauth/atlassian.js +6 -0
  125. package/dist/oauth/bitbucket.cjs +19 -15
  126. package/dist/oauth/bitbucket.d.ts +7 -2
  127. package/dist/oauth/bitbucket.js +1 -1
  128. package/dist/oauth/discord.cjs +27 -24
  129. package/dist/oauth/discord.d.ts +7 -2
  130. package/dist/oauth/discord.js +1 -1
  131. package/dist/oauth/dropbox.cjs +53 -0
  132. package/dist/oauth/dropbox.d.ts +12 -0
  133. package/dist/oauth/dropbox.js +6 -0
  134. package/dist/oauth/figma.cjs +19 -16
  135. package/dist/oauth/figma.d.ts +7 -2
  136. package/dist/oauth/figma.js +1 -1
  137. package/dist/oauth/github.cjs +19 -8
  138. package/dist/oauth/github.d.ts +7 -2
  139. package/dist/oauth/github.js +1 -1
  140. package/dist/oauth/gitlab.cjs +19 -16
  141. package/dist/oauth/gitlab.d.ts +7 -2
  142. package/dist/oauth/gitlab.js +1 -1
  143. package/dist/oauth/index.cjs +529 -239
  144. package/dist/oauth/index.d.ts +7 -2
  145. package/dist/oauth/index.js +39 -22
  146. package/dist/oauth/mailchimp.cjs +19 -16
  147. package/dist/oauth/mailchimp.d.ts +7 -2
  148. package/dist/oauth/mailchimp.js +1 -1
  149. package/dist/oauth/notion.cjs +131 -0
  150. package/dist/oauth/notion.d.ts +12 -0
  151. package/dist/oauth/notion.js +9 -0
  152. package/dist/oauth/pinterest.cjs +19 -16
  153. package/dist/oauth/pinterest.d.ts +7 -2
  154. package/dist/oauth/pinterest.js +1 -1
  155. package/dist/oauth/spotify.cjs +19 -16
  156. package/dist/oauth/spotify.d.ts +7 -2
  157. package/dist/oauth/spotify.js +1 -1
  158. package/dist/oauth/strava.cjs +19 -16
  159. package/dist/oauth/strava.d.ts +7 -2
  160. package/dist/oauth/strava.js +1 -1
  161. package/dist/oauth/twitch.cjs +95 -0
  162. package/dist/oauth/twitch.d.ts +12 -0
  163. package/dist/oauth/twitch.js +7 -0
  164. package/dist/oauth/x.cjs +19 -16
  165. package/dist/oauth/x.d.ts +7 -2
  166. package/dist/oauth/x.js +1 -1
  167. package/dist/schemas.cjs +89 -42
  168. package/dist/schemas.d.ts +114 -18
  169. package/dist/schemas.js +5 -3
  170. package/dist/secure.cjs +73 -31
  171. package/dist/secure.d.ts +11 -11
  172. package/dist/secure.js +7 -6
  173. package/dist/utils.cjs +203 -90
  174. package/dist/utils.d.ts +21 -40
  175. package/dist/utils.js +21 -12
  176. package/package.json +9 -6
  177. package/dist/chunk-3EUWD5BB.js +0 -63
  178. package/dist/chunk-42XB3YCW.js +0 -22
  179. package/dist/chunk-6R2YZ4AC.js +0 -22
  180. package/dist/chunk-A3N4PVAT.js +0 -70
  181. package/dist/chunk-B737EUJV.js +0 -22
  182. package/dist/chunk-CXLATHS5.js +0 -143
  183. package/dist/chunk-E3OXBRYF.js +0 -22
  184. package/dist/chunk-EIL2FPSS.js +0 -22
  185. package/dist/chunk-EMKJA2GJ.js +0 -89
  186. package/dist/chunk-FIPU4MLT.js +0 -21
  187. package/dist/chunk-FKRDCWBF.js +0 -22
  188. package/dist/chunk-GA2SMTJO.js +0 -58
  189. package/dist/chunk-HP34YGGJ.js +0 -22
  190. package/dist/chunk-IKHPGFCW.js +0 -14
  191. package/dist/chunk-IUYZQTJV.js +0 -30
  192. package/dist/chunk-IVET23KF.js +0 -58
  193. package/dist/chunk-JVFTCTTE.js +0 -33
  194. package/dist/chunk-KRNOMBXQ.js +0 -22
  195. package/dist/chunk-KSWLO5ZU.js +0 -102
  196. package/dist/chunk-N2APGLXA.js +0 -71
  197. package/dist/chunk-N4SX7TZT.js +0 -96
  198. package/dist/chunk-STHEPPUZ.js +0 -11
  199. package/dist/chunk-TLE4PXY3.js +0 -39
  200. package/dist/index-B8jeIElf.d.ts +0 -679
  201. /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-W6LG7BFW.js";
7
+ } from "./chunk-XY5R3EHH.js";
5
8
  import {
6
- cacheControl
7
- } from "./chunk-STHEPPUZ.js";
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
- const headers = new Headers(cacheControl);
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
+ };