@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,279 @@
1
+ "use strict";
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __export = (target, all) => {
7
+ for (var name in all)
8
+ __defProp(target, name, { get: all[name], enumerable: true });
9
+ };
10
+ var __copyProps = (to, from, except, desc) => {
11
+ if (from && typeof from === "object" || typeof from === "function") {
12
+ for (let key of __getOwnPropNames(from))
13
+ if (!__hasOwnProp.call(to, key) && key !== except)
14
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
15
+ }
16
+ return to;
17
+ };
18
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
19
+
20
+ // src/api/signOut.ts
21
+ var signOut_exports = {};
22
+ __export(signOut_exports, {
23
+ signOut: () => signOut
24
+ });
25
+ module.exports = __toCommonJS(signOut_exports);
26
+
27
+ // src/env.ts
28
+ var import_meta = {};
29
+ var env = new Proxy({}, {
30
+ get(_, prop) {
31
+ if (typeof prop !== "string") return void 0;
32
+ const hasProperty = (process2) => {
33
+ return process2 && Object.prototype.hasOwnProperty.call(process2, prop);
34
+ };
35
+ try {
36
+ if (typeof process !== "undefined" && hasProperty(process.env)) {
37
+ return process.env[prop];
38
+ }
39
+ if (typeof import_meta !== "undefined" && hasProperty(import_meta.env)) {
40
+ return import_meta.env[prop];
41
+ }
42
+ if (typeof Deno !== "undefined" && Deno.env?.get) {
43
+ return Deno.env.get(prop);
44
+ }
45
+ if (typeof Bun !== "undefined" && hasProperty(Bun.env)) {
46
+ return Bun.env[prop];
47
+ }
48
+ const globalValue = globalThis[prop];
49
+ return typeof globalValue === "string" ? globalValue : void 0;
50
+ } catch {
51
+ return void 0;
52
+ }
53
+ }
54
+ });
55
+
56
+ // src/cookie.ts
57
+ var import_cookie = require("@aura-stack/router/cookie");
58
+
59
+ // src/errors.ts
60
+ var AuthInternalError = class extends Error {
61
+ type = "AUTH_INTERNAL_ERROR";
62
+ code;
63
+ constructor(code, message, options) {
64
+ super(message, options);
65
+ this.code = code;
66
+ this.name = new.target.name;
67
+ Error.captureStackTrace(this, new.target);
68
+ }
69
+ };
70
+ var AuthSecurityError = class extends Error {
71
+ type = "AUTH_SECURITY_ERROR";
72
+ code;
73
+ constructor(code, message, options) {
74
+ super(message, options);
75
+ this.code = code;
76
+ this.name = new.target.name;
77
+ Error.captureStackTrace(this, new.target);
78
+ }
79
+ };
80
+
81
+ // src/cookie.ts
82
+ var defaultCookieOptions = {
83
+ httpOnly: true,
84
+ sameSite: "lax",
85
+ path: "/",
86
+ maxAge: 60 * 60 * 24 * 15
87
+ };
88
+ var oauthCookieOptions = {
89
+ httpOnly: true,
90
+ maxAge: 5 * 60,
91
+ sameSite: "lax",
92
+ expires: new Date(Date.now() + 5 * 60 * 1e3)
93
+ };
94
+ var expiredCookieAttributes = {
95
+ ...defaultCookieOptions,
96
+ expires: /* @__PURE__ */ new Date(0),
97
+ maxAge: 0,
98
+ secure: true
99
+ };
100
+ var getCookie = (request, cookieName) => {
101
+ const cookies = request instanceof Request ? request.headers.get("Cookie") : request.get("Cookie");
102
+ if (!cookies) {
103
+ throw new AuthInternalError("COOKIE_NOT_FOUND", "No cookies found. There is no active session");
104
+ }
105
+ const value = (0, import_cookie.parse)(cookies)[cookieName];
106
+ if (!value) {
107
+ throw new AuthInternalError("COOKIE_NOT_FOUND", `Cookie "${cookieName}" not found. There is no active session`);
108
+ }
109
+ return value;
110
+ };
111
+
112
+ // src/utils.ts
113
+ var import_router = require("@aura-stack/router");
114
+
115
+ // src/assert.ts
116
+ var import_crypto = require("@aura-stack/jose/crypto");
117
+ var isJWTPayloadWithToken = (payload) => {
118
+ return typeof payload === "object" && payload !== null && "token" in payload && typeof payload?.token === "string";
119
+ };
120
+ var timingSafeEqual = (a, b) => {
121
+ const bufferA = import_crypto.encoder.encode(a);
122
+ const bufferB = import_crypto.encoder.encode(b);
123
+ const len = Math.max(bufferA.length, bufferB.length);
124
+ let diff = 0;
125
+ for (let i = 0; i < len; i++) {
126
+ diff |= (bufferA[i] ?? 0) ^ (bufferB[i] ?? 0);
127
+ }
128
+ return diff === 0 && bufferA.length === bufferB.length;
129
+ };
130
+
131
+ // src/utils.ts
132
+ var equals = (a, b) => {
133
+ if (a === null || b === null || a === void 0 || b === void 0) return false;
134
+ return a === b;
135
+ };
136
+ var getErrorName = (error) => {
137
+ if (error instanceof Error) {
138
+ return error.name;
139
+ }
140
+ return typeof error === "string" ? error : "UnknownError";
141
+ };
142
+
143
+ // src/jose.ts
144
+ var import_jose = require("@aura-stack/jose");
145
+ var import_jose2 = require("@aura-stack/jose/jose");
146
+ var import_crypto2 = require("@aura-stack/jose/crypto");
147
+ var jwtVerificationOptions = {
148
+ algorithms: ["HS256"],
149
+ typ: "JWT"
150
+ };
151
+
152
+ // src/secure.ts
153
+ var verifyCSRF = async (jose, cookie, header) => {
154
+ try {
155
+ const cookiePayload = await jose.verifyJWS(cookie, jwtVerificationOptions);
156
+ const headerPayload = await jose.verifyJWS(header, jwtVerificationOptions);
157
+ if (!isJWTPayloadWithToken(cookiePayload)) {
158
+ throw new AuthSecurityError("CSRF_TOKEN_INVALID", "Cookie payload missing token field.");
159
+ }
160
+ if (!isJWTPayloadWithToken(headerPayload)) {
161
+ throw new AuthSecurityError("CSRF_TOKEN_INVALID", "Header payload missing token field.");
162
+ }
163
+ if (!equals(cookiePayload.token.length, headerPayload.token.length)) {
164
+ throw new AuthSecurityError("CSRF_TOKEN_INVALID", "The CSRF tokens do not match.");
165
+ }
166
+ if (!timingSafeEqual(cookiePayload.token, headerPayload.token)) {
167
+ throw new AuthSecurityError("CSRF_TOKEN_INVALID", "The CSRF tokens do not match.");
168
+ }
169
+ return true;
170
+ } catch {
171
+ throw new AuthSecurityError("CSRF_TOKEN_INVALID", "The CSRF tokens do not match.");
172
+ }
173
+ };
174
+
175
+ // src/headers.ts
176
+ var cacheControl = {
177
+ "Cache-Control": "no-store",
178
+ Pragma: "no-cache",
179
+ Expires: "0",
180
+ Vary: "Cookie"
181
+ };
182
+ var contentSecurityPolicy = {
183
+ "Content-Security-Policy": [
184
+ "default-src 'none'",
185
+ "script-src 'self'",
186
+ "frame-src 'none'",
187
+ "object-src 'none'",
188
+ "frame-ancestors 'none'",
189
+ "base-uri 'none'"
190
+ ].join("; ")
191
+ };
192
+ var secureHeaders = {
193
+ "X-Content-Type-Options": "nosniff",
194
+ "X-Frame-Options": "DENY",
195
+ "Referrer-Policy": "strict-origin-when-cross-origin"
196
+ };
197
+ var secureApiHeaders = {
198
+ ...cacheControl,
199
+ ...contentSecurityPolicy,
200
+ ...secureHeaders
201
+ };
202
+
203
+ // src/api/signOut.ts
204
+ var import_router2 = require("@aura-stack/router");
205
+ var signOut = async ({
206
+ ctx,
207
+ headers: headersInit,
208
+ redirectTo = "/",
209
+ skipCSRFCheck = false
210
+ }) => {
211
+ const headers = new Headers(headersInit);
212
+ const header = headers.get("X-CSRF-Token");
213
+ let session = null;
214
+ let csrfToken = null;
215
+ try {
216
+ session = getCookie(headers, ctx.cookies.sessionToken.name);
217
+ } catch {
218
+ throw new AuthSecurityError("SESSION_TOKEN_MISSING", "The sessionToken is missing.");
219
+ }
220
+ try {
221
+ csrfToken = getCookie(headers, ctx.cookies.csrfToken.name);
222
+ } catch {
223
+ throw new AuthSecurityError("CSRF_TOKEN_MISSING", "The CSRF token is missing.");
224
+ }
225
+ ctx?.logger?.log("SIGN_OUT_ATTEMPT", {
226
+ structuredData: {
227
+ has_session: Boolean(session),
228
+ has_csrf_token: Boolean(csrfToken),
229
+ has_csrf_header: Boolean(header),
230
+ skip_csrf_check: skipCSRFCheck
231
+ }
232
+ });
233
+ if (!session) {
234
+ ctx?.logger?.log("SESSION_TOKEN_MISSING");
235
+ throw new AuthSecurityError("SESSION_TOKEN_MISSING", "The sessionToken is missing.");
236
+ }
237
+ if (!skipCSRFCheck) {
238
+ if (!csrfToken) {
239
+ ctx?.logger?.log("CSRF_TOKEN_MISSING");
240
+ throw new AuthSecurityError("CSRF_TOKEN_MISSING", "The CSRF token is missing.");
241
+ }
242
+ if (!header) {
243
+ ctx?.logger?.log("CSRF_HEADER_MISSING");
244
+ throw new AuthSecurityError("CSRF_HEADER_MISSING", "The CSRF header is missing.");
245
+ }
246
+ try {
247
+ await verifyCSRF(ctx.jose, csrfToken, header);
248
+ } catch (error) {
249
+ ctx?.logger?.log("CSRF_TOKEN_INVALID", { structuredData: { error_type: getErrorName(error) } });
250
+ throw new AuthSecurityError("CSRF_TOKEN_INVALID", "CSRF token verification failed");
251
+ }
252
+ ctx?.logger?.log("SIGN_OUT_CSRF_VERIFIED");
253
+ } else {
254
+ try {
255
+ await ctx.jose.verifyJWS(csrfToken);
256
+ } catch (error) {
257
+ ctx?.logger?.log("CSRF_TOKEN_INVALID", { structuredData: { error_type: getErrorName(error) } });
258
+ throw new AuthSecurityError("CSRF_TOKEN_INVALID", "CSRF token verification failed");
259
+ }
260
+ }
261
+ try {
262
+ await ctx.jose.decodeJWT(session);
263
+ ctx?.logger?.log("SIGN_OUT_SUCCESS");
264
+ } catch (error) {
265
+ ctx?.logger?.log("INVALID_JWT_TOKEN", { structuredData: { error_type: getErrorName(error) } });
266
+ }
267
+ const headersList = new import_router2.HeadersBuilder(secureApiHeaders).setHeader("Location", redirectTo).setCookie(ctx.cookies.csrfToken.name, "", expiredCookieAttributes).setCookie(ctx.cookies.sessionToken.name, "", expiredCookieAttributes).toHeaders();
268
+ return Response.json(
269
+ { redirect: Boolean(redirectTo), url: redirectTo },
270
+ {
271
+ status: 202,
272
+ headers: headersList
273
+ }
274
+ );
275
+ };
276
+ // Annotate the CommonJS export names for ESM import in node:
277
+ 0 && (module.exports = {
278
+ signOut
279
+ });
@@ -0,0 +1,16 @@
1
+ import { F as FunctionAPIContext, i as SignOutAPIOptions } from '../index-_aXtxb_s.js';
2
+ import 'zod';
3
+ import '../schemas.js';
4
+ import 'zod/v4';
5
+ import '@aura-stack/jose';
6
+ import '@aura-stack/jose/jose';
7
+ import '@aura-stack/jose/crypto';
8
+ import '@aura-stack/router/cookie';
9
+ import '../@types/utility.js';
10
+ import 'jose';
11
+ import '@aura-stack/router';
12
+ import 'zod/v4/core';
13
+
14
+ declare const signOut: ({ ctx, headers: headersInit, redirectTo, skipCSRFCheck, }: FunctionAPIContext<SignOutAPIOptions>) => Promise<Response>;
15
+
16
+ export { signOut };
@@ -0,0 +1,13 @@
1
+ import {
2
+ signOut
3
+ } from "../chunk-KBXWTD6E.js";
4
+ import "../chunk-V6LLEAR4.js";
5
+ import "../chunk-UZQJJD6A.js";
6
+ import "../chunk-LX3TJ2TJ.js";
7
+ import "../chunk-XY5R3EHH.js";
8
+ import "../chunk-WHNDRO3N.js";
9
+ import "../chunk-U5663F2U.js";
10
+ import "../chunk-EBAMFRB7.js";
11
+ export {
12
+ signOut
13
+ };
package/dist/assert.cjs CHANGED
@@ -22,28 +22,173 @@ var assert_exports = {};
22
22
  __export(assert_exports, {
23
23
  isFalsy: () => isFalsy,
24
24
  isJWTPayloadWithToken: () => isJWTPayloadWithToken,
25
+ isRelativeURL: () => isRelativeURL,
25
26
  isRequest: () => isRequest,
26
- isValidURL: () => isValidURL
27
+ isSameOrigin: () => isSameOrigin,
28
+ isTrustedOrigin: () => isTrustedOrigin,
29
+ isValidURL: () => isValidURL,
30
+ patternToRegex: () => patternToRegex,
31
+ timingSafeEqual: () => timingSafeEqual,
32
+ unsafeChars: () => unsafeChars
27
33
  });
28
34
  module.exports = __toCommonJS(assert_exports);
35
+
36
+ // src/utils.ts
37
+ var import_router = require("@aura-stack/router");
38
+
39
+ // src/env.ts
40
+ var import_meta = {};
41
+ var env = new Proxy({}, {
42
+ get(_, prop) {
43
+ if (typeof prop !== "string") return void 0;
44
+ const hasProperty = (process2) => {
45
+ return process2 && Object.prototype.hasOwnProperty.call(process2, prop);
46
+ };
47
+ try {
48
+ if (typeof process !== "undefined" && hasProperty(process.env)) {
49
+ return process.env[prop];
50
+ }
51
+ if (typeof import_meta !== "undefined" && hasProperty(import_meta.env)) {
52
+ return import_meta.env[prop];
53
+ }
54
+ if (typeof Deno !== "undefined" && Deno.env?.get) {
55
+ return Deno.env.get(prop);
56
+ }
57
+ if (typeof Bun !== "undefined" && hasProperty(Bun.env)) {
58
+ return Bun.env[prop];
59
+ }
60
+ const globalValue = globalThis[prop];
61
+ return typeof globalValue === "string" ? globalValue : void 0;
62
+ } catch {
63
+ return void 0;
64
+ }
65
+ }
66
+ });
67
+
68
+ // src/utils.ts
69
+ var equals = (a, b) => {
70
+ if (a === null || b === null || a === void 0 || b === void 0) return false;
71
+ return a === b;
72
+ };
73
+
74
+ // src/assert.ts
75
+ var import_crypto = require("@aura-stack/jose/crypto");
29
76
  var isFalsy = (value) => {
30
77
  return value === false || value === 0 || value === "" || value === null || value === void 0 || Number.isNaN(value);
31
78
  };
32
79
  var isRequest = (value) => {
33
80
  return typeof Request !== "undefined" && value instanceof Request;
34
81
  };
82
+ var unsafeChars = [
83
+ "<",
84
+ ">",
85
+ '"',
86
+ "`",
87
+ " ",
88
+ "\r",
89
+ "\n",
90
+ " ",
91
+ "\\",
92
+ "%2F",
93
+ "%5C",
94
+ "%2f",
95
+ "%5c",
96
+ "\r\n",
97
+ "%0A",
98
+ "%0D",
99
+ "%0a",
100
+ "%0d",
101
+ "..",
102
+ "//",
103
+ "///",
104
+ "...",
105
+ "%20",
106
+ "\0"
107
+ ];
35
108
  var isValidURL = (value) => {
36
- if (value.includes("\r\n") || value.includes("\n") || value.includes("\r")) return false;
37
- const regex = /^https?:\/\/(?:[a-zA-Z0-9._-]+|localhost|\[[0-9a-fA-F:]+\])(?::\d{1,5})?(?:\/[a-zA-Z0-9._~!$&'()*+,;=:@-]*)*\/?$/;
38
- return regex.test(value);
109
+ if (!new RegExp(/^https?:\/\/[^/]/).test(value)) {
110
+ return false;
111
+ }
112
+ const match = value.match(/^(https?:\/\/)(.*)$/);
113
+ if (!match) return false;
114
+ const rest = match[2];
115
+ for (const char of unsafeChars) {
116
+ if (rest.includes(char)) return false;
117
+ }
118
+ const regex = /^https?:\/\/(?:[a-zA-Z0-9._-]+|localhost|\[[0-9a-fA-F:]+\])(?::\d{1,5})?(?:\/[a-zA-Z0-9._~!$&'()?#*+,;=:@-]*)*\/?$/;
119
+ return regex.test(match[0]);
39
120
  };
40
121
  var isJWTPayloadWithToken = (payload) => {
41
122
  return typeof payload === "object" && payload !== null && "token" in payload && typeof payload?.token === "string";
42
123
  };
124
+ var isRelativeURL = (value) => {
125
+ if (value.length > 100) return false;
126
+ for (const char of unsafeChars) {
127
+ if (value.includes(char)) return false;
128
+ }
129
+ const regex = /^\/[a-zA-Z0-9\-_\/.?&=#]*\/?$/;
130
+ return regex.test(value);
131
+ };
132
+ var isSameOrigin = (origin, expected) => {
133
+ const originURL = new URL(origin);
134
+ const expectedURL = new URL(expected);
135
+ return equals(originURL.origin, expectedURL.origin);
136
+ };
137
+ var patternToRegex = (pattern) => {
138
+ try {
139
+ if (pattern.length > 2048) return null;
140
+ pattern = pattern.replace(/\\/g, "");
141
+ const match = pattern.match(/^(https?):\/\/([a-zA-Z0-9.*-]{1,253})(?::(\d{1,5}|\*))?(?:\/.*)?$/);
142
+ if (!match) return null;
143
+ const [, protocol, host, port] = match;
144
+ const hasWildcard = host.includes("*");
145
+ if (hasWildcard && !host.startsWith("*.")) return null;
146
+ if (hasWildcard && host.slice(2).includes("*")) return null;
147
+ const domain = hasWildcard ? host.slice(2) : host;
148
+ const escapedDomain = domain.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
149
+ const hostRegex = hasWildcard ? `[^.]+\\.${escapedDomain}` : escapedDomain;
150
+ const portRegex = port === "*" ? ":\\d{1,5}" : port ? `:${port}` : "";
151
+ return new RegExp(`^${protocol}:\\/\\/${hostRegex}${portRegex}$`);
152
+ } catch {
153
+ return null;
154
+ }
155
+ };
156
+ var isTrustedOrigin = (url, trustedOrigins) => {
157
+ if (!isValidURL(url) || trustedOrigins.length === 0) return false;
158
+ try {
159
+ const urlOrigin = new URL(url).origin;
160
+ for (const pattern of trustedOrigins) {
161
+ const regex = patternToRegex(pattern);
162
+ if (regex?.test(urlOrigin)) return true;
163
+ try {
164
+ if (isValidURL(pattern) && equals(new URL(pattern).origin, urlOrigin)) return true;
165
+ } catch {
166
+ }
167
+ }
168
+ } catch {
169
+ }
170
+ return false;
171
+ };
172
+ var timingSafeEqual = (a, b) => {
173
+ const bufferA = import_crypto.encoder.encode(a);
174
+ const bufferB = import_crypto.encoder.encode(b);
175
+ const len = Math.max(bufferA.length, bufferB.length);
176
+ let diff = 0;
177
+ for (let i = 0; i < len; i++) {
178
+ diff |= (bufferA[i] ?? 0) ^ (bufferB[i] ?? 0);
179
+ }
180
+ return diff === 0 && bufferA.length === bufferB.length;
181
+ };
43
182
  // Annotate the CommonJS export names for ESM import in node:
44
183
  0 && (module.exports = {
45
184
  isFalsy,
46
185
  isJWTPayloadWithToken,
186
+ isRelativeURL,
47
187
  isRequest,
48
- isValidURL
188
+ isSameOrigin,
189
+ isTrustedOrigin,
190
+ isValidURL,
191
+ patternToRegex,
192
+ timingSafeEqual,
193
+ unsafeChars
49
194
  });
package/dist/assert.d.ts CHANGED
@@ -1,14 +1,37 @@
1
- import { J as JWTPayloadWithToken } from './index-B8jeIElf.js';
1
+ import { J as JWTPayloadWithToken } from './index-_aXtxb_s.js';
2
2
  import 'zod';
3
3
  import './schemas.js';
4
- import '@aura-stack/router/cookie';
4
+ import 'zod/v4';
5
5
  import '@aura-stack/jose';
6
6
  import '@aura-stack/jose/jose';
7
+ import '@aura-stack/jose/crypto';
8
+ import '@aura-stack/router/cookie';
7
9
  import './@types/utility.js';
10
+ import 'jose';
11
+ import '@aura-stack/router';
12
+ import 'zod/v4/core';
8
13
 
9
14
  declare const isFalsy: (value: unknown) => boolean;
10
15
  declare const isRequest: (value: unknown) => value is Request;
16
+ declare const unsafeChars: string[];
11
17
  declare const isValidURL: (value: string) => boolean;
12
18
  declare const isJWTPayloadWithToken: (payload: unknown) => payload is JWTPayloadWithToken;
19
+ declare const isRelativeURL: (value: string) => boolean;
20
+ declare const isSameOrigin: (origin: string, expected: string) => boolean;
21
+ /**
22
+ * Converts a trusted origin pattern to a regex for matching.
23
+ * Supports `*` as subdomain wildcard: `https://*.example.com` matches `https://app.example.com`
24
+ * @todo: add support to Custom URI Schemes (e.g. `myapp://*`).
25
+ */
26
+ declare const patternToRegex: (pattern: string) => RegExp | null;
27
+ /**
28
+ * Checks if a URL matches any of the trusted origin patterns.
29
+ * A URL is trusted if its origin matches any pattern (exact or wildcard).
30
+ *
31
+ * @param url - The URL to validate (e.g. from Referer, Origin, redirectTo)
32
+ * @param trustedOrigins - Array of exact URLs or patterns (e.g. `https://*.example.com`)
33
+ */
34
+ declare const isTrustedOrigin: (url: string, trustedOrigins: string[]) => boolean;
35
+ declare const timingSafeEqual: (a: string, b: string) => boolean;
13
36
 
14
- export { isFalsy, isJWTPayloadWithToken, isRequest, isValidURL };
37
+ export { isFalsy, isJWTPayloadWithToken, isRelativeURL, isRequest, isSameOrigin, isTrustedOrigin, isValidURL, patternToRegex, timingSafeEqual, unsafeChars };
package/dist/assert.js CHANGED
@@ -1,12 +1,26 @@
1
1
  import {
2
2
  isFalsy,
3
3
  isJWTPayloadWithToken,
4
+ isRelativeURL,
4
5
  isRequest,
5
- isValidURL
6
- } from "./chunk-EIL2FPSS.js";
6
+ isSameOrigin,
7
+ isTrustedOrigin,
8
+ isValidURL,
9
+ patternToRegex,
10
+ timingSafeEqual,
11
+ unsafeChars
12
+ } from "./chunk-LX3TJ2TJ.js";
13
+ import "./chunk-WHNDRO3N.js";
14
+ import "./chunk-U5663F2U.js";
7
15
  export {
8
16
  isFalsy,
9
17
  isJWTPayloadWithToken,
18
+ isRelativeURL,
10
19
  isRequest,
11
- isValidURL
20
+ isSameOrigin,
21
+ isTrustedOrigin,
22
+ isValidURL,
23
+ patternToRegex,
24
+ timingSafeEqual,
25
+ unsafeChars
12
26
  };
@@ -1,11 +1,56 @@
1
1
  // src/schemas.ts
2
- import { object, string, enum as options, number, z, null as nullable } from "zod";
2
+ import { object, string, enum as options, number, z, null as nullable, union, array } from "zod/v4";
3
+ var AuthorizeConfigSchema = z.union([
4
+ string().url(),
5
+ object({
6
+ url: string().url(),
7
+ params: object({
8
+ responseType: options(["code", "token", "id_token", "refresh_token"]).optional(),
9
+ scope: string().optional()
10
+ })
11
+ })
12
+ ]);
13
+ var AccessTokenConfigSchema = z.union([
14
+ string().url(),
15
+ object({
16
+ url: string().url(),
17
+ headers: z.record(string(), string()).optional()
18
+ })
19
+ ]);
20
+ var UserInfoConfigSchema = z.union([
21
+ string().url(),
22
+ object({
23
+ url: string().url(),
24
+ headers: z.record(string(), string()).optional(),
25
+ method: string().optional()
26
+ })
27
+ ]);
28
+ var OAuthProviderCredentialsSchema = object({
29
+ id: string(),
30
+ name: string(),
31
+ authorize: AuthorizeConfigSchema.optional(),
32
+ /** @deprecated */
33
+ authorizeURL: string().url().optional(),
34
+ accessToken: AccessTokenConfigSchema,
35
+ /** @deprecated */
36
+ scope: string().optional(),
37
+ userInfo: UserInfoConfigSchema,
38
+ /** @deprecated */
39
+ responseType: options(["code", "token", "id_token", "refresh_token"]).optional(),
40
+ clientId: string(),
41
+ clientSecret: string(),
42
+ profile: z.function().optional()
43
+ });
3
44
  var OAuthProviderConfigSchema = object({
4
- authorizeURL: string().url(),
5
- accessToken: string().url(),
45
+ authorize: AuthorizeConfigSchema.optional(),
46
+ /** @deprecated */
47
+ authorizeURL: string().url().optional(),
48
+ accessToken: AccessTokenConfigSchema,
49
+ /** @deprecated */
6
50
  scope: string().optional(),
7
- userInfo: string().url(),
8
- responseType: options(["code", "token", "id_token"]),
51
+ userInfo: UserInfoConfigSchema,
52
+ /** @deprecated */
53
+ responseType: options(["code", "token", "id_token", "refresh_token"]).optional(),
9
54
  clientId: string(),
10
55
  clientSecret: string()
11
56
  });
@@ -43,7 +88,7 @@ var OAuthAccessTokenResponse = object({
43
88
  token_type: string().optional(),
44
89
  expires_in: number().optional(),
45
90
  refresh_token: string().optional(),
46
- scope: string().optional().or(nullable())
91
+ scope: union([string().optional().or(nullable()), array(string()).optional()])
47
92
  });
48
93
  var OAuthAccessTokenErrorResponse = object({
49
94
  error: options([
@@ -67,6 +112,7 @@ var OAuthEnvSchema = object({
67
112
  });
68
113
 
69
114
  export {
115
+ OAuthProviderCredentialsSchema,
70
116
  OAuthProviderConfigSchema,
71
117
  OAuthAuthorization,
72
118
  OAuthAuthorizationResponse,
@@ -0,0 +1,40 @@
1
+ import {
2
+ getEnv
3
+ } from "./chunk-WHNDRO3N.js";
4
+
5
+ // src/oauth/twitch.ts
6
+ var twitch = (options) => {
7
+ const clientId = options?.clientId ?? getEnv("TWITCH_CLIENT_ID");
8
+ return {
9
+ id: "twitch",
10
+ name: "Twitch",
11
+ authorize: {
12
+ url: "https://id.twitch.tv/oauth2/authorize",
13
+ params: { scope: "user:read:email", responseType: "code" }
14
+ },
15
+ accessToken: "https://id.twitch.tv/oauth2/token",
16
+ userInfo: {
17
+ url: "https://api.twitch.tv/helix/users",
18
+ headers: {
19
+ "Client-ID": clientId
20
+ }
21
+ },
22
+ profile(profile) {
23
+ const user = profile.data[0];
24
+ if (!user) {
25
+ throw new Error("No user data found in Twitch profile response");
26
+ }
27
+ return {
28
+ sub: user.id,
29
+ name: user.display_name,
30
+ email: user.email,
31
+ picture: user.profile_image_url
32
+ };
33
+ },
34
+ ...options
35
+ };
36
+ };
37
+
38
+ export {
39
+ twitch
40
+ };