@aura-stack/auth 0.4.0 → 0.6.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/README.md +36 -1
- package/dist/@types/index.cjs +0 -18
- package/dist/@types/index.d.ts +2 -8
- package/dist/@types/index.js +0 -1
- package/dist/assert-B3iQSYlK.js +3 -0
- package/dist/assert-NJGroSJd.cjs +3 -0
- package/dist/client/index.cjs +1 -0
- package/dist/client/index.d.ts +11 -0
- package/dist/client/index.js +1 -0
- package/dist/crypto-Bz8nIciY.js +1 -0
- package/dist/crypto-CoXA5w_4.cjs +1 -0
- package/dist/env-bq387KyP.cjs +1 -0
- package/dist/env-nvh8QBNz.js +1 -0
- package/dist/errors-CCYPHuBO.cjs +1 -0
- package/dist/errors-DFWHOho6.js +1 -0
- package/dist/index-BkpwQ0l4.d.cts +2279 -0
- package/dist/index-nqLV2t91.d.ts +2279 -0
- package/dist/index.cjs +1 -1839
- package/dist/index.d.cts +2 -0
- package/dist/index.d.ts +2 -35
- package/dist/index.js +1 -132
- package/dist/logger-C59_CDMk.js +1 -0
- package/dist/logger-UnUhYL2V.cjs +1 -0
- package/dist/oauth/atlassian.cjs +1 -0
- package/dist/oauth/atlassian.d.ts +2 -0
- package/dist/oauth/atlassian.js +1 -0
- package/dist/oauth/bitbucket.cjs +1 -49
- package/dist/oauth/bitbucket.d.ts +2 -8
- package/dist/oauth/bitbucket.js +1 -6
- package/dist/oauth/discord.cjs +1 -57
- package/dist/oauth/discord.d.ts +2 -8
- package/dist/oauth/discord.js +1 -6
- package/dist/oauth/dropbox.cjs +1 -0
- package/dist/oauth/dropbox.d.ts +2 -0
- package/dist/oauth/dropbox.js +1 -0
- package/dist/oauth/figma.cjs +1 -49
- package/dist/oauth/figma.d.ts +2 -8
- package/dist/oauth/figma.js +1 -6
- package/dist/oauth/github.cjs +1 -49
- package/dist/oauth/github.d.ts +2 -8
- package/dist/oauth/github.js +1 -6
- package/dist/oauth/gitlab.cjs +1 -49
- package/dist/oauth/gitlab.d.ts +2 -8
- package/dist/oauth/gitlab.js +1 -6
- package/dist/oauth/index.cjs +1 -483
- package/dist/oauth/index.d.ts +2 -8
- package/dist/oauth/index.js +1 -52
- package/dist/oauth/mailchimp.cjs +1 -49
- package/dist/oauth/mailchimp.d.ts +2 -8
- package/dist/oauth/mailchimp.js +1 -6
- package/dist/oauth/notion.cjs +1 -0
- package/dist/oauth/notion.d.ts +2 -0
- package/dist/oauth/notion.js +1 -0
- package/dist/oauth/pinterest.cjs +1 -49
- package/dist/oauth/pinterest.d.ts +2 -8
- package/dist/oauth/pinterest.js +1 -6
- package/dist/oauth/spotify.cjs +1 -49
- package/dist/oauth/spotify.d.ts +2 -8
- package/dist/oauth/spotify.js +1 -6
- package/dist/oauth/strava.cjs +1 -49
- package/dist/oauth/strava.d.ts +2 -8
- package/dist/oauth/strava.js +1 -6
- package/dist/oauth/twitch.cjs +1 -0
- package/dist/oauth/twitch.d.ts +2 -0
- package/dist/oauth/twitch.js +1 -0
- package/dist/oauth/x.cjs +1 -49
- package/dist/oauth/x.d.ts +2 -8
- package/dist/oauth/x.js +1 -6
- package/dist/oauth-BntNm6aE.cjs +1 -0
- package/dist/oauth-DmHy9VrB.js +1 -0
- package/dist/shared/crypto.cjs +1 -0
- package/dist/shared/crypto.d.ts +47 -0
- package/dist/shared/crypto.js +1 -0
- package/dist/shared/identity.cjs +1 -0
- package/dist/shared/identity.d.ts +2 -0
- package/dist/shared/identity.js +1 -0
- package/dist/shared/index.cjs +1 -0
- package/dist/shared/index.d.ts +5 -0
- package/dist/shared/index.js +1 -0
- package/package.json +39 -12
- package/dist/@types/router.d.cjs +0 -1
- package/dist/@types/router.d.d.ts +0 -12
- package/dist/@types/router.d.js +0 -0
- package/dist/@types/utility.cjs +0 -18
- package/dist/@types/utility.d.ts +0 -6
- package/dist/@types/utility.js +0 -1
- package/dist/actions/callback/access-token.cjs +0 -206
- package/dist/actions/callback/access-token.d.ts +0 -29
- package/dist/actions/callback/access-token.js +0 -9
- package/dist/actions/callback/callback.cjs +0 -649
- package/dist/actions/callback/callback.d.ts +0 -13
- package/dist/actions/callback/callback.js +0 -19
- package/dist/actions/callback/userinfo.cjs +0 -250
- package/dist/actions/callback/userinfo.d.ts +0 -21
- package/dist/actions/callback/userinfo.js +0 -14
- package/dist/actions/csrfToken/csrfToken.cjs +0 -197
- package/dist/actions/csrfToken/csrfToken.d.ts +0 -5
- package/dist/actions/csrfToken/csrfToken.js +0 -14
- package/dist/actions/index.cjs +0 -954
- package/dist/actions/index.d.ts +0 -14
- package/dist/actions/index.js +0 -36
- package/dist/actions/session/session.cjs +0 -136
- package/dist/actions/session/session.d.ts +0 -5
- package/dist/actions/session/session.js +0 -10
- package/dist/actions/signIn/authorization.cjs +0 -322
- package/dist/actions/signIn/authorization.d.ts +0 -53
- package/dist/actions/signIn/authorization.js +0 -18
- package/dist/actions/signIn/signIn.cjs +0 -467
- package/dist/actions/signIn/signIn.d.ts +0 -13
- package/dist/actions/signIn/signIn.js +0 -15
- package/dist/actions/signOut/signOut.cjs +0 -493
- package/dist/actions/signOut/signOut.d.ts +0 -8
- package/dist/actions/signOut/signOut.js +0 -16
- package/dist/assert.cjs +0 -161
- package/dist/assert.d.ts +0 -33
- package/dist/assert.js +0 -26
- package/dist/chunk-4EKY7655.js +0 -123
- package/dist/chunk-4MYWAOLG.js +0 -31
- package/dist/chunk-4YHJ4IEQ.js +0 -25
- package/dist/chunk-54CZPKR4.js +0 -25
- package/dist/chunk-5LZ7TOM3.js +0 -25
- package/dist/chunk-5W4BRQYG.js +0 -201
- package/dist/chunk-6MXFPFR3.js +0 -143
- package/dist/chunk-7QF22LHP.js +0 -67
- package/dist/chunk-ALG3GIV4.js +0 -95
- package/dist/chunk-E6G5YCI6.js +0 -25
- package/dist/chunk-EBAMFRB7.js +0 -34
- package/dist/chunk-EEE7UM5T.js +0 -25
- package/dist/chunk-FRJFWTOY.js +0 -70
- package/dist/chunk-FW4W3REU.js +0 -25
- package/dist/chunk-ICAZ4OVS.js +0 -37
- package/dist/chunk-IPKO6UQN.js +0 -25
- package/dist/chunk-ITQ7352M.js +0 -0
- package/dist/chunk-KJBAQZX2.js +0 -92
- package/dist/chunk-KMMAZFSJ.js +0 -25
- package/dist/chunk-LDU7A2JE.js +0 -25
- package/dist/chunk-NUDITUKX.js +0 -73
- package/dist/chunk-OVHNRULD.js +0 -33
- package/dist/chunk-PG7UYFG5.js +0 -0
- package/dist/chunk-PHFH2MGS.js +0 -36
- package/dist/chunk-QQVSRXGX.js +0 -149
- package/dist/chunk-RRLIF4PQ.js +0 -55
- package/dist/chunk-TM5IPSNF.js +0 -113
- package/dist/chunk-TZB6MUXN.js +0 -78
- package/dist/chunk-VNCNJKS2.js +0 -267
- package/dist/chunk-XGLBNXL4.js +0 -75
- package/dist/chunk-XUP6KKNG.js +0 -106
- package/dist/chunk-ZNCZVF6U.js +0 -14
- package/dist/cookie.cjs +0 -246
- package/dist/cookie.d.ts +0 -85
- package/dist/cookie.js +0 -29
- package/dist/env.cjs +0 -56
- package/dist/env.d.ts +0 -7
- package/dist/env.js +0 -6
- package/dist/errors.cjs +0 -85
- package/dist/errors.d.ts +0 -50
- package/dist/errors.js +0 -18
- package/dist/headers.cjs +0 -61
- package/dist/headers.d.ts +0 -33
- package/dist/headers.js +0 -12
- package/dist/index-CSyIJmCM.d.ts +0 -1007
- package/dist/jose.cjs +0 -128
- package/dist/jose.d.ts +0 -25
- package/dist/jose.js +0 -12
- package/dist/logger.cjs +0 -292
- package/dist/logger.d.ts +0 -8
- package/dist/logger.js +0 -8
- package/dist/request.cjs +0 -38
- package/dist/request.d.ts +0 -13
- package/dist/request.js +0 -6
- package/dist/schemas.cjs +0 -125
- package/dist/schemas.d.ts +0 -149
- package/dist/schemas.js +0 -24
- package/dist/secure.cjs +0 -170
- package/dist/secure.d.ts +0 -35
- package/dist/secure.js +0 -19
- package/dist/utils.cjs +0 -223
- package/dist/utils.d.ts +0 -24
- package/dist/utils.js +0 -29
package/dist/index.cjs
CHANGED
|
@@ -1,1839 +1 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __create = Object.create;
|
|
3
|
-
var __defProp = Object.defineProperty;
|
|
4
|
-
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
-
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
-
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
-
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
-
var __export = (target, all) => {
|
|
9
|
-
for (var name in all)
|
|
10
|
-
__defProp(target, name, { get: all[name], enumerable: true });
|
|
11
|
-
};
|
|
12
|
-
var __copyProps = (to, from, except, desc) => {
|
|
13
|
-
if (from && typeof from === "object" || typeof from === "function") {
|
|
14
|
-
for (let key of __getOwnPropNames(from))
|
|
15
|
-
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
16
|
-
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
17
|
-
}
|
|
18
|
-
return to;
|
|
19
|
-
};
|
|
20
|
-
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
21
|
-
// If the importer is in node compatibility mode or this is not an ESM
|
|
22
|
-
// file that has been converted to a CommonJS file using a Babel-
|
|
23
|
-
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
24
|
-
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
25
|
-
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
26
|
-
mod
|
|
27
|
-
));
|
|
28
|
-
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
29
|
-
|
|
30
|
-
// src/index.ts
|
|
31
|
-
var index_exports = {};
|
|
32
|
-
__export(index_exports, {
|
|
33
|
-
createAuth: () => createAuth
|
|
34
|
-
});
|
|
35
|
-
module.exports = __toCommonJS(index_exports);
|
|
36
|
-
var import_router7 = require("@aura-stack/router");
|
|
37
|
-
|
|
38
|
-
// src/env.ts
|
|
39
|
-
var import_meta = {};
|
|
40
|
-
var env = new Proxy({}, {
|
|
41
|
-
get(_, prop) {
|
|
42
|
-
if (typeof prop !== "string") return void 0;
|
|
43
|
-
const hasProperty = (process2) => {
|
|
44
|
-
return process2 && Object.prototype.hasOwnProperty.call(process2, prop);
|
|
45
|
-
};
|
|
46
|
-
try {
|
|
47
|
-
if (typeof process !== "undefined" && hasProperty(process.env)) {
|
|
48
|
-
return process.env[prop];
|
|
49
|
-
}
|
|
50
|
-
if (typeof import_meta !== "undefined" && hasProperty(import_meta.env)) {
|
|
51
|
-
return import_meta.env[prop];
|
|
52
|
-
}
|
|
53
|
-
if (typeof Deno !== "undefined" && Deno.env?.get) {
|
|
54
|
-
return Deno.env.get(prop);
|
|
55
|
-
}
|
|
56
|
-
if (typeof Bun !== "undefined" && hasProperty(Bun.env)) {
|
|
57
|
-
return Bun.env[prop];
|
|
58
|
-
}
|
|
59
|
-
const globalValue = globalThis[prop];
|
|
60
|
-
return typeof globalValue === "string" ? globalValue : void 0;
|
|
61
|
-
} catch {
|
|
62
|
-
return void 0;
|
|
63
|
-
}
|
|
64
|
-
}
|
|
65
|
-
});
|
|
66
|
-
|
|
67
|
-
// src/jose.ts
|
|
68
|
-
var import_jose = require("@aura-stack/jose");
|
|
69
|
-
|
|
70
|
-
// src/errors.ts
|
|
71
|
-
var OAuthProtocolError = class extends Error {
|
|
72
|
-
type = "OAUTH_PROTOCOL_ERROR";
|
|
73
|
-
error;
|
|
74
|
-
errorURI;
|
|
75
|
-
constructor(error, description, errorURI, options2) {
|
|
76
|
-
super(description, options2);
|
|
77
|
-
this.error = error;
|
|
78
|
-
this.errorURI = errorURI;
|
|
79
|
-
this.name = new.target.name;
|
|
80
|
-
Error.captureStackTrace(this, new.target);
|
|
81
|
-
}
|
|
82
|
-
};
|
|
83
|
-
var AuthInternalError = class extends Error {
|
|
84
|
-
type = "AUTH_INTERNAL_ERROR";
|
|
85
|
-
code;
|
|
86
|
-
constructor(code, message, options2) {
|
|
87
|
-
super(message, options2);
|
|
88
|
-
this.code = code;
|
|
89
|
-
this.name = new.target.name;
|
|
90
|
-
Error.captureStackTrace(this, new.target);
|
|
91
|
-
}
|
|
92
|
-
};
|
|
93
|
-
var AuthSecurityError = class extends Error {
|
|
94
|
-
type = "AUTH_SECURITY_ERROR";
|
|
95
|
-
code;
|
|
96
|
-
constructor(code, message, options2) {
|
|
97
|
-
super(message, options2);
|
|
98
|
-
this.code = code;
|
|
99
|
-
this.name = new.target.name;
|
|
100
|
-
Error.captureStackTrace(this, new.target);
|
|
101
|
-
}
|
|
102
|
-
};
|
|
103
|
-
var isNativeError = (error) => {
|
|
104
|
-
return error instanceof Error;
|
|
105
|
-
};
|
|
106
|
-
var isOAuthProtocolError = (error) => {
|
|
107
|
-
return error instanceof OAuthProtocolError;
|
|
108
|
-
};
|
|
109
|
-
var isAuthInternalError = (error) => {
|
|
110
|
-
return error instanceof AuthInternalError;
|
|
111
|
-
};
|
|
112
|
-
var isAuthSecurityError = (error) => {
|
|
113
|
-
return error instanceof AuthSecurityError;
|
|
114
|
-
};
|
|
115
|
-
|
|
116
|
-
// src/jose.ts
|
|
117
|
-
var createJoseInstance = (secret) => {
|
|
118
|
-
secret ??= env.AURA_AUTH_SECRET ?? env.AUTH_SECRET;
|
|
119
|
-
if (!secret) {
|
|
120
|
-
throw new AuthInternalError(
|
|
121
|
-
"JOSE_INITIALIZATION_FAILED",
|
|
122
|
-
"AURA_AUTH_SECRET environment variable is not set and no secret was provided."
|
|
123
|
-
);
|
|
124
|
-
}
|
|
125
|
-
const salt = env.AURA_AUTH_SALT ?? env.AUTH_SALT;
|
|
126
|
-
if (!salt) {
|
|
127
|
-
throw new AuthInternalError(
|
|
128
|
-
"JOSE_INITIALIZATION_FAILED",
|
|
129
|
-
"AURA_AUTH_SALT or AUTH_SALT environment variable is not set. A salt value is required for key derivation."
|
|
130
|
-
);
|
|
131
|
-
}
|
|
132
|
-
try {
|
|
133
|
-
(0, import_jose.createSecret)(salt);
|
|
134
|
-
} catch (error) {
|
|
135
|
-
throw new AuthInternalError(
|
|
136
|
-
"INVALID_SALT_SECRET_VALUE",
|
|
137
|
-
"AURA_AUTH_SALT/AUTH_SALT is invalid. It must be at least 32 bytes long and meet entropy requirements.",
|
|
138
|
-
{ cause: error }
|
|
139
|
-
);
|
|
140
|
-
}
|
|
141
|
-
const { derivedKey: derivedSigningKey } = (0, import_jose.createDeriveKey)(secret, salt, "signing");
|
|
142
|
-
const { derivedKey: derivedEncryptionKey } = (0, import_jose.createDeriveKey)(secret, salt, "encryption");
|
|
143
|
-
const { derivedKey: derivedCsrfTokenKey } = (0, import_jose.createDeriveKey)(secret, salt, "csrfToken");
|
|
144
|
-
const { decodeJWT, encodeJWT } = (0, import_jose.createJWT)({ jws: derivedSigningKey, jwe: derivedEncryptionKey });
|
|
145
|
-
const { signJWS, verifyJWS } = (0, import_jose.createJWS)(derivedCsrfTokenKey);
|
|
146
|
-
const { encryptJWE, decryptJWE } = (0, import_jose.createJWE)(derivedEncryptionKey);
|
|
147
|
-
return {
|
|
148
|
-
decodeJWT,
|
|
149
|
-
encodeJWT,
|
|
150
|
-
signJWS,
|
|
151
|
-
verifyJWS,
|
|
152
|
-
encryptJWE,
|
|
153
|
-
decryptJWE
|
|
154
|
-
};
|
|
155
|
-
};
|
|
156
|
-
var jwtVerificationOptions = {
|
|
157
|
-
algorithms: ["HS256"],
|
|
158
|
-
typ: "JWT"
|
|
159
|
-
};
|
|
160
|
-
|
|
161
|
-
// src/cookie.ts
|
|
162
|
-
var import_cookie = require("@aura-stack/router/cookie");
|
|
163
|
-
var COOKIE_NAME = "aura-auth";
|
|
164
|
-
var defaultCookieOptions = {
|
|
165
|
-
httpOnly: true,
|
|
166
|
-
sameSite: "lax",
|
|
167
|
-
path: "/",
|
|
168
|
-
maxAge: 60 * 60 * 24 * 15
|
|
169
|
-
};
|
|
170
|
-
var defaultStandardCookieConfig = {
|
|
171
|
-
secure: false,
|
|
172
|
-
httpOnly: true
|
|
173
|
-
};
|
|
174
|
-
var defaultSecureCookieConfig = {
|
|
175
|
-
secure: true,
|
|
176
|
-
httpOnly: true
|
|
177
|
-
};
|
|
178
|
-
var defaultHostCookieConfig = {
|
|
179
|
-
secure: true,
|
|
180
|
-
httpOnly: true,
|
|
181
|
-
path: "/",
|
|
182
|
-
domain: void 0
|
|
183
|
-
};
|
|
184
|
-
var oauthCookieOptions = {
|
|
185
|
-
httpOnly: true,
|
|
186
|
-
maxAge: 5 * 60,
|
|
187
|
-
sameSite: "lax",
|
|
188
|
-
expires: new Date(Date.now() + 5 * 60 * 1e3)
|
|
189
|
-
};
|
|
190
|
-
var setCookie = (cookieName, value, options2) => {
|
|
191
|
-
return (0, import_cookie.serialize)(cookieName, value, options2);
|
|
192
|
-
};
|
|
193
|
-
var expiredCookieAttributes = {
|
|
194
|
-
...defaultCookieOptions,
|
|
195
|
-
expires: /* @__PURE__ */ new Date(0),
|
|
196
|
-
maxAge: 0,
|
|
197
|
-
secure: true
|
|
198
|
-
};
|
|
199
|
-
var getCookie = (request, cookieName) => {
|
|
200
|
-
const cookies = request.headers.get("Cookie");
|
|
201
|
-
if (!cookies) {
|
|
202
|
-
throw new AuthInternalError("COOKIE_NOT_FOUND", "No cookies found. There is no active session");
|
|
203
|
-
}
|
|
204
|
-
const value = (0, import_cookie.parse)(cookies)[cookieName];
|
|
205
|
-
if (!value) {
|
|
206
|
-
throw new AuthInternalError("COOKIE_NOT_FOUND", `Cookie "${cookieName}" not found. There is no active session`);
|
|
207
|
-
}
|
|
208
|
-
return value;
|
|
209
|
-
};
|
|
210
|
-
var createSessionCookie = async (jose, session) => {
|
|
211
|
-
try {
|
|
212
|
-
const encoded = await jose.encodeJWT(session);
|
|
213
|
-
return encoded;
|
|
214
|
-
} catch (error) {
|
|
215
|
-
throw new AuthInternalError("INVALID_JWT_TOKEN", "Failed to create session cookie", { cause: error });
|
|
216
|
-
}
|
|
217
|
-
};
|
|
218
|
-
var defineSecureCookieOptions = (useSecure, attributes, strategy, logger) => {
|
|
219
|
-
if (!attributes.httpOnly) {
|
|
220
|
-
logger?.log("COOKIE_HTTPONLY_DISABLED");
|
|
221
|
-
}
|
|
222
|
-
if (attributes.domain === "*") {
|
|
223
|
-
attributes.domain = void 0;
|
|
224
|
-
logger?.log("COOKIE_WILDCARD_DOMAIN");
|
|
225
|
-
}
|
|
226
|
-
if (!useSecure) {
|
|
227
|
-
if (attributes.secure) {
|
|
228
|
-
logger?.log("COOKIE_SECURE_DISABLED");
|
|
229
|
-
}
|
|
230
|
-
if (attributes.sameSite == "none") {
|
|
231
|
-
attributes.sameSite = "lax";
|
|
232
|
-
logger?.log("COOKIE_SAMESITE_NONE_WITHOUT_SECURE");
|
|
233
|
-
}
|
|
234
|
-
if (process.env.NODE_ENV === "production") {
|
|
235
|
-
logger?.log("COOKIE_INSECURE_IN_PRODUCTION");
|
|
236
|
-
}
|
|
237
|
-
if (strategy === "host") {
|
|
238
|
-
logger?.log("COOKIE_HOST_STRATEGY_INSECURE");
|
|
239
|
-
}
|
|
240
|
-
return {
|
|
241
|
-
...defaultCookieOptions,
|
|
242
|
-
...attributes,
|
|
243
|
-
...defaultStandardCookieConfig
|
|
244
|
-
};
|
|
245
|
-
}
|
|
246
|
-
return strategy === "host" ? {
|
|
247
|
-
...defaultCookieOptions,
|
|
248
|
-
...attributes,
|
|
249
|
-
...defaultHostCookieConfig
|
|
250
|
-
} : { ...defaultCookieOptions, ...attributes, ...defaultSecureCookieConfig };
|
|
251
|
-
};
|
|
252
|
-
var createCookieStore = (useSecure, prefix, overrides, logger) => {
|
|
253
|
-
prefix ??= COOKIE_NAME;
|
|
254
|
-
const securePrefix = useSecure ? "__Secure-" : "";
|
|
255
|
-
const hostPrefix = useSecure ? "__Host-" : "";
|
|
256
|
-
return {
|
|
257
|
-
sessionToken: {
|
|
258
|
-
name: `${securePrefix}${prefix}.${overrides?.sessionToken?.name ?? "session_token"}`,
|
|
259
|
-
attributes: defineSecureCookieOptions(
|
|
260
|
-
useSecure,
|
|
261
|
-
{
|
|
262
|
-
...defaultCookieOptions,
|
|
263
|
-
...overrides?.sessionToken?.attributes
|
|
264
|
-
},
|
|
265
|
-
overrides?.sessionToken?.attributes?.strategy ?? "secure",
|
|
266
|
-
logger
|
|
267
|
-
)
|
|
268
|
-
},
|
|
269
|
-
state: {
|
|
270
|
-
name: `${securePrefix}${prefix}.${overrides?.state?.name ?? "state"}`,
|
|
271
|
-
attributes: defineSecureCookieOptions(
|
|
272
|
-
useSecure,
|
|
273
|
-
{
|
|
274
|
-
...oauthCookieOptions,
|
|
275
|
-
...overrides?.state?.attributes
|
|
276
|
-
},
|
|
277
|
-
overrides?.state?.attributes?.strategy ?? "secure",
|
|
278
|
-
logger
|
|
279
|
-
)
|
|
280
|
-
},
|
|
281
|
-
csrfToken: {
|
|
282
|
-
name: `${hostPrefix}${prefix}.${overrides?.csrfToken?.name ?? "csrf_token"}`,
|
|
283
|
-
attributes: defineSecureCookieOptions(
|
|
284
|
-
useSecure,
|
|
285
|
-
{
|
|
286
|
-
...overrides?.csrfToken?.attributes,
|
|
287
|
-
...defaultHostCookieConfig,
|
|
288
|
-
sameSite: "strict"
|
|
289
|
-
},
|
|
290
|
-
overrides?.csrfToken?.attributes?.strategy ?? "host",
|
|
291
|
-
logger
|
|
292
|
-
)
|
|
293
|
-
},
|
|
294
|
-
redirectTo: {
|
|
295
|
-
name: `${securePrefix}${prefix}.${overrides?.redirectTo?.name ?? "redirect_to"}`,
|
|
296
|
-
attributes: defineSecureCookieOptions(
|
|
297
|
-
useSecure,
|
|
298
|
-
{
|
|
299
|
-
...oauthCookieOptions,
|
|
300
|
-
...overrides?.redirectTo?.attributes
|
|
301
|
-
},
|
|
302
|
-
overrides?.redirectTo?.attributes?.strategy ?? "secure",
|
|
303
|
-
logger
|
|
304
|
-
)
|
|
305
|
-
},
|
|
306
|
-
redirectURI: {
|
|
307
|
-
name: `${securePrefix}${prefix}.${overrides?.redirectURI?.name ?? "redirect_uri"}`,
|
|
308
|
-
attributes: defineSecureCookieOptions(
|
|
309
|
-
useSecure,
|
|
310
|
-
{
|
|
311
|
-
...oauthCookieOptions,
|
|
312
|
-
...overrides?.redirectURI?.attributes
|
|
313
|
-
},
|
|
314
|
-
overrides?.redirectURI?.attributes?.strategy ?? "secure",
|
|
315
|
-
logger
|
|
316
|
-
)
|
|
317
|
-
},
|
|
318
|
-
codeVerifier: {
|
|
319
|
-
name: `${securePrefix}${prefix}.${overrides?.codeVerifier?.name ?? "code_verifier"}`,
|
|
320
|
-
attributes: defineSecureCookieOptions(
|
|
321
|
-
useSecure,
|
|
322
|
-
{
|
|
323
|
-
...oauthCookieOptions,
|
|
324
|
-
...overrides?.codeVerifier?.attributes
|
|
325
|
-
},
|
|
326
|
-
overrides?.codeVerifier?.attributes?.strategy ?? "secure",
|
|
327
|
-
logger
|
|
328
|
-
)
|
|
329
|
-
}
|
|
330
|
-
};
|
|
331
|
-
};
|
|
332
|
-
|
|
333
|
-
// src/utils.ts
|
|
334
|
-
var import_router = require("@aura-stack/router");
|
|
335
|
-
var toSnakeCase = (str) => {
|
|
336
|
-
return str.replace(/([a-z0-9])([A-Z])/g, "$1_$2").replace(/([A-Z]+)([A-Z][a-z])/g, "$1_$2").toLowerCase().replace(/^_+/, "");
|
|
337
|
-
};
|
|
338
|
-
var toUpperCase = (str) => {
|
|
339
|
-
return str.toUpperCase();
|
|
340
|
-
};
|
|
341
|
-
var toCastCase = (obj, type = "snake") => {
|
|
342
|
-
return Object.entries(obj).reduce((previous, [key, value]) => {
|
|
343
|
-
const newKey = type === "snake" ? toSnakeCase(key) : toUpperCase(key);
|
|
344
|
-
return { ...previous, [newKey]: value };
|
|
345
|
-
}, {});
|
|
346
|
-
};
|
|
347
|
-
var equals = (a, b) => {
|
|
348
|
-
if (a === null || b === null || a === void 0 || b === void 0) return false;
|
|
349
|
-
return a === b;
|
|
350
|
-
};
|
|
351
|
-
var createErrorHandler = (logger) => {
|
|
352
|
-
return (error) => {
|
|
353
|
-
if ((0, import_router.isRouterError)(error)) {
|
|
354
|
-
const { message, status, statusText } = error;
|
|
355
|
-
logger?.log("ROUTER_INTERNAL_ERROR");
|
|
356
|
-
return Response.json({ type: "ROUTER_ERROR", code: "ROUTER_INTERNAL_ERROR", message }, { status, statusText });
|
|
357
|
-
}
|
|
358
|
-
if ((0, import_router.isInvalidZodSchemaError)(error)) {
|
|
359
|
-
logger?.log("INVALID_REQUEST");
|
|
360
|
-
return Response.json({ type: "ROUTER_ERROR", code: "INVALID_REQUEST", message: error.errors }, { status: 422 });
|
|
361
|
-
}
|
|
362
|
-
if (isOAuthProtocolError(error)) {
|
|
363
|
-
const { error: errorCode, message, type, errorURI } = error;
|
|
364
|
-
logger?.log("OAUTH_PROTOCOL_ERROR", {
|
|
365
|
-
structuredData: {
|
|
366
|
-
error: errorCode,
|
|
367
|
-
error_description: message,
|
|
368
|
-
error_uri: errorURI ?? ""
|
|
369
|
-
}
|
|
370
|
-
});
|
|
371
|
-
return Response.json(
|
|
372
|
-
{
|
|
373
|
-
type,
|
|
374
|
-
message
|
|
375
|
-
},
|
|
376
|
-
{ status: 400 }
|
|
377
|
-
);
|
|
378
|
-
}
|
|
379
|
-
if (isAuthInternalError(error)) {
|
|
380
|
-
const { type, code, message } = error;
|
|
381
|
-
logger?.log("INVALID_OAUTH_CONFIGURATION", {
|
|
382
|
-
structuredData: {
|
|
383
|
-
error: code,
|
|
384
|
-
error_description: message
|
|
385
|
-
}
|
|
386
|
-
});
|
|
387
|
-
return Response.json(
|
|
388
|
-
{
|
|
389
|
-
type,
|
|
390
|
-
message
|
|
391
|
-
},
|
|
392
|
-
{ status: 400 }
|
|
393
|
-
);
|
|
394
|
-
}
|
|
395
|
-
if (isAuthSecurityError(error)) {
|
|
396
|
-
const { type, code, message } = error;
|
|
397
|
-
logger?.log("INVALID_OAUTH_CONFIGURATION", {
|
|
398
|
-
structuredData: {
|
|
399
|
-
error: code,
|
|
400
|
-
error_description: message
|
|
401
|
-
}
|
|
402
|
-
});
|
|
403
|
-
return Response.json(
|
|
404
|
-
{
|
|
405
|
-
type,
|
|
406
|
-
code,
|
|
407
|
-
message
|
|
408
|
-
},
|
|
409
|
-
{ status: 400 }
|
|
410
|
-
);
|
|
411
|
-
}
|
|
412
|
-
logger?.log("SERVER_ERROR");
|
|
413
|
-
return Response.json(
|
|
414
|
-
{ type: "SERVER_ERROR", code: "SERVER_ERROR", message: "An unexpected error occurred" },
|
|
415
|
-
{ status: 500 }
|
|
416
|
-
);
|
|
417
|
-
};
|
|
418
|
-
};
|
|
419
|
-
var getBaseURL = (request) => {
|
|
420
|
-
const url = new URL(request.url);
|
|
421
|
-
return `${url.origin}${url.pathname}`;
|
|
422
|
-
};
|
|
423
|
-
var toISOString = (date) => {
|
|
424
|
-
return new Date(date).toISOString();
|
|
425
|
-
};
|
|
426
|
-
var useSecureCookies = (request, trustedProxyHeaders) => {
|
|
427
|
-
return trustedProxyHeaders ? request.url.startsWith("https://") || request.headers.get("X-Forwarded-Proto") === "https" || (request.headers.get("Forwarded")?.includes("proto=https") ?? false) : request.url.startsWith("https://");
|
|
428
|
-
};
|
|
429
|
-
var formatZodError = (error) => {
|
|
430
|
-
if (!error.issues || error.issues.length === 0) {
|
|
431
|
-
return {};
|
|
432
|
-
}
|
|
433
|
-
return error.issues.reduce((previous, issue) => {
|
|
434
|
-
const key = issue.path.join(".");
|
|
435
|
-
return {
|
|
436
|
-
...previous,
|
|
437
|
-
[key]: {
|
|
438
|
-
code: issue.code,
|
|
439
|
-
message: issue.message
|
|
440
|
-
}
|
|
441
|
-
};
|
|
442
|
-
}, {});
|
|
443
|
-
};
|
|
444
|
-
var extractPath = (url) => {
|
|
445
|
-
const pathRegex = /^https?:\/\/[a-zA-Z0-9_\-\.]+(:\d+)?(\/.*)$/;
|
|
446
|
-
const match = url.match(pathRegex);
|
|
447
|
-
return match && match[2] ? match[2] : "/";
|
|
448
|
-
};
|
|
449
|
-
var getErrorName = (error) => {
|
|
450
|
-
if (error instanceof Error) {
|
|
451
|
-
return error.name;
|
|
452
|
-
}
|
|
453
|
-
return typeof error === "string" ? error : "UnknownError";
|
|
454
|
-
};
|
|
455
|
-
|
|
456
|
-
// src/oauth/github.ts
|
|
457
|
-
var github = (options2) => {
|
|
458
|
-
return {
|
|
459
|
-
id: "github",
|
|
460
|
-
name: "GitHub",
|
|
461
|
-
authorizeURL: "https://github.com/login/oauth/authorize",
|
|
462
|
-
accessToken: "https://github.com/login/oauth/access_token",
|
|
463
|
-
userInfo: "https://api.github.com/user",
|
|
464
|
-
scope: "read:user user:email",
|
|
465
|
-
responseType: "code",
|
|
466
|
-
profile: (profile) => {
|
|
467
|
-
return {
|
|
468
|
-
sub: profile.id.toString(),
|
|
469
|
-
name: profile.name ?? profile.login,
|
|
470
|
-
email: profile.email ?? void 0,
|
|
471
|
-
image: profile.avatar_url
|
|
472
|
-
};
|
|
473
|
-
},
|
|
474
|
-
...options2
|
|
475
|
-
};
|
|
476
|
-
};
|
|
477
|
-
|
|
478
|
-
// src/oauth/bitbucket.ts
|
|
479
|
-
var bitbucket = (options2) => {
|
|
480
|
-
return {
|
|
481
|
-
id: "bitbucket",
|
|
482
|
-
name: "Bitbucket",
|
|
483
|
-
authorizeURL: "https://bitbucket.org/site/oauth2/authorize",
|
|
484
|
-
accessToken: "https://bitbucket.org/site/oauth2/access_token",
|
|
485
|
-
userInfo: "https://api.bitbucket.org/2.0/user",
|
|
486
|
-
scope: "account email",
|
|
487
|
-
responseType: "code",
|
|
488
|
-
profile(profile) {
|
|
489
|
-
return {
|
|
490
|
-
sub: profile.uuid ?? profile.account_id,
|
|
491
|
-
name: profile.display_name ?? profile.nickname,
|
|
492
|
-
image: profile.links.avatar?.href,
|
|
493
|
-
email: void 0
|
|
494
|
-
};
|
|
495
|
-
},
|
|
496
|
-
...options2
|
|
497
|
-
};
|
|
498
|
-
};
|
|
499
|
-
|
|
500
|
-
// src/oauth/figma.ts
|
|
501
|
-
var figma = (options2) => {
|
|
502
|
-
return {
|
|
503
|
-
id: "figma",
|
|
504
|
-
name: "Figma",
|
|
505
|
-
authorizeURL: "https://www.figma.com/oauth",
|
|
506
|
-
accessToken: "https://api.figma.com/v1/oauth/token",
|
|
507
|
-
userInfo: "https://api.figma.com/v1/me",
|
|
508
|
-
scope: "current_user:read",
|
|
509
|
-
responseType: "code",
|
|
510
|
-
profile(profile) {
|
|
511
|
-
return {
|
|
512
|
-
sub: profile.id,
|
|
513
|
-
name: profile.handle,
|
|
514
|
-
email: profile.email,
|
|
515
|
-
image: profile.img_url
|
|
516
|
-
};
|
|
517
|
-
},
|
|
518
|
-
...options2
|
|
519
|
-
};
|
|
520
|
-
};
|
|
521
|
-
|
|
522
|
-
// src/oauth/discord.ts
|
|
523
|
-
var discord = (options2) => {
|
|
524
|
-
return {
|
|
525
|
-
id: "discord",
|
|
526
|
-
name: "Discord",
|
|
527
|
-
authorizeURL: "https://discord.com/oauth2/authorize",
|
|
528
|
-
accessToken: "https://discord.com/api/oauth2/token",
|
|
529
|
-
userInfo: "https://discord.com/api/users/@me",
|
|
530
|
-
scope: "identify email",
|
|
531
|
-
responseType: "code",
|
|
532
|
-
profile(profile) {
|
|
533
|
-
let image = "";
|
|
534
|
-
if (profile.avatar === null) {
|
|
535
|
-
const index = profile.discriminator === "0" ? (BigInt(profile.id) >> 22n) % 6n : Number(profile.discriminator) % 5;
|
|
536
|
-
image = `https://cdn.discordapp.com/embed/avatars/${index}.png`;
|
|
537
|
-
} else {
|
|
538
|
-
const format = profile.avatar.startsWith("a_") ? "gif" : "png";
|
|
539
|
-
image = `https://cdn.discordapp.com/avatars/${profile.id}/${profile.avatar}.${format}`;
|
|
540
|
-
}
|
|
541
|
-
return {
|
|
542
|
-
sub: profile.id,
|
|
543
|
-
name: profile.global_name ?? profile.username,
|
|
544
|
-
email: profile.email ?? "",
|
|
545
|
-
image
|
|
546
|
-
};
|
|
547
|
-
},
|
|
548
|
-
...options2
|
|
549
|
-
};
|
|
550
|
-
};
|
|
551
|
-
|
|
552
|
-
// src/oauth/gitlab.ts
|
|
553
|
-
var gitlab = (options2) => {
|
|
554
|
-
return {
|
|
555
|
-
id: "gitlab",
|
|
556
|
-
name: "GitLab",
|
|
557
|
-
authorizeURL: "https://gitlab.com/oauth/authorize",
|
|
558
|
-
accessToken: "https://gitlab.com/oauth/token",
|
|
559
|
-
userInfo: "https://gitlab.com/api/v4/user",
|
|
560
|
-
scope: "read_user",
|
|
561
|
-
responseType: "code",
|
|
562
|
-
profile(profile) {
|
|
563
|
-
return {
|
|
564
|
-
sub: profile.id.toString(),
|
|
565
|
-
name: profile.name ?? profile.username,
|
|
566
|
-
email: profile.email,
|
|
567
|
-
image: profile.avatar_url
|
|
568
|
-
};
|
|
569
|
-
},
|
|
570
|
-
...options2
|
|
571
|
-
};
|
|
572
|
-
};
|
|
573
|
-
|
|
574
|
-
// src/oauth/spotify.ts
|
|
575
|
-
var spotify = (options2) => {
|
|
576
|
-
return {
|
|
577
|
-
id: "spotify",
|
|
578
|
-
name: "Spotify",
|
|
579
|
-
authorizeURL: "https://accounts.spotify.com/authorize",
|
|
580
|
-
accessToken: "https://accounts.spotify.com/api/token",
|
|
581
|
-
userInfo: "https://api.spotify.com/v1/me",
|
|
582
|
-
scope: "user-read-private user-read-email",
|
|
583
|
-
responseType: "code",
|
|
584
|
-
profile(profile) {
|
|
585
|
-
return {
|
|
586
|
-
sub: profile.id,
|
|
587
|
-
name: profile.display_name,
|
|
588
|
-
email: profile.email,
|
|
589
|
-
image: profile.images[0]?.url ?? void 0
|
|
590
|
-
};
|
|
591
|
-
},
|
|
592
|
-
...options2
|
|
593
|
-
};
|
|
594
|
-
};
|
|
595
|
-
|
|
596
|
-
// src/oauth/x.ts
|
|
597
|
-
var x = (options2) => {
|
|
598
|
-
return {
|
|
599
|
-
id: "x",
|
|
600
|
-
name: "X",
|
|
601
|
-
authorizeURL: "https://twitter.com/i/oauth2/authorize",
|
|
602
|
-
accessToken: "https://api.twitter.com/2/oauth2/token",
|
|
603
|
-
userInfo: "https://api.twitter.com/2/users/me?user.fields=profile_image_url",
|
|
604
|
-
scope: "tweet.read users.read offline.access",
|
|
605
|
-
responseType: "code",
|
|
606
|
-
profile(profile) {
|
|
607
|
-
return {
|
|
608
|
-
sub: profile.data.id,
|
|
609
|
-
name: profile.data.name,
|
|
610
|
-
image: profile.data.profile_image_url,
|
|
611
|
-
email: void 0
|
|
612
|
-
};
|
|
613
|
-
},
|
|
614
|
-
...options2
|
|
615
|
-
};
|
|
616
|
-
};
|
|
617
|
-
|
|
618
|
-
// src/oauth/strava.ts
|
|
619
|
-
var strava = (options2) => {
|
|
620
|
-
return {
|
|
621
|
-
id: "strava",
|
|
622
|
-
name: "Strava",
|
|
623
|
-
authorizeURL: "https://www.strava.com/oauth/authorize",
|
|
624
|
-
accessToken: "https://www.strava.com/oauth/token",
|
|
625
|
-
userInfo: "https://www.strava.com/api/v3/athlete",
|
|
626
|
-
scope: "read",
|
|
627
|
-
responseType: "code",
|
|
628
|
-
profile(profile) {
|
|
629
|
-
return {
|
|
630
|
-
sub: profile.id.toString(),
|
|
631
|
-
name: `${profile.firstname} ${profile.lastname}`,
|
|
632
|
-
image: profile.profile,
|
|
633
|
-
email: void 0
|
|
634
|
-
};
|
|
635
|
-
},
|
|
636
|
-
...options2
|
|
637
|
-
};
|
|
638
|
-
};
|
|
639
|
-
|
|
640
|
-
// src/oauth/mailchimp.ts
|
|
641
|
-
var mailchimp = (options2) => {
|
|
642
|
-
return {
|
|
643
|
-
id: "mailchimp",
|
|
644
|
-
name: "Mailchimp",
|
|
645
|
-
authorizeURL: "https://login.mailchimp.com/oauth2/authorize",
|
|
646
|
-
accessToken: "https://login.mailchimp.com/oauth2/token",
|
|
647
|
-
userInfo: "https://login.mailchimp.com/oauth2/metadata",
|
|
648
|
-
scope: "",
|
|
649
|
-
responseType: "code",
|
|
650
|
-
profile(profile) {
|
|
651
|
-
return {
|
|
652
|
-
sub: profile.user_id,
|
|
653
|
-
name: profile.accountname,
|
|
654
|
-
email: profile.login.email,
|
|
655
|
-
image: profile.login.avatar
|
|
656
|
-
};
|
|
657
|
-
},
|
|
658
|
-
...options2
|
|
659
|
-
};
|
|
660
|
-
};
|
|
661
|
-
|
|
662
|
-
// src/oauth/pinterest.ts
|
|
663
|
-
var pinterest = (options2) => {
|
|
664
|
-
return {
|
|
665
|
-
id: "pinterest",
|
|
666
|
-
name: "Pinterest",
|
|
667
|
-
authorizeURL: "https://www.pinterest.com/oauth",
|
|
668
|
-
accessToken: "https://api.pinterest.com/v5/oauth/token",
|
|
669
|
-
userInfo: "https://api.pinterest.com/v5/user_account",
|
|
670
|
-
scope: "user_accounts:read",
|
|
671
|
-
responseType: "code",
|
|
672
|
-
profile(profile) {
|
|
673
|
-
return {
|
|
674
|
-
sub: profile.id,
|
|
675
|
-
name: profile.username,
|
|
676
|
-
image: profile.profile_image,
|
|
677
|
-
email: void 0
|
|
678
|
-
};
|
|
679
|
-
},
|
|
680
|
-
...options2
|
|
681
|
-
};
|
|
682
|
-
};
|
|
683
|
-
|
|
684
|
-
// src/schemas.ts
|
|
685
|
-
var import_zod = require("zod");
|
|
686
|
-
var OAuthProviderCredentialsSchema = (0, import_zod.object)({
|
|
687
|
-
id: (0, import_zod.string)(),
|
|
688
|
-
name: (0, import_zod.string)(),
|
|
689
|
-
authorizeURL: (0, import_zod.string)().url(),
|
|
690
|
-
accessToken: (0, import_zod.string)().url(),
|
|
691
|
-
scope: (0, import_zod.string)(),
|
|
692
|
-
userInfo: (0, import_zod.string)().url(),
|
|
693
|
-
responseType: (0, import_zod.enum)(["code", "token", "id_token"]),
|
|
694
|
-
clientId: (0, import_zod.string)(),
|
|
695
|
-
clientSecret: (0, import_zod.string)(),
|
|
696
|
-
profile: import_zod.z.function().optional()
|
|
697
|
-
});
|
|
698
|
-
var OAuthProviderConfigSchema = (0, import_zod.object)({
|
|
699
|
-
authorizeURL: (0, import_zod.string)().url(),
|
|
700
|
-
accessToken: (0, import_zod.string)().url(),
|
|
701
|
-
scope: (0, import_zod.string)().optional(),
|
|
702
|
-
userInfo: (0, import_zod.string)().url(),
|
|
703
|
-
responseType: (0, import_zod.enum)(["code", "token", "id_token"]),
|
|
704
|
-
clientId: (0, import_zod.string)(),
|
|
705
|
-
clientSecret: (0, import_zod.string)()
|
|
706
|
-
});
|
|
707
|
-
var OAuthAuthorization = OAuthProviderConfigSchema.extend({
|
|
708
|
-
redirectURI: (0, import_zod.string)(),
|
|
709
|
-
state: (0, import_zod.string)(),
|
|
710
|
-
codeChallenge: (0, import_zod.string)(),
|
|
711
|
-
codeChallengeMethod: (0, import_zod.enum)(["plain", "S256"])
|
|
712
|
-
});
|
|
713
|
-
var OAuthAuthorizationResponse = (0, import_zod.object)({
|
|
714
|
-
state: (0, import_zod.string)({ message: "Missing state parameter in the OAuth authorization response." }),
|
|
715
|
-
code: (0, import_zod.string)({ message: "Missing code parameter in the OAuth authorization response." })
|
|
716
|
-
});
|
|
717
|
-
var OAuthAuthorizationErrorResponse = (0, import_zod.object)({
|
|
718
|
-
error: (0, import_zod.enum)([
|
|
719
|
-
"invalid_request",
|
|
720
|
-
"unauthorized_client",
|
|
721
|
-
"access_denied",
|
|
722
|
-
"unsupported_response_type",
|
|
723
|
-
"invalid_scope",
|
|
724
|
-
"server_error",
|
|
725
|
-
"temporarily_unavailable"
|
|
726
|
-
]),
|
|
727
|
-
error_description: (0, import_zod.string)().optional(),
|
|
728
|
-
error_uri: (0, import_zod.string)().optional(),
|
|
729
|
-
state: (0, import_zod.string)()
|
|
730
|
-
});
|
|
731
|
-
var OAuthAccessToken = OAuthProviderConfigSchema.extend({
|
|
732
|
-
redirectURI: (0, import_zod.string)(),
|
|
733
|
-
code: (0, import_zod.string)(),
|
|
734
|
-
codeVerifier: (0, import_zod.string)().min(43).max(128)
|
|
735
|
-
});
|
|
736
|
-
var OAuthAccessTokenResponse = (0, import_zod.object)({
|
|
737
|
-
access_token: (0, import_zod.string)(),
|
|
738
|
-
token_type: (0, import_zod.string)().optional(),
|
|
739
|
-
expires_in: (0, import_zod.number)().optional(),
|
|
740
|
-
refresh_token: (0, import_zod.string)().optional(),
|
|
741
|
-
scope: (0, import_zod.string)().optional().or((0, import_zod.null)())
|
|
742
|
-
});
|
|
743
|
-
var OAuthAccessTokenErrorResponse = (0, import_zod.object)({
|
|
744
|
-
error: (0, import_zod.enum)([
|
|
745
|
-
"invalid_request",
|
|
746
|
-
"invalid_client",
|
|
747
|
-
"invalid_grant",
|
|
748
|
-
"unauthorized_client",
|
|
749
|
-
"unsupported_grant_type",
|
|
750
|
-
"invalid_scope"
|
|
751
|
-
]),
|
|
752
|
-
error_description: (0, import_zod.string)().optional(),
|
|
753
|
-
error_uri: (0, import_zod.string)().optional()
|
|
754
|
-
});
|
|
755
|
-
var OAuthErrorResponse = (0, import_zod.object)({
|
|
756
|
-
error: (0, import_zod.string)(),
|
|
757
|
-
error_description: (0, import_zod.string)().optional()
|
|
758
|
-
});
|
|
759
|
-
var OAuthEnvSchema = (0, import_zod.object)({
|
|
760
|
-
clientId: import_zod.z.string().min(1, "OAuth Client ID is required in the environment variables."),
|
|
761
|
-
clientSecret: import_zod.z.string().min(1, "OAuth Client Secret is required in the environment variables.")
|
|
762
|
-
});
|
|
763
|
-
|
|
764
|
-
// src/oauth/index.ts
|
|
765
|
-
var builtInOAuthProviders = {
|
|
766
|
-
github,
|
|
767
|
-
bitbucket,
|
|
768
|
-
figma,
|
|
769
|
-
discord,
|
|
770
|
-
gitlab,
|
|
771
|
-
spotify,
|
|
772
|
-
x,
|
|
773
|
-
strava,
|
|
774
|
-
mailchimp,
|
|
775
|
-
pinterest
|
|
776
|
-
};
|
|
777
|
-
var defineOAuthEnvironment = (oauth) => {
|
|
778
|
-
const clientIdSuffix = `${oauth.toUpperCase()}_CLIENT_ID`;
|
|
779
|
-
const clientSecretSuffix = `${oauth.toUpperCase()}_CLIENT_SECRET`;
|
|
780
|
-
const loadEnvs = OAuthEnvSchema.safeParse({
|
|
781
|
-
clientId: env[`AURA_AUTH_${clientIdSuffix}`] ?? env[`AUTH_${clientIdSuffix}`] ?? env[`${clientIdSuffix}`],
|
|
782
|
-
clientSecret: env[`AURA_AUTH_${clientSecretSuffix}`] ?? env[`AUTH_${clientSecretSuffix}`] ?? env[`${clientSecretSuffix}`]
|
|
783
|
-
});
|
|
784
|
-
if (!loadEnvs.success) {
|
|
785
|
-
const msg = JSON.stringify(formatZodError(loadEnvs.error), null, 2);
|
|
786
|
-
throw new AuthInternalError("INVALID_ENVIRONMENT_CONFIGURATION", msg);
|
|
787
|
-
}
|
|
788
|
-
return loadEnvs.data;
|
|
789
|
-
};
|
|
790
|
-
var defineOAuthProviderConfig = (config2) => {
|
|
791
|
-
if (typeof config2 === "string") {
|
|
792
|
-
const definition = defineOAuthEnvironment(config2);
|
|
793
|
-
const oauthConfig = builtInOAuthProviders[config2]();
|
|
794
|
-
const parsed2 = OAuthProviderCredentialsSchema.safeParse({ ...oauthConfig, ...definition });
|
|
795
|
-
if (!parsed2.success) {
|
|
796
|
-
const details = JSON.stringify(formatZodError(parsed2.error), null, 2);
|
|
797
|
-
throw new AuthInternalError(
|
|
798
|
-
"INVALID_OAUTH_PROVIDER_CONFIGURATION",
|
|
799
|
-
`Invalid configuration for OAuth provider "${config2}": ${details}`
|
|
800
|
-
);
|
|
801
|
-
}
|
|
802
|
-
return parsed2.data;
|
|
803
|
-
}
|
|
804
|
-
const hasCredentials = config2.clientId && config2.clientSecret;
|
|
805
|
-
const envConfig = hasCredentials ? {} : defineOAuthEnvironment(config2.id);
|
|
806
|
-
const parsed = OAuthProviderCredentialsSchema.safeParse({ ...envConfig, ...config2 });
|
|
807
|
-
if (!parsed.success) {
|
|
808
|
-
const details = JSON.stringify(formatZodError(parsed.error), null, 2);
|
|
809
|
-
throw new AuthInternalError(
|
|
810
|
-
"INVALID_OAUTH_PROVIDER_CONFIGURATION",
|
|
811
|
-
`Invalid configuration for OAuth provider "${config2.id}": ${details}`
|
|
812
|
-
);
|
|
813
|
-
}
|
|
814
|
-
return parsed.data;
|
|
815
|
-
};
|
|
816
|
-
var createBuiltInOAuthProviders = (oauth = []) => {
|
|
817
|
-
return oauth.reduce((previous, config2) => {
|
|
818
|
-
const oauthConfig = defineOAuthProviderConfig(config2);
|
|
819
|
-
if (oauthConfig.id in previous) {
|
|
820
|
-
throw new AuthInternalError(
|
|
821
|
-
"DUPLICATED_OAUTH_PROVIDER_ID",
|
|
822
|
-
`Duplicate OAuth provider id "${oauthConfig.id}" found. Each provider must have a unique id.`
|
|
823
|
-
);
|
|
824
|
-
}
|
|
825
|
-
return { ...previous, [oauthConfig.id]: oauthConfig };
|
|
826
|
-
}, {});
|
|
827
|
-
};
|
|
828
|
-
|
|
829
|
-
// src/actions/signIn/signIn.ts
|
|
830
|
-
var import_zod2 = require("zod");
|
|
831
|
-
var import_router2 = require("@aura-stack/router");
|
|
832
|
-
|
|
833
|
-
// src/headers.ts
|
|
834
|
-
var cacheControl = {
|
|
835
|
-
"Cache-Control": "no-store",
|
|
836
|
-
Pragma: "no-cache",
|
|
837
|
-
Expires: "0",
|
|
838
|
-
Vary: "Cookie"
|
|
839
|
-
};
|
|
840
|
-
var contentSecurityPolicy = {
|
|
841
|
-
"Content-Security-Policy": [
|
|
842
|
-
"default-src 'none'",
|
|
843
|
-
"script-src 'self'",
|
|
844
|
-
"frame-src 'none'",
|
|
845
|
-
"object-src 'none'",
|
|
846
|
-
"frame-ancestors 'none'",
|
|
847
|
-
"base-uri 'none'"
|
|
848
|
-
].join("; ")
|
|
849
|
-
};
|
|
850
|
-
var secureHeaders = {
|
|
851
|
-
"X-Content-Type-Options": "nosniff",
|
|
852
|
-
"X-Frame-Options": "DENY",
|
|
853
|
-
"Referrer-Policy": "strict-origin-when-cross-origin"
|
|
854
|
-
};
|
|
855
|
-
var secureApiHeaders = {
|
|
856
|
-
...cacheControl,
|
|
857
|
-
...contentSecurityPolicy,
|
|
858
|
-
...secureHeaders
|
|
859
|
-
};
|
|
860
|
-
|
|
861
|
-
// src/secure.ts
|
|
862
|
-
var import_crypto2 = __toESM(require("crypto"), 1);
|
|
863
|
-
|
|
864
|
-
// src/assert.ts
|
|
865
|
-
var import_crypto = require("crypto");
|
|
866
|
-
var unsafeChars = [
|
|
867
|
-
"<",
|
|
868
|
-
">",
|
|
869
|
-
'"',
|
|
870
|
-
"`",
|
|
871
|
-
" ",
|
|
872
|
-
"\r",
|
|
873
|
-
"\n",
|
|
874
|
-
" ",
|
|
875
|
-
"\\",
|
|
876
|
-
"%2F",
|
|
877
|
-
"%5C",
|
|
878
|
-
"%2f",
|
|
879
|
-
"%5c",
|
|
880
|
-
"\r\n",
|
|
881
|
-
"%0A",
|
|
882
|
-
"%0D",
|
|
883
|
-
"%0a",
|
|
884
|
-
"%0d",
|
|
885
|
-
"..",
|
|
886
|
-
"//",
|
|
887
|
-
"///",
|
|
888
|
-
"...",
|
|
889
|
-
"%20",
|
|
890
|
-
"\0"
|
|
891
|
-
];
|
|
892
|
-
var isValidURL = (value) => {
|
|
893
|
-
if (!new RegExp(/^https?:\/\/[^/]/).test(value)) {
|
|
894
|
-
return false;
|
|
895
|
-
}
|
|
896
|
-
const match = value.match(/^(https?:\/\/)(.*)$/);
|
|
897
|
-
if (!match) return false;
|
|
898
|
-
const rest = match[2];
|
|
899
|
-
for (const char of unsafeChars) {
|
|
900
|
-
if (rest.includes(char)) return false;
|
|
901
|
-
}
|
|
902
|
-
const regex = /^https?:\/\/(?:[a-zA-Z0-9._-]+|localhost|\[[0-9a-fA-F:]+\])(?::\d{1,5})?(?:\/[a-zA-Z0-9._~!$&'()?#*+,;=:@-]*)*\/?$/;
|
|
903
|
-
return regex.test(match[0]);
|
|
904
|
-
};
|
|
905
|
-
var isJWTPayloadWithToken = (payload) => {
|
|
906
|
-
return typeof payload === "object" && payload !== null && "token" in payload && typeof payload?.token === "string";
|
|
907
|
-
};
|
|
908
|
-
var isRelativeURL = (value) => {
|
|
909
|
-
if (value.length > 100) return false;
|
|
910
|
-
for (const char of unsafeChars) {
|
|
911
|
-
if (value.includes(char)) return false;
|
|
912
|
-
}
|
|
913
|
-
const regex = /^\/[a-zA-Z0-9\-_\/.?&=#]*\/?$/;
|
|
914
|
-
return regex.test(value);
|
|
915
|
-
};
|
|
916
|
-
var isSameOrigin = (origin, expected) => {
|
|
917
|
-
const originURL = new URL(origin);
|
|
918
|
-
const expectedURL = new URL(expected);
|
|
919
|
-
return equals(originURL.origin, expectedURL.origin);
|
|
920
|
-
};
|
|
921
|
-
var patternToRegex = (pattern) => {
|
|
922
|
-
try {
|
|
923
|
-
if (pattern.length > 2048) return null;
|
|
924
|
-
pattern = pattern.replace(/\\/g, "");
|
|
925
|
-
const match = pattern.match(/^(https?):\/\/([a-zA-Z0-9.*-]{1,253})(?::(\d{1,5}|\*))?(?:\/.*)?$/);
|
|
926
|
-
if (!match) return null;
|
|
927
|
-
const [, protocol, host, port] = match;
|
|
928
|
-
const hasWildcard = host.includes("*");
|
|
929
|
-
if (hasWildcard && !host.startsWith("*.")) return null;
|
|
930
|
-
if (hasWildcard && host.slice(2).includes("*")) return null;
|
|
931
|
-
const domain = hasWildcard ? host.slice(2) : host;
|
|
932
|
-
const escapedDomain = domain.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
|
933
|
-
const hostRegex = hasWildcard ? `[^.]+\\.${escapedDomain}` : escapedDomain;
|
|
934
|
-
const portRegex = port === "*" ? ":\\d{1,5}" : port ? `:${port}` : "";
|
|
935
|
-
return new RegExp(`^${protocol}:\\/\\/${hostRegex}${portRegex}$`);
|
|
936
|
-
} catch {
|
|
937
|
-
return null;
|
|
938
|
-
}
|
|
939
|
-
};
|
|
940
|
-
var isTrustedOrigin = (url, trustedOrigins) => {
|
|
941
|
-
if (!isValidURL(url) || trustedOrigins.length === 0) return false;
|
|
942
|
-
try {
|
|
943
|
-
const urlOrigin = new URL(url).origin;
|
|
944
|
-
for (const pattern of trustedOrigins) {
|
|
945
|
-
const regex = patternToRegex(pattern);
|
|
946
|
-
if (regex?.test(urlOrigin)) return true;
|
|
947
|
-
try {
|
|
948
|
-
if (isValidURL(pattern) && equals(new URL(pattern).origin, urlOrigin)) return true;
|
|
949
|
-
} catch {
|
|
950
|
-
}
|
|
951
|
-
}
|
|
952
|
-
} catch {
|
|
953
|
-
}
|
|
954
|
-
return false;
|
|
955
|
-
};
|
|
956
|
-
var safeEquals = (a, b) => {
|
|
957
|
-
const bufferA = Buffer.from(a);
|
|
958
|
-
const bufferB = Buffer.from(b);
|
|
959
|
-
if (bufferA.length !== bufferB.length) {
|
|
960
|
-
return false;
|
|
961
|
-
}
|
|
962
|
-
return (0, import_crypto.timingSafeEqual)(bufferA, bufferB);
|
|
963
|
-
};
|
|
964
|
-
|
|
965
|
-
// src/secure.ts
|
|
966
|
-
var generateSecure = (length = 32) => {
|
|
967
|
-
return import_crypto2.default.randomBytes(length).toString("base64url");
|
|
968
|
-
};
|
|
969
|
-
var createHash = (data, base = "hex") => {
|
|
970
|
-
return import_crypto2.default.createHash("sha256").update(data).digest().toString(base);
|
|
971
|
-
};
|
|
972
|
-
var createPKCE = async (verifier) => {
|
|
973
|
-
const byteLength = verifier ? void 0 : Math.floor(Math.random() * (96 - 32 + 1) + 32);
|
|
974
|
-
const codeVerifier = verifier ?? generateSecure(byteLength ?? 64);
|
|
975
|
-
if (codeVerifier.length < 43 || codeVerifier.length > 128) {
|
|
976
|
-
throw new AuthSecurityError("PKCE_VERIFIER_INVALID", "The code verifier must be between 43 and 128 characters in length.");
|
|
977
|
-
}
|
|
978
|
-
const codeChallenge = createHash(codeVerifier, "base64url");
|
|
979
|
-
return { codeVerifier, codeChallenge, method: "S256" };
|
|
980
|
-
};
|
|
981
|
-
var createCSRF = async (jose, csrfCookie) => {
|
|
982
|
-
try {
|
|
983
|
-
const token = generateSecure(32);
|
|
984
|
-
if (csrfCookie) {
|
|
985
|
-
await jose.verifyJWS(csrfCookie, jwtVerificationOptions);
|
|
986
|
-
return csrfCookie;
|
|
987
|
-
}
|
|
988
|
-
return jose.signJWS({ token });
|
|
989
|
-
} catch {
|
|
990
|
-
const token = generateSecure(32);
|
|
991
|
-
return jose.signJWS({ token });
|
|
992
|
-
}
|
|
993
|
-
};
|
|
994
|
-
var verifyCSRF = async (jose, cookie, header) => {
|
|
995
|
-
try {
|
|
996
|
-
const cookiePayload = await jose.verifyJWS(cookie, jwtVerificationOptions);
|
|
997
|
-
const headerPayload = await jose.verifyJWS(header, jwtVerificationOptions);
|
|
998
|
-
if (!isJWTPayloadWithToken(cookiePayload)) {
|
|
999
|
-
throw new AuthSecurityError("CSRF_TOKEN_INVALID", "Cookie payload missing token field.");
|
|
1000
|
-
}
|
|
1001
|
-
if (!isJWTPayloadWithToken(headerPayload)) {
|
|
1002
|
-
throw new AuthSecurityError("CSRF_TOKEN_INVALID", "Header payload missing token field.");
|
|
1003
|
-
}
|
|
1004
|
-
if (!equals(cookiePayload.token.length, headerPayload.token.length)) {
|
|
1005
|
-
throw new AuthSecurityError("CSRF_TOKEN_INVALID", "The CSRF tokens do not match.");
|
|
1006
|
-
}
|
|
1007
|
-
if (!safeEquals(cookiePayload.token, headerPayload.token)) {
|
|
1008
|
-
throw new AuthSecurityError("CSRF_TOKEN_INVALID", "The CSRF tokens do not match.");
|
|
1009
|
-
}
|
|
1010
|
-
return true;
|
|
1011
|
-
} catch {
|
|
1012
|
-
throw new AuthSecurityError("CSRF_TOKEN_INVALID", "The CSRF tokens do not match.");
|
|
1013
|
-
}
|
|
1014
|
-
};
|
|
1015
|
-
|
|
1016
|
-
// src/actions/signIn/authorization.ts
|
|
1017
|
-
var createAuthorizationURL = (oauthConfig, redirectURI, state, codeChallenge, codeChallengeMethod, logger) => {
|
|
1018
|
-
const parsed = OAuthAuthorization.safeParse({ ...oauthConfig, redirectURI, state, codeChallenge, codeChallengeMethod });
|
|
1019
|
-
if (!parsed.success) {
|
|
1020
|
-
logger?.log("INVALID_OAUTH_CONFIGURATION", {
|
|
1021
|
-
structuredData: {
|
|
1022
|
-
scope: oauthConfig.scope,
|
|
1023
|
-
redirect_uri: redirectURI,
|
|
1024
|
-
has_state: Boolean(state),
|
|
1025
|
-
has_code_challenge: Boolean(codeChallenge),
|
|
1026
|
-
code_challenge_method: codeChallengeMethod
|
|
1027
|
-
}
|
|
1028
|
-
});
|
|
1029
|
-
throw new AuthInternalError("INVALID_OAUTH_CONFIGURATION", "The OAuth provider configuration is invalid.");
|
|
1030
|
-
}
|
|
1031
|
-
const { authorizeURL, ...options2 } = parsed.data;
|
|
1032
|
-
const { userInfo, accessToken, clientSecret, ...required } = options2;
|
|
1033
|
-
const searchParams = new URLSearchParams(toCastCase(required));
|
|
1034
|
-
return `${authorizeURL}?${searchParams}`;
|
|
1035
|
-
};
|
|
1036
|
-
var getTrustedOrigins = async (request, trustedOrigins) => {
|
|
1037
|
-
if (!trustedOrigins) return [];
|
|
1038
|
-
const raw = typeof trustedOrigins === "function" ? await trustedOrigins(request) : trustedOrigins;
|
|
1039
|
-
return Array.isArray(raw) ? raw : typeof raw === "string" ? [raw] : [];
|
|
1040
|
-
};
|
|
1041
|
-
var getOriginURL = async (request, context) => {
|
|
1042
|
-
const headers = request.headers;
|
|
1043
|
-
let origin = new URL(request.url).origin;
|
|
1044
|
-
const trustedOrigins = await getTrustedOrigins(request, context?.trustedOrigins);
|
|
1045
|
-
trustedOrigins.push(origin);
|
|
1046
|
-
if (context?.trustedProxyHeaders) {
|
|
1047
|
-
const protocol = headers.get("Forwarded")?.match(/proto=([^;]+)/i)?.[1] ?? headers.get("X-Forwarded-Proto") ?? "http";
|
|
1048
|
-
const host = headers.get("Host") ?? headers.get("Forwarded")?.match(/host=([^;]+)/i)?.[1] ?? headers.get("X-Forwarded-Host") ?? null;
|
|
1049
|
-
origin = `${protocol}://${host}`;
|
|
1050
|
-
}
|
|
1051
|
-
if (!isTrustedOrigin(origin, trustedOrigins)) {
|
|
1052
|
-
context?.logger?.log("UNTRUSTED_ORIGIN", { structuredData: { origin } });
|
|
1053
|
-
throw new AuthInternalError("UNTRUSTED_ORIGIN", "The constructed origin URL is not trusted.");
|
|
1054
|
-
}
|
|
1055
|
-
return origin;
|
|
1056
|
-
};
|
|
1057
|
-
var createRedirectURI = async (request, oauth, context) => {
|
|
1058
|
-
const origin = await getOriginURL(request, context);
|
|
1059
|
-
return `${origin}${context.basePath}/callback/${oauth}`;
|
|
1060
|
-
};
|
|
1061
|
-
var createRedirectTo = async (request, redirectTo, context) => {
|
|
1062
|
-
try {
|
|
1063
|
-
const headers = request.headers;
|
|
1064
|
-
const requestOrigin = await getOriginURL(request, context);
|
|
1065
|
-
const origins = await getTrustedOrigins(request, context?.trustedOrigins);
|
|
1066
|
-
const validateURL = (url) => {
|
|
1067
|
-
if (!isRelativeURL(url) && !isValidURL(url)) return "/";
|
|
1068
|
-
if (isRelativeURL(url)) return url;
|
|
1069
|
-
if (origins.length > 0) {
|
|
1070
|
-
if (isTrustedOrigin(url, origins)) {
|
|
1071
|
-
const urlOrigin = new URL(url).origin;
|
|
1072
|
-
for (const pattern of origins) {
|
|
1073
|
-
const regex = patternToRegex(pattern);
|
|
1074
|
-
if (regex?.test(urlOrigin)) {
|
|
1075
|
-
return isSameOrigin(url, request.url) ? extractPath(url) : url;
|
|
1076
|
-
}
|
|
1077
|
-
if (isValidURL(pattern) && equals(new URL(pattern).origin, urlOrigin)) return url;
|
|
1078
|
-
}
|
|
1079
|
-
}
|
|
1080
|
-
context?.logger?.log("OPEN_REDIRECT_ATTACK");
|
|
1081
|
-
return "/";
|
|
1082
|
-
}
|
|
1083
|
-
if (isSameOrigin(url, requestOrigin)) {
|
|
1084
|
-
return extractPath(url);
|
|
1085
|
-
}
|
|
1086
|
-
context?.logger?.log("OPEN_REDIRECT_ATTACK");
|
|
1087
|
-
return "/";
|
|
1088
|
-
};
|
|
1089
|
-
return validateURL(redirectTo ?? headers.get("Referer") ?? headers.get("Origin") ?? "/");
|
|
1090
|
-
} catch (error) {
|
|
1091
|
-
context?.logger?.log("POTENTIAL_OPEN_REDIRECT_ATTACK_DETECTED");
|
|
1092
|
-
return "/";
|
|
1093
|
-
}
|
|
1094
|
-
};
|
|
1095
|
-
|
|
1096
|
-
// src/actions/signIn/signIn.ts
|
|
1097
|
-
var signInConfig = (oauth) => {
|
|
1098
|
-
return (0, import_router2.createEndpointConfig)("/signIn/:oauth", {
|
|
1099
|
-
schemas: {
|
|
1100
|
-
params: import_zod2.z.object({
|
|
1101
|
-
oauth: import_zod2.z.enum(
|
|
1102
|
-
Object.keys(oauth),
|
|
1103
|
-
"The OAuth provider is not supported or invalid."
|
|
1104
|
-
)
|
|
1105
|
-
}),
|
|
1106
|
-
searchParams: import_zod2.z.object({
|
|
1107
|
-
redirectTo: import_zod2.z.string().optional()
|
|
1108
|
-
})
|
|
1109
|
-
}
|
|
1110
|
-
});
|
|
1111
|
-
};
|
|
1112
|
-
var signInAction = (oauth) => {
|
|
1113
|
-
return (0, import_router2.createEndpoint)(
|
|
1114
|
-
"GET",
|
|
1115
|
-
"/signIn/:oauth",
|
|
1116
|
-
async (ctx) => {
|
|
1117
|
-
const {
|
|
1118
|
-
request,
|
|
1119
|
-
params: { oauth: oauth2 },
|
|
1120
|
-
searchParams: { redirectTo },
|
|
1121
|
-
context
|
|
1122
|
-
} = ctx;
|
|
1123
|
-
const { oauth: providers, cookies, logger } = context;
|
|
1124
|
-
const state = generateSecure();
|
|
1125
|
-
const redirectURI = await createRedirectURI(request, oauth2, context);
|
|
1126
|
-
const redirectToValue = await createRedirectTo(request, redirectTo, context);
|
|
1127
|
-
const { codeVerifier, codeChallenge, method } = await createPKCE();
|
|
1128
|
-
const authorization = createAuthorizationURL(providers[oauth2], redirectURI, state, codeChallenge, method, logger);
|
|
1129
|
-
logger?.log("SIGN_IN_INITIATED", {
|
|
1130
|
-
structuredData: { oauth_provider: oauth2, code_challenge_method: method }
|
|
1131
|
-
});
|
|
1132
|
-
const headers = new import_router2.HeadersBuilder(cacheControl).setHeader("Location", authorization).setCookie(cookies.state.name, state, cookies.state.attributes).setCookie(cookies.redirectURI.name, redirectURI, cookies.redirectURI.attributes).setCookie(cookies.redirectTo.name, redirectToValue, cookies.redirectTo.attributes).setCookie(cookies.codeVerifier.name, codeVerifier, cookies.codeVerifier.attributes).toHeaders();
|
|
1133
|
-
return Response.json(
|
|
1134
|
-
{ oauth: oauth2 },
|
|
1135
|
-
{
|
|
1136
|
-
status: 302,
|
|
1137
|
-
headers
|
|
1138
|
-
}
|
|
1139
|
-
);
|
|
1140
|
-
},
|
|
1141
|
-
signInConfig(oauth)
|
|
1142
|
-
);
|
|
1143
|
-
};
|
|
1144
|
-
|
|
1145
|
-
// src/actions/callback/callback.ts
|
|
1146
|
-
var import_zod3 = require("zod");
|
|
1147
|
-
var import_router3 = require("@aura-stack/router");
|
|
1148
|
-
|
|
1149
|
-
// src/request.ts
|
|
1150
|
-
var fetchAsync = async (url, options2 = {}, timeout = 5e3) => {
|
|
1151
|
-
const controller = new AbortController();
|
|
1152
|
-
const timeoutId = setTimeout(() => controller.abort(), timeout);
|
|
1153
|
-
const response = await fetch(url, {
|
|
1154
|
-
...options2,
|
|
1155
|
-
signal: controller.signal
|
|
1156
|
-
}).finally(() => clearTimeout(timeoutId));
|
|
1157
|
-
return response;
|
|
1158
|
-
};
|
|
1159
|
-
|
|
1160
|
-
// src/actions/callback/userinfo.ts
|
|
1161
|
-
var getDefaultUserInfo = (profile) => {
|
|
1162
|
-
const sub = generateSecure(16);
|
|
1163
|
-
return {
|
|
1164
|
-
sub: profile?.id ?? profile?.sub ?? sub,
|
|
1165
|
-
email: profile?.email,
|
|
1166
|
-
name: profile?.name ?? profile?.username ?? profile?.nickname,
|
|
1167
|
-
image: profile?.image ?? profile?.picture
|
|
1168
|
-
};
|
|
1169
|
-
};
|
|
1170
|
-
var getUserInfo = async (oauthConfig, accessToken, logger) => {
|
|
1171
|
-
const userinfoEndpoint = oauthConfig.userInfo;
|
|
1172
|
-
try {
|
|
1173
|
-
logger?.log("OAUTH_USERINFO_REQUEST_INITIATED", {
|
|
1174
|
-
structuredData: {
|
|
1175
|
-
endpoint: userinfoEndpoint
|
|
1176
|
-
}
|
|
1177
|
-
});
|
|
1178
|
-
const response = await fetchAsync(userinfoEndpoint, {
|
|
1179
|
-
method: "GET",
|
|
1180
|
-
headers: {
|
|
1181
|
-
Accept: "application/json",
|
|
1182
|
-
Authorization: `Bearer ${accessToken}`
|
|
1183
|
-
}
|
|
1184
|
-
});
|
|
1185
|
-
if (!response.ok) {
|
|
1186
|
-
logger?.log("OAUTH_USERINFO_INVALID_RESPONSE");
|
|
1187
|
-
throw new OAuthProtocolError("INVALID_REQUEST", "Invalid userinfo response format");
|
|
1188
|
-
}
|
|
1189
|
-
const json = await response.json();
|
|
1190
|
-
const { success, data } = OAuthErrorResponse.safeParse(json);
|
|
1191
|
-
if (success) {
|
|
1192
|
-
logger?.log("OAUTH_USERINFO_ERROR", {
|
|
1193
|
-
message: "Error response received from OAuth userinfo endpoint",
|
|
1194
|
-
structuredData: {
|
|
1195
|
-
error: data.error,
|
|
1196
|
-
error_description: data.error_description ?? ""
|
|
1197
|
-
}
|
|
1198
|
-
});
|
|
1199
|
-
throw new OAuthProtocolError("INVALID_REQUEST", "An error was received from the OAuth userinfo endpoint.");
|
|
1200
|
-
}
|
|
1201
|
-
logger?.log("OAUTH_USERINFO_SUCCESS");
|
|
1202
|
-
return oauthConfig?.profile ? oauthConfig.profile(json) : getDefaultUserInfo(json);
|
|
1203
|
-
} catch (error) {
|
|
1204
|
-
if (isOAuthProtocolError(error)) {
|
|
1205
|
-
throw error;
|
|
1206
|
-
}
|
|
1207
|
-
logger?.log("OAUTH_USERINFO_REQUEST_FAILED");
|
|
1208
|
-
if (isNativeError(error)) {
|
|
1209
|
-
throw new OAuthProtocolError("SERVER_ERROR", "Failed to fetch user information from OAuth provider", "", {
|
|
1210
|
-
cause: error
|
|
1211
|
-
});
|
|
1212
|
-
}
|
|
1213
|
-
throw new OAuthProtocolError("SERVER_ERROR", "Failed to fetch user information", "", { cause: error });
|
|
1214
|
-
}
|
|
1215
|
-
};
|
|
1216
|
-
|
|
1217
|
-
// src/actions/callback/access-token.ts
|
|
1218
|
-
var createAccessToken = async (oauthConfig, redirectURI, code, codeVerifier, logger) => {
|
|
1219
|
-
const parsed = OAuthAccessToken.safeParse({ ...oauthConfig, redirectURI, code, codeVerifier });
|
|
1220
|
-
if (!parsed.success) {
|
|
1221
|
-
logger?.log("INVALID_OAUTH_CONFIGURATION");
|
|
1222
|
-
throw new AuthInternalError("INVALID_OAUTH_CONFIGURATION", "The OAuth provider configuration is invalid.");
|
|
1223
|
-
}
|
|
1224
|
-
const { accessToken, clientId, clientSecret, code: codeParsed, redirectURI: redirectParsed } = parsed.data;
|
|
1225
|
-
try {
|
|
1226
|
-
logger?.log("OAUTH_ACCESS_TOKEN_REQUEST_INITIATED", {
|
|
1227
|
-
structuredData: {
|
|
1228
|
-
has_client_id: Boolean(clientId),
|
|
1229
|
-
redirect_uri: redirectParsed,
|
|
1230
|
-
grant_type: "authorization_code"
|
|
1231
|
-
}
|
|
1232
|
-
});
|
|
1233
|
-
const response = await fetchAsync(accessToken, {
|
|
1234
|
-
method: "POST",
|
|
1235
|
-
headers: {
|
|
1236
|
-
Accept: "application/json",
|
|
1237
|
-
"Content-Type": "application/x-www-form-urlencoded"
|
|
1238
|
-
},
|
|
1239
|
-
body: new URLSearchParams({
|
|
1240
|
-
client_id: clientId,
|
|
1241
|
-
client_secret: clientSecret,
|
|
1242
|
-
code: codeParsed,
|
|
1243
|
-
redirect_uri: redirectParsed,
|
|
1244
|
-
grant_type: "authorization_code",
|
|
1245
|
-
code_verifier: codeVerifier
|
|
1246
|
-
}).toString()
|
|
1247
|
-
});
|
|
1248
|
-
if (!response.ok) {
|
|
1249
|
-
logger?.log("INVALID_OAUTH_ACCESS_TOKEN_RESPONSE");
|
|
1250
|
-
throw new OAuthProtocolError("invalid_request", "Invalid access token response");
|
|
1251
|
-
}
|
|
1252
|
-
const json = await response.json();
|
|
1253
|
-
const token = OAuthAccessTokenResponse.safeParse(json);
|
|
1254
|
-
if (!token.success) {
|
|
1255
|
-
const { success, data } = OAuthAccessTokenErrorResponse.safeParse(json);
|
|
1256
|
-
if (!success) {
|
|
1257
|
-
logger?.log("INVALID_OAUTH_ACCESS_TOKEN_RESPONSE");
|
|
1258
|
-
throw new OAuthProtocolError("invalid_request", "Invalid access token response format");
|
|
1259
|
-
}
|
|
1260
|
-
logger?.log("OAUTH_ACCESS_TOKEN_ERROR", {
|
|
1261
|
-
structuredData: {
|
|
1262
|
-
error: data.error,
|
|
1263
|
-
error_description: data.error_description ?? ""
|
|
1264
|
-
}
|
|
1265
|
-
});
|
|
1266
|
-
throw new OAuthProtocolError("INVALID_ACCESS_TOKEN", "Failed to retrieve access token");
|
|
1267
|
-
}
|
|
1268
|
-
logger?.log("OAUTH_ACCESS_TOKEN_SUCCESS");
|
|
1269
|
-
return token.data;
|
|
1270
|
-
} catch (error) {
|
|
1271
|
-
logger?.log("OAUTH_ACCESS_TOKEN_REQUEST_FAILED");
|
|
1272
|
-
if (error instanceof Error) {
|
|
1273
|
-
throw new OAuthProtocolError("server_error", "Failed to communicate with OAuth provider", "", { cause: error });
|
|
1274
|
-
}
|
|
1275
|
-
throw error;
|
|
1276
|
-
}
|
|
1277
|
-
};
|
|
1278
|
-
|
|
1279
|
-
// src/actions/callback/callback.ts
|
|
1280
|
-
var callbackConfig = (oauth) => {
|
|
1281
|
-
return (0, import_router3.createEndpointConfig)("/callback/:oauth", {
|
|
1282
|
-
schemas: {
|
|
1283
|
-
params: import_zod3.z.object({
|
|
1284
|
-
oauth: import_zod3.z.enum(
|
|
1285
|
-
Object.keys(oauth),
|
|
1286
|
-
"The OAuth provider is not supported or invalid."
|
|
1287
|
-
)
|
|
1288
|
-
}),
|
|
1289
|
-
searchParams: import_zod3.z.object({
|
|
1290
|
-
code: import_zod3.z.string("Missing code parameter in the OAuth authorization response."),
|
|
1291
|
-
state: import_zod3.z.string("Missing state parameter in the OAuth authorization response.")
|
|
1292
|
-
})
|
|
1293
|
-
},
|
|
1294
|
-
middlewares: [
|
|
1295
|
-
(ctx) => {
|
|
1296
|
-
const {
|
|
1297
|
-
searchParams,
|
|
1298
|
-
context: { logger }
|
|
1299
|
-
} = ctx;
|
|
1300
|
-
const response = OAuthAuthorizationErrorResponse.safeParse(searchParams);
|
|
1301
|
-
if (response.success) {
|
|
1302
|
-
const { error, error_description } = response.data;
|
|
1303
|
-
const criticalAuthErrors = ["access_denied", "server_error"];
|
|
1304
|
-
const severity = criticalAuthErrors.includes(error.toLowerCase()) ? "critical" : "warning";
|
|
1305
|
-
logger?.log("OAUTH_AUTHORIZATION_ERROR", {
|
|
1306
|
-
severity,
|
|
1307
|
-
structuredData: {
|
|
1308
|
-
error,
|
|
1309
|
-
error_description: error_description ?? ""
|
|
1310
|
-
}
|
|
1311
|
-
});
|
|
1312
|
-
throw new OAuthProtocolError(error, error_description || "OAuth Authorization Error");
|
|
1313
|
-
}
|
|
1314
|
-
return ctx;
|
|
1315
|
-
}
|
|
1316
|
-
]
|
|
1317
|
-
});
|
|
1318
|
-
};
|
|
1319
|
-
var callbackAction = (oauth) => {
|
|
1320
|
-
return (0, import_router3.createEndpoint)(
|
|
1321
|
-
"GET",
|
|
1322
|
-
"/callback/:oauth",
|
|
1323
|
-
async (ctx) => {
|
|
1324
|
-
const {
|
|
1325
|
-
request,
|
|
1326
|
-
params: { oauth: oauth2 },
|
|
1327
|
-
searchParams: { code, state },
|
|
1328
|
-
context
|
|
1329
|
-
} = ctx;
|
|
1330
|
-
const { oauth: providers, cookies, jose, logger, trustedOrigins } = context;
|
|
1331
|
-
const oauthConfig = providers[oauth2];
|
|
1332
|
-
const cookieState = getCookie(request, cookies.state.name);
|
|
1333
|
-
const codeVerifier = getCookie(request, cookies.codeVerifier.name);
|
|
1334
|
-
const cookieRedirectTo = getCookie(request, cookies.redirectTo.name);
|
|
1335
|
-
const cookieRedirectURI = getCookie(request, cookies.redirectURI.name);
|
|
1336
|
-
if (!safeEquals(cookieState, state)) {
|
|
1337
|
-
logger?.log("MISMATCHING_STATE", {
|
|
1338
|
-
structuredData: {
|
|
1339
|
-
oauth_provider: oauth2
|
|
1340
|
-
}
|
|
1341
|
-
});
|
|
1342
|
-
throw new AuthSecurityError(
|
|
1343
|
-
"MISMATCHING_STATE",
|
|
1344
|
-
"The provided state passed in the OAuth response does not match the stored state."
|
|
1345
|
-
);
|
|
1346
|
-
}
|
|
1347
|
-
const accessToken = await createAccessToken(oauthConfig, cookieRedirectURI, code, codeVerifier, logger);
|
|
1348
|
-
const origins = await getTrustedOrigins(request, trustedOrigins);
|
|
1349
|
-
const requestOrigin = await getOriginURL(request, context);
|
|
1350
|
-
if (!isRelativeURL(cookieRedirectTo)) {
|
|
1351
|
-
const isValid = origins.length > 0 ? isTrustedOrigin(cookieRedirectTo, origins) : isSameOrigin(cookieRedirectTo, requestOrigin);
|
|
1352
|
-
if (!isValid) {
|
|
1353
|
-
logger?.log("POTENTIAL_OPEN_REDIRECT_ATTACK_DETECTED", {
|
|
1354
|
-
structuredData: {
|
|
1355
|
-
redirect_path: cookieRedirectTo,
|
|
1356
|
-
provider: oauth2,
|
|
1357
|
-
has_trusted_origins: origins.length > 0,
|
|
1358
|
-
request_origin: requestOrigin
|
|
1359
|
-
}
|
|
1360
|
-
});
|
|
1361
|
-
throw new AuthSecurityError(
|
|
1362
|
-
"POTENTIAL_OPEN_REDIRECT_ATTACK_DETECTED",
|
|
1363
|
-
"Invalid redirect path. Potential open redirect attack detected."
|
|
1364
|
-
);
|
|
1365
|
-
}
|
|
1366
|
-
}
|
|
1367
|
-
const userInfo = await getUserInfo(oauthConfig, accessToken.access_token, logger);
|
|
1368
|
-
const sessionCookie = await createSessionCookie(jose, userInfo);
|
|
1369
|
-
const csrfToken = await createCSRF(jose);
|
|
1370
|
-
logger?.log("OAUTH_CALLBACK_SUCCESS", {
|
|
1371
|
-
structuredData: {
|
|
1372
|
-
provider: oauth2
|
|
1373
|
-
}
|
|
1374
|
-
});
|
|
1375
|
-
const headers = new import_router3.HeadersBuilder(cacheControl).setHeader("Location", cookieRedirectTo).setCookie(cookies.sessionToken.name, sessionCookie, cookies.sessionToken.attributes).setCookie(cookies.csrfToken.name, csrfToken, cookies.csrfToken.attributes).setCookie(cookies.state.name, "", expiredCookieAttributes).setCookie(cookies.redirectURI.name, "", expiredCookieAttributes).setCookie(cookies.redirectTo.name, "", expiredCookieAttributes).setCookie(cookies.codeVerifier.name, "", expiredCookieAttributes).toHeaders();
|
|
1376
|
-
return Response.json({ oauth: oauth2 }, { status: 302, headers });
|
|
1377
|
-
},
|
|
1378
|
-
callbackConfig(oauth)
|
|
1379
|
-
);
|
|
1380
|
-
};
|
|
1381
|
-
|
|
1382
|
-
// src/actions/session/session.ts
|
|
1383
|
-
var import_router4 = require("@aura-stack/router");
|
|
1384
|
-
var sessionAction = (0, import_router4.createEndpoint)("GET", "/session", async (ctx) => {
|
|
1385
|
-
const {
|
|
1386
|
-
request,
|
|
1387
|
-
context: { jose, cookies, logger }
|
|
1388
|
-
} = ctx;
|
|
1389
|
-
try {
|
|
1390
|
-
const session = getCookie(request, cookies.sessionToken.name);
|
|
1391
|
-
const decoded = await jose.decodeJWT(session);
|
|
1392
|
-
logger?.log("AUTH_SESSION_VALID");
|
|
1393
|
-
const { exp, iat, jti, nbf, ...user } = decoded;
|
|
1394
|
-
const headers = new Headers(secureApiHeaders);
|
|
1395
|
-
return Response.json({ user, expires: toISOString(exp * 1e3) }, { headers });
|
|
1396
|
-
} catch (error) {
|
|
1397
|
-
logger?.log("AUTH_SESSION_INVALID", { structuredData: { error_type: getErrorName(error) } });
|
|
1398
|
-
const headers = new import_router4.HeadersBuilder(secureApiHeaders).setCookie(cookies.sessionToken.name, "", expiredCookieAttributes).toHeaders();
|
|
1399
|
-
return Response.json({ authenticated: false, message: "Unauthorized" }, { status: 401, headers });
|
|
1400
|
-
}
|
|
1401
|
-
});
|
|
1402
|
-
|
|
1403
|
-
// src/actions/signOut/signOut.ts
|
|
1404
|
-
var import_zod4 = require("zod");
|
|
1405
|
-
var import_router5 = require("@aura-stack/router");
|
|
1406
|
-
var config = (0, import_router5.createEndpointConfig)({
|
|
1407
|
-
schemas: {
|
|
1408
|
-
searchParams: import_zod4.z.object({
|
|
1409
|
-
token_type_hint: import_zod4.z.literal("session_token"),
|
|
1410
|
-
redirectTo: import_zod4.z.string().optional()
|
|
1411
|
-
})
|
|
1412
|
-
}
|
|
1413
|
-
});
|
|
1414
|
-
var signOutAction = (0, import_router5.createEndpoint)(
|
|
1415
|
-
"POST",
|
|
1416
|
-
"/signOut",
|
|
1417
|
-
async (ctx) => {
|
|
1418
|
-
const {
|
|
1419
|
-
request,
|
|
1420
|
-
headers,
|
|
1421
|
-
searchParams: { redirectTo },
|
|
1422
|
-
context
|
|
1423
|
-
} = ctx;
|
|
1424
|
-
const { jose, cookies, logger } = context;
|
|
1425
|
-
const session = headers.getCookie(cookies.sessionToken.name);
|
|
1426
|
-
const csrfToken = headers.getCookie(cookies.csrfToken.name);
|
|
1427
|
-
const header = headers.getHeader("X-CSRF-Token");
|
|
1428
|
-
logger?.log("SIGN_OUT_ATTEMPT", {
|
|
1429
|
-
structuredData: {
|
|
1430
|
-
has_session: Boolean(session),
|
|
1431
|
-
has_csrf_token: Boolean(csrfToken),
|
|
1432
|
-
has_csrf_header: Boolean(header)
|
|
1433
|
-
}
|
|
1434
|
-
});
|
|
1435
|
-
if (!session) {
|
|
1436
|
-
logger?.log("SESSION_TOKEN_MISSING");
|
|
1437
|
-
throw new AuthSecurityError("SESSION_TOKEN_MISSING", "The sessionToken is missing.");
|
|
1438
|
-
}
|
|
1439
|
-
if (!csrfToken) {
|
|
1440
|
-
logger?.log("CSRF_TOKEN_MISSING");
|
|
1441
|
-
throw new AuthSecurityError("CSRF_TOKEN_MISSING", "The CSRF token is missing.");
|
|
1442
|
-
}
|
|
1443
|
-
if (!header) {
|
|
1444
|
-
logger?.log("CSRF_HEADER_MISSING");
|
|
1445
|
-
throw new AuthSecurityError("CSRF_HEADER_MISSING", "The CSRF header is missing.");
|
|
1446
|
-
}
|
|
1447
|
-
try {
|
|
1448
|
-
await verifyCSRF(jose, csrfToken, header);
|
|
1449
|
-
} catch (error) {
|
|
1450
|
-
logger?.log("CSRF_TOKEN_INVALID", { structuredData: { error_type: getErrorName(error) } });
|
|
1451
|
-
throw new AuthSecurityError("CSRF_TOKEN_INVALID", "CSRF token verification failed");
|
|
1452
|
-
}
|
|
1453
|
-
logger?.log("SIGN_OUT_CSRF_VERIFIED");
|
|
1454
|
-
try {
|
|
1455
|
-
await jose.decodeJWT(session);
|
|
1456
|
-
logger?.log("SIGN_OUT_SUCCESS");
|
|
1457
|
-
} catch (error) {
|
|
1458
|
-
logger?.log("INVALID_JWT_TOKEN", { structuredData: { error_type: getErrorName(error) } });
|
|
1459
|
-
}
|
|
1460
|
-
const baseURL = getBaseURL(request);
|
|
1461
|
-
const location = await createRedirectTo(
|
|
1462
|
-
new Request(baseURL, {
|
|
1463
|
-
headers: headers.toHeaders()
|
|
1464
|
-
}),
|
|
1465
|
-
redirectTo,
|
|
1466
|
-
context
|
|
1467
|
-
);
|
|
1468
|
-
logger?.log("SIGN_OUT_REDIRECT", { structuredData: { location } });
|
|
1469
|
-
const headersList = new import_router5.HeadersBuilder(secureApiHeaders).setHeader("Location", location).setCookie(cookies.csrfToken.name, "", expiredCookieAttributes).setCookie(cookies.sessionToken.name, "", expiredCookieAttributes).toHeaders();
|
|
1470
|
-
return Response.json({ message: "Signed out successfully" }, { status: import_router5.statusCode.ACCEPTED, headers: headersList });
|
|
1471
|
-
},
|
|
1472
|
-
config
|
|
1473
|
-
);
|
|
1474
|
-
|
|
1475
|
-
// src/actions/csrfToken/csrfToken.ts
|
|
1476
|
-
var import_router6 = require("@aura-stack/router");
|
|
1477
|
-
var getCSRFToken = (request, cookieName) => {
|
|
1478
|
-
try {
|
|
1479
|
-
return getCookie(request, cookieName);
|
|
1480
|
-
} catch {
|
|
1481
|
-
return void 0;
|
|
1482
|
-
}
|
|
1483
|
-
};
|
|
1484
|
-
var csrfTokenAction = (0, import_router6.createEndpoint)("GET", "/csrfToken", async (ctx) => {
|
|
1485
|
-
const {
|
|
1486
|
-
request,
|
|
1487
|
-
context: { jose, cookies, logger }
|
|
1488
|
-
} = ctx;
|
|
1489
|
-
const token = getCSRFToken(request, cookies.csrfToken.name);
|
|
1490
|
-
logger?.log("CSRF_TOKEN_REQUESTED", { structuredData: { has_token: Boolean(token) } });
|
|
1491
|
-
const csrfToken = await createCSRF(jose, token);
|
|
1492
|
-
logger?.log("CSRF_TOKEN_ISSUED", { structuredData: { issued: Boolean(csrfToken) } });
|
|
1493
|
-
const headers = new Headers(secureApiHeaders);
|
|
1494
|
-
headers.append("Set-Cookie", setCookie(cookies.csrfToken.name, csrfToken, cookies.csrfToken.attributes));
|
|
1495
|
-
return Response.json({ csrfToken }, { headers });
|
|
1496
|
-
});
|
|
1497
|
-
|
|
1498
|
-
// src/logger.ts
|
|
1499
|
-
var logMessages = {
|
|
1500
|
-
ROUTER_INTERNAL_ERROR: {
|
|
1501
|
-
facility: 10,
|
|
1502
|
-
severity: "error",
|
|
1503
|
-
msgId: "ROUTER_INTERNAL_ERROR",
|
|
1504
|
-
message: "Unhandled router error while processing the request"
|
|
1505
|
-
},
|
|
1506
|
-
INVALID_REQUEST: {
|
|
1507
|
-
facility: 10,
|
|
1508
|
-
severity: "warning",
|
|
1509
|
-
msgId: "INVALID_REQUEST",
|
|
1510
|
-
message: "Request validation failed against the expected schema"
|
|
1511
|
-
},
|
|
1512
|
-
SERVER_ERROR: {
|
|
1513
|
-
facility: 10,
|
|
1514
|
-
severity: "error",
|
|
1515
|
-
msgId: "SERVER_ERROR",
|
|
1516
|
-
message: "Unexpected internal server error during authentication"
|
|
1517
|
-
},
|
|
1518
|
-
OAUTH_PROTOCOL_ERROR: {
|
|
1519
|
-
facility: 10,
|
|
1520
|
-
severity: "warning",
|
|
1521
|
-
msgId: "OAUTH_PROTOCOL_ERROR",
|
|
1522
|
-
message: "OAuth provider returned an invalid or unexpected protocol response"
|
|
1523
|
-
},
|
|
1524
|
-
OAUTH_AUTHORIZATION_ERROR: {
|
|
1525
|
-
facility: 10,
|
|
1526
|
-
severity: "error",
|
|
1527
|
-
msgId: "OAUTH_AUTHORIZATION_ERROR",
|
|
1528
|
-
message: "OAuth authorization request was rejected or failed"
|
|
1529
|
-
},
|
|
1530
|
-
INVALID_OAUTH_CONFIGURATION: {
|
|
1531
|
-
facility: 10,
|
|
1532
|
-
severity: "error",
|
|
1533
|
-
msgId: "INVALID_OAUTH_CONFIGURATION",
|
|
1534
|
-
message: "The OAuth provider configuration is invalid or incomplete"
|
|
1535
|
-
},
|
|
1536
|
-
OAUTH_ACCESS_TOKEN_REQUEST_INITIATED: {
|
|
1537
|
-
facility: 10,
|
|
1538
|
-
severity: "debug",
|
|
1539
|
-
msgId: "OAUTH_ACCESS_TOKEN_REQUEST_INITIATED",
|
|
1540
|
-
message: "Starting OAuth access token request to the provider"
|
|
1541
|
-
},
|
|
1542
|
-
INVALID_OAUTH_ACCESS_TOKEN_RESPONSE: {
|
|
1543
|
-
facility: 10,
|
|
1544
|
-
severity: "error",
|
|
1545
|
-
msgId: "INVALID_OAUTH_ACCESS_TOKEN_RESPONSE",
|
|
1546
|
-
message: "OAuth access token endpoint returned an invalid or malformed response"
|
|
1547
|
-
},
|
|
1548
|
-
OAUTH_ACCESS_TOKEN_ERROR: {
|
|
1549
|
-
facility: 10,
|
|
1550
|
-
severity: "error",
|
|
1551
|
-
msgId: "OAUTH_ACCESS_TOKEN_ERROR",
|
|
1552
|
-
message: "OAuth access token endpoint returned an error response"
|
|
1553
|
-
},
|
|
1554
|
-
OAUTH_ACCESS_TOKEN_SUCCESS: {
|
|
1555
|
-
facility: 10,
|
|
1556
|
-
severity: "info",
|
|
1557
|
-
msgId: "OAUTH_ACCESS_TOKEN_SUCCESS",
|
|
1558
|
-
message: "Successfully retrieved OAuth access token from the provider"
|
|
1559
|
-
},
|
|
1560
|
-
OAUTH_ACCESS_TOKEN_REQUEST_FAILED: {
|
|
1561
|
-
facility: 10,
|
|
1562
|
-
severity: "error",
|
|
1563
|
-
msgId: "OAUTH_ACCESS_TOKEN_REQUEST_FAILED",
|
|
1564
|
-
message: "Network or server error while requesting OAuth access token"
|
|
1565
|
-
},
|
|
1566
|
-
OAUTH_USERINFO_REQUEST_INITIATED: {
|
|
1567
|
-
facility: 10,
|
|
1568
|
-
severity: "debug",
|
|
1569
|
-
msgId: "OAUTH_USERINFO_REQUEST_INITIATED",
|
|
1570
|
-
message: "Starting OAuth userinfo request to the provider"
|
|
1571
|
-
},
|
|
1572
|
-
OAUTH_USERINFO_INVALID_RESPONSE: {
|
|
1573
|
-
facility: 10,
|
|
1574
|
-
severity: "error",
|
|
1575
|
-
msgId: "OAUTH_USERINFO_INVALID_RESPONSE",
|
|
1576
|
-
message: "OAuth userinfo endpoint returned an invalid or malformed response"
|
|
1577
|
-
},
|
|
1578
|
-
OAUTH_USERINFO_ERROR: {
|
|
1579
|
-
facility: 10,
|
|
1580
|
-
severity: "error",
|
|
1581
|
-
msgId: "OAUTH_USERINFO_ERROR",
|
|
1582
|
-
message: "OAuth userinfo endpoint returned an error response"
|
|
1583
|
-
},
|
|
1584
|
-
OAUTH_USERINFO_SUCCESS: {
|
|
1585
|
-
facility: 10,
|
|
1586
|
-
severity: "info",
|
|
1587
|
-
msgId: "OAUTH_USERINFO_SUCCESS",
|
|
1588
|
-
message: "Successfully retrieved user information from the OAuth provider"
|
|
1589
|
-
},
|
|
1590
|
-
OAUTH_USERINFO_REQUEST_FAILED: {
|
|
1591
|
-
facility: 10,
|
|
1592
|
-
severity: "error",
|
|
1593
|
-
msgId: "OAUTH_USERINFO_REQUEST_FAILED",
|
|
1594
|
-
message: "Network or server error while requesting user information from the OAuth provider"
|
|
1595
|
-
},
|
|
1596
|
-
OAUTH_CALLBACK_SUCCESS: {
|
|
1597
|
-
facility: 4,
|
|
1598
|
-
severity: "info",
|
|
1599
|
-
msgId: "OAUTH_CALLBACK_SUCCESS",
|
|
1600
|
-
message: "OAuth callback completed successfully and session was created"
|
|
1601
|
-
},
|
|
1602
|
-
MISMATCHING_STATE: {
|
|
1603
|
-
facility: 4,
|
|
1604
|
-
severity: "critical",
|
|
1605
|
-
msgId: "MISMATCHING_STATE",
|
|
1606
|
-
message: "OAuth response state parameter does not match the stored state value"
|
|
1607
|
-
},
|
|
1608
|
-
POTENTIAL_OPEN_REDIRECT_ATTACK_DETECTED: {
|
|
1609
|
-
facility: 4,
|
|
1610
|
-
severity: "critical",
|
|
1611
|
-
msgId: "POTENTIAL_OPEN_REDIRECT_ATTACK_DETECTED",
|
|
1612
|
-
message: "Blocked redirect to untrusted or external URL (potential open redirect attack)"
|
|
1613
|
-
},
|
|
1614
|
-
OPEN_REDIRECT_ATTACK: {
|
|
1615
|
-
facility: 4,
|
|
1616
|
-
severity: "warning",
|
|
1617
|
-
msgId: "OPEN_REDIRECT_ATTACK",
|
|
1618
|
-
message: "Detected redirect target that does not match the trusted origin"
|
|
1619
|
-
},
|
|
1620
|
-
SESSION_TOKEN_MISSING: {
|
|
1621
|
-
facility: 4,
|
|
1622
|
-
severity: "warning",
|
|
1623
|
-
msgId: "SESSION_TOKEN_MISSING",
|
|
1624
|
-
message: "Session cookie is missing from the request"
|
|
1625
|
-
},
|
|
1626
|
-
CSRF_TOKEN_MISSING: {
|
|
1627
|
-
facility: 4,
|
|
1628
|
-
severity: "warning",
|
|
1629
|
-
msgId: "CSRF_TOKEN_MISSING",
|
|
1630
|
-
message: "CSRF token cookie is missing from the request"
|
|
1631
|
-
},
|
|
1632
|
-
CSRF_HEADER_MISSING: {
|
|
1633
|
-
facility: 4,
|
|
1634
|
-
severity: "warning",
|
|
1635
|
-
msgId: "CSRF_HEADER_MISSING",
|
|
1636
|
-
message: "CSRF header is missing from the request"
|
|
1637
|
-
},
|
|
1638
|
-
CSRF_TOKEN_INVALID: {
|
|
1639
|
-
facility: 4,
|
|
1640
|
-
severity: "error",
|
|
1641
|
-
msgId: "CSRF_TOKEN_INVALID",
|
|
1642
|
-
message: "CSRF token verification failed or token is invalid"
|
|
1643
|
-
},
|
|
1644
|
-
SIGN_IN_INITIATED: {
|
|
1645
|
-
facility: 4,
|
|
1646
|
-
severity: "info",
|
|
1647
|
-
msgId: "SIGN_IN_INITIATED",
|
|
1648
|
-
message: "Starting OAuth sign-in flow for the selected provider"
|
|
1649
|
-
},
|
|
1650
|
-
SIGN_OUT_ATTEMPT: {
|
|
1651
|
-
facility: 4,
|
|
1652
|
-
severity: "debug",
|
|
1653
|
-
msgId: "SIGN_OUT_ATTEMPT",
|
|
1654
|
-
message: "Received sign-out request from client"
|
|
1655
|
-
},
|
|
1656
|
-
SIGN_OUT_CSRF_VERIFIED: {
|
|
1657
|
-
facility: 4,
|
|
1658
|
-
severity: "info",
|
|
1659
|
-
msgId: "SIGN_OUT_CSRF_VERIFIED",
|
|
1660
|
-
message: "CSRF token was successfully verified during sign-out"
|
|
1661
|
-
},
|
|
1662
|
-
SIGN_OUT_SUCCESS: {
|
|
1663
|
-
facility: 4,
|
|
1664
|
-
severity: "info",
|
|
1665
|
-
msgId: "SIGN_OUT_SUCCESS",
|
|
1666
|
-
message: "User session was cleared and sign-out completed successfully"
|
|
1667
|
-
},
|
|
1668
|
-
SIGN_OUT_REDIRECT: {
|
|
1669
|
-
facility: 4,
|
|
1670
|
-
severity: "debug",
|
|
1671
|
-
msgId: "SIGN_OUT_REDIRECT",
|
|
1672
|
-
message: "Redirecting client after successful sign-out"
|
|
1673
|
-
},
|
|
1674
|
-
AUTH_SESSION_VALID: {
|
|
1675
|
-
facility: 4,
|
|
1676
|
-
severity: "info",
|
|
1677
|
-
msgId: "AUTH_SESSION_VALID",
|
|
1678
|
-
message: "Session token is valid and user session was returned"
|
|
1679
|
-
},
|
|
1680
|
-
AUTH_SESSION_INVALID: {
|
|
1681
|
-
facility: 4,
|
|
1682
|
-
severity: "notice",
|
|
1683
|
-
msgId: "AUTH_SESSION_INVALID",
|
|
1684
|
-
message: "Session token is missing, expired, or invalid"
|
|
1685
|
-
},
|
|
1686
|
-
INVALID_JWT_TOKEN: {
|
|
1687
|
-
facility: 4,
|
|
1688
|
-
severity: "warning",
|
|
1689
|
-
msgId: "INVALID_JWT_TOKEN",
|
|
1690
|
-
message: "JWT session token failed validation during sign-out"
|
|
1691
|
-
},
|
|
1692
|
-
CSRF_TOKEN_REQUESTED: {
|
|
1693
|
-
facility: 4,
|
|
1694
|
-
severity: "debug",
|
|
1695
|
-
msgId: "CSRF_TOKEN_REQUESTED",
|
|
1696
|
-
message: "Client requested a CSRF token"
|
|
1697
|
-
},
|
|
1698
|
-
CSRF_TOKEN_ISSUED: {
|
|
1699
|
-
facility: 4,
|
|
1700
|
-
severity: "debug",
|
|
1701
|
-
msgId: "CSRF_TOKEN_ISSUED",
|
|
1702
|
-
message: "Issued a new CSRF token to the client"
|
|
1703
|
-
},
|
|
1704
|
-
INVALID_URL: {
|
|
1705
|
-
facility: 10,
|
|
1706
|
-
severity: "error",
|
|
1707
|
-
msgId: "INVALID_URL",
|
|
1708
|
-
message: "Derived origin URL is invalid or malformed"
|
|
1709
|
-
},
|
|
1710
|
-
COOKIE_HTTPONLY_DISABLED: {
|
|
1711
|
-
facility: 10,
|
|
1712
|
-
severity: "critical",
|
|
1713
|
-
msgId: "COOKIE_HTTPONLY_DISABLED",
|
|
1714
|
-
message: "Cookie is configured without HttpOnly. This allows JavaScript access via document.cookie and increases XSS exposure."
|
|
1715
|
-
},
|
|
1716
|
-
COOKIE_WILDCARD_DOMAIN: {
|
|
1717
|
-
facility: 10,
|
|
1718
|
-
severity: "critical",
|
|
1719
|
-
msgId: "COOKIE_WILDCARD_DOMAIN",
|
|
1720
|
-
message: "Cookie 'Domain' is set to a wildcard, which is insecure and should be avoided."
|
|
1721
|
-
},
|
|
1722
|
-
COOKIE_SECURE_DISABLED: {
|
|
1723
|
-
facility: 10,
|
|
1724
|
-
severity: "warning",
|
|
1725
|
-
msgId: "COOKIE_SECURE_DISABLED",
|
|
1726
|
-
message: "Cookie is configured with 'Secure' but the request is not HTTPS. The 'Secure' attribute will be ignored by the browser."
|
|
1727
|
-
},
|
|
1728
|
-
COOKIE_SAMESITE_NONE_WITHOUT_SECURE: {
|
|
1729
|
-
facility: 10,
|
|
1730
|
-
severity: "warning",
|
|
1731
|
-
msgId: "COOKIE_SAMESITE_NONE_WITHOUT_SECURE",
|
|
1732
|
-
message: "Cookie uses SameSite=None without Secure. Falling back to SameSite=Lax for safer defaults."
|
|
1733
|
-
},
|
|
1734
|
-
COOKIE_INSECURE_IN_PRODUCTION: {
|
|
1735
|
-
facility: 10,
|
|
1736
|
-
severity: "critical",
|
|
1737
|
-
msgId: "COOKIE_INSECURE_IN_PRODUCTION",
|
|
1738
|
-
message: "Cookies are being served over an insecure connection in production, which is a serious security risk."
|
|
1739
|
-
},
|
|
1740
|
-
COOKIE_HOST_STRATEGY_INSECURE: {
|
|
1741
|
-
facility: 10,
|
|
1742
|
-
severity: "critical",
|
|
1743
|
-
msgId: "COOKIE_HOST_STRATEGY_INSECURE",
|
|
1744
|
-
message: "__Host- cookies require a secure HTTPS context. Falling back to standard cookie settings."
|
|
1745
|
-
},
|
|
1746
|
-
UNTRUSTED_ORIGIN: {
|
|
1747
|
-
facility: 10,
|
|
1748
|
-
severity: "error",
|
|
1749
|
-
msgId: "UNTRUSTED_ORIGIN",
|
|
1750
|
-
message: "The constructed origin URL is not trusted."
|
|
1751
|
-
}
|
|
1752
|
-
};
|
|
1753
|
-
var createLogEntry = (key, overrides) => {
|
|
1754
|
-
const message = logMessages[key];
|
|
1755
|
-
return {
|
|
1756
|
-
...message,
|
|
1757
|
-
...overrides
|
|
1758
|
-
};
|
|
1759
|
-
};
|
|
1760
|
-
|
|
1761
|
-
// src/index.ts
|
|
1762
|
-
var logLevelToSeverity = {
|
|
1763
|
-
debug: ["debug", "info", "notice", "warning", "error", "critical", "alert", "emergency"],
|
|
1764
|
-
info: ["info", "notice", "warning", "error", "critical", "alert", "emergency"],
|
|
1765
|
-
warn: ["warning", "error", "critical", "alert", "emergency"],
|
|
1766
|
-
error: ["error", "critical", "alert", "emergency"]
|
|
1767
|
-
};
|
|
1768
|
-
var createLoggerProxy = (logger) => {
|
|
1769
|
-
if (!logger) return void 0;
|
|
1770
|
-
const level = logger.level;
|
|
1771
|
-
const allowedSeverities = logLevelToSeverity[level] ?? [];
|
|
1772
|
-
const internalLogger = {
|
|
1773
|
-
level,
|
|
1774
|
-
log(key, overrides) {
|
|
1775
|
-
const entry = createLogEntry(key, overrides);
|
|
1776
|
-
if (!allowedSeverities.includes(entry.severity)) return entry;
|
|
1777
|
-
logger.log({
|
|
1778
|
-
timestamp: entry.timestamp ?? (/* @__PURE__ */ new Date()).toISOString(),
|
|
1779
|
-
appName: entry.appName ?? "aura-auth",
|
|
1780
|
-
hostname: entry.hostname ?? "aura-auth",
|
|
1781
|
-
...entry
|
|
1782
|
-
});
|
|
1783
|
-
return entry;
|
|
1784
|
-
}
|
|
1785
|
-
};
|
|
1786
|
-
return internalLogger;
|
|
1787
|
-
};
|
|
1788
|
-
var createInternalConfig = (authConfig) => {
|
|
1789
|
-
const useSecure = authConfig?.trustedProxyHeaders ?? false;
|
|
1790
|
-
const logger = authConfig?.logger;
|
|
1791
|
-
const internalLogger = createLoggerProxy(logger);
|
|
1792
|
-
return {
|
|
1793
|
-
basePath: authConfig?.basePath ?? "/auth",
|
|
1794
|
-
onError: createErrorHandler(internalLogger),
|
|
1795
|
-
context: {
|
|
1796
|
-
oauth: createBuiltInOAuthProviders(authConfig?.oauth),
|
|
1797
|
-
cookies: createCookieStore(
|
|
1798
|
-
useSecure,
|
|
1799
|
-
authConfig?.cookies?.prefix,
|
|
1800
|
-
authConfig?.cookies?.overrides ?? {},
|
|
1801
|
-
internalLogger
|
|
1802
|
-
),
|
|
1803
|
-
jose: createJoseInstance(authConfig?.secret),
|
|
1804
|
-
secret: authConfig?.secret,
|
|
1805
|
-
basePath: authConfig?.basePath ?? "/auth",
|
|
1806
|
-
trustedProxyHeaders: useSecure,
|
|
1807
|
-
trustedOrigins: authConfig?.trustedOrigins,
|
|
1808
|
-
logger: internalLogger
|
|
1809
|
-
},
|
|
1810
|
-
middlewares: [
|
|
1811
|
-
(ctx) => {
|
|
1812
|
-
const useSecure2 = useSecureCookies(ctx.request, ctx.context.trustedProxyHeaders);
|
|
1813
|
-
const cookies = createCookieStore(
|
|
1814
|
-
useSecure2,
|
|
1815
|
-
authConfig?.cookies?.prefix,
|
|
1816
|
-
authConfig?.cookies?.overrides ?? {},
|
|
1817
|
-
internalLogger
|
|
1818
|
-
);
|
|
1819
|
-
ctx.context.cookies = cookies;
|
|
1820
|
-
return ctx;
|
|
1821
|
-
}
|
|
1822
|
-
]
|
|
1823
|
-
};
|
|
1824
|
-
};
|
|
1825
|
-
var createAuth = (authConfig) => {
|
|
1826
|
-
const config2 = createInternalConfig(authConfig);
|
|
1827
|
-
const router = (0, import_router7.createRouter)(
|
|
1828
|
-
[signInAction(config2.context.oauth), callbackAction(config2.context.oauth), sessionAction, signOutAction, csrfTokenAction],
|
|
1829
|
-
config2
|
|
1830
|
-
);
|
|
1831
|
-
return {
|
|
1832
|
-
handlers: router,
|
|
1833
|
-
jose: config2.context.jose
|
|
1834
|
-
};
|
|
1835
|
-
};
|
|
1836
|
-
// Annotate the CommonJS export names for ESM import in node:
|
|
1837
|
-
0 && (module.exports = {
|
|
1838
|
-
createAuth
|
|
1839
|
-
});
|
|
1
|
+
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`});const e=require(`./errors-CCYPHuBO.cjs`),t=require(`./oauth-BntNm6aE.cjs`),n=require(`./env-bq387KyP.cjs`),r=require(`./assert-NJGroSJd.cjs`),i=require(`./crypto-CoXA5w_4.cjs`),a=require(`./shared/identity.cjs`),o=require(`./logger-UnUhYL2V.cjs`);let s=require(`@aura-stack/router`),c=require(`zod/v4`),l=require(`@aura-stack/router/cookie`);const u={"Cache-Control":`no-store`,Pragma:`no-cache`,Expires:`0`,Vary:`Cookie`},d={"Content-Security-Policy":[`default-src 'none'`,`script-src 'self'`,`frame-src 'none'`,`object-src 'none'`,`frame-ancestors 'none'`,`base-uri 'none'`].join(`; `)},f={"X-Content-Type-Options":`nosniff`,"X-Frame-Options":`DENY`,"Referrer-Policy":`strict-origin-when-cross-origin`},p={...u,...d,...f},m=(e,t)=>{for(let[n,r]of Object.entries(t))r!==void 0&&r!==``&&e.searchParams.set(n,r)},h=(t,n,r,i,a)=>{let o=t.authorize,s=typeof o==`string`?o:o?.url??t.authorizeURL;if(!s)throw new e.n(`INVALID_OAUTH_CONFIGURATION`,`Missing authorization URL in OAuth provider configuration.`);let c=new URL(s),l=typeof o==`string`?void 0:o?.params;return m(c,{response_type:l?.responseType??t.responseType??`code`,client_id:t.clientId,redirect_uri:n,state:r,code_challenge:i,code_challenge_method:a,scope:l?.scope??t.scope,prompt:l?.prompt,response_mode:l?.responseMode,login_hint:l?.loginHint,nonce:l?.nonce,display:l?.display,audience:l?.audience}),c.toString()},g=async(n,r,a)=>{let o=i.i(),{codeVerifier:s,codeChallenge:c,method:l}=await i.r(),u=h(n,r,o,c,l);if(!t.a.safeParse({...n,redirectURI:r,state:o,codeChallenge:c,codeChallengeMethod:l}).success)throw a?.logger?.log(`INVALID_OAUTH_CONFIGURATION`,{structuredData:{scope:n?.scope??``,redirect_uri:r,has_state:!!o,has_code_challenge:!!c,code_challenge_method:l}}),new e.n(`INVALID_OAUTH_CONFIGURATION`,`The OAuth provider configuration is invalid.`);return{authorization:u,state:o,codeVerifier:s,method:l}},_=async(e,t)=>{if(!t)return[];let n=typeof t==`function`?await t(e):t;return Array.isArray(n)?n:typeof n==`string`?[n]:[]},v=async({ctx:t,request:r,headers:i})=>{let a=n.n(`BASE_URL`)||t?.baseURL;if(a&&a!==`/`)return a;if(t?.trustedProxyHeaders){let t=i&&new Headers(i)||r?.headers,n=t?.get(`Forwarded`)?.match(/proto=([^;]+)/i)?.[1]??t?.get(`X-Forwarded-Proto`)??`http`,a=t?.get(`Host`)??t?.get(`Forwarded`)?.match(/host=([^;]+)/i)?.[1]??t?.get(`X-Forwarded-Host`)??null;if(a)return`${n}://${a}`;throw new e.n(`INVALID_OAUTH_CONFIGURATION`,`The URL cannot be constructed. Please set the BASE_URL environment variable or provide trusted proxy host headers.`)}try{return new URL(r?.url??`not-found`).origin}catch(t){throw new e.n(`INVALID_OAUTH_CONFIGURATION`,`The URL cannot be constructed. Please set the BASE_URL environment variable or enable trustedProxyHeaders.`,{cause:t})}},y=async(t,n)=>{let i=await _(t,n?.trustedOrigins);i.push(new URL(t.url).origin);let a=await v({request:t,ctx:n});if(!r.s(a,i))throw n?.logger?.log(`UNTRUSTED_ORIGIN`,{structuredData:{origin:a}}),new e.n(`UNTRUSTED_ORIGIN`,`The constructed origin URL is not trusted.`);return a},b=async(e,t,n)=>`${await y(e,n)}${n.basePath}/callback/${t}`,x=async({request:e,oauth:t,ctx:n,redirectTo:r})=>{let i=await y(e,n),a=new URLSearchParams;return r!==void 0&&a.set(`redirectTo`,String(r)),`${i}${n.basePath}/signIn/${t}?${a.toString()}`},S=async(e,t,n)=>{try{let i=e.headers,a=await y(e,n),o=await _(e,n?.trustedOrigins);return(t=>{if(!r.r(t)&&!r.c(t))return`/`;if(r.r(t))return t;if(o.length>0){if(r.s(t,o)){let n=new URL(t).origin;for(let i of o){if(r.g(i)?.test(n))return r.i(t,e.url)?r.f(t):t;if(r.c(i)&&r.d(new URL(i).origin,n))return t}}return n?.logger?.log(`OPEN_REDIRECT_ATTACK`),`/`}return r.i(t,a)?r.f(t):(n?.logger?.log(`OPEN_REDIRECT_ATTACK`),`/`)})(t??i.get(`Referer`)??i.get(`Origin`)??`/`)}catch{return n?.logger?.log(`POTENTIAL_OPEN_REDIRECT_ATTACK_DETECTED`),`/`}},C=async(t,{ctx:n,request:r,headers:i,redirect:a,redirectTo:o})=>{try{let c=new Headers(i),l=n.oauth[t];if(!l)throw new e.n(`INVALID_OAUTH_CONFIGURATION`,`The OAuth provider "${t}" is not configured.`);let d=r;if(!d){let e=`${await v({ctx:n,headers:c})}${n.basePath}/signIn/${t}`;d=new Request(e,{headers:c})}if(a===!1){n?.logger?.log(`SIGN_IN_INITIATED`,{structuredData:{oauth_provider:t}});let e=await x({request:d,oauth:t,ctx:n,redirectTo:o}),r=new Headers(p);return{success:!0,redirect:!1,signInURL:e,headers:r,toResponse:()=>Response.json({success:!0,redirect:!1,signInURL:e},{status:200,headers:r})}}let f=await b(d,t,n),m=await S(d,o,n),{authorization:h,state:_,codeVerifier:y}=await g(l,f,n);n?.logger?.log(`SIGN_IN_INITIATED`,{structuredData:{oauth_provider:t}});let C=new s.HeadersBuilder(u).setHeader(`Location`,h).setCookie(n.cookies.state.name,_,n.cookies.state.attributes).setCookie(n.cookies.redirectURI.name,f,n.cookies.redirectURI.attributes).setCookie(n.cookies.redirectTo.name,m,n.cookies.redirectTo.attributes).setCookie(n.cookies.codeVerifier.name,y,n.cookies.codeVerifier.attributes).toHeaders();return{success:!0,redirect:!0,signInURL:h,headers:C,toResponse:()=>Response.json({success:!0,redirect:!0,signInURL:h},{status:302,headers:C})}}catch(t){let n=`AUTH_SIGN_IN_FAILED`,r=`An error occurred during the sign-in process.`;return e.s(t)&&(n=t.code,r=t.message),{success:!1,redirect:!1,signInURL:null,error:{code:n,message:r},headers:new Headers(p),toResponse:()=>Response.json({success:!1,redirect:!1,signInURL:null,error:{code:n,message:r}},{status:500,headers:p})}}},w=async({ctx:t,payload:n,request:r,headers:a,redirectTo:o})=>{let{cookies:c,credentials:l,sessionStrategy:u,logger:d}=t;try{let f=r;if(!f){let e=`${await v({ctx:t,headers:a})}${t.basePath}/signIn/credentials`;f=new Request(e,{headers:a})}await y(f,t);let m=await l?.authorize({credentials:n,deriveSecret:l?.hash??i.a,verifySecret:l?.verify??i.s});if(!m)throw new e.a(`INVALID_CREDENTIALS`,`The provided credentials are invalid.`);let h=await u.createSession(m),g=await i.t(t.jose);d?.log(`CREDENTIALS_SIGN_IN_SUCCESS`);let _=await S(f,o,t),b=new s.HeadersBuilder(p).setHeader(`Location`,_).setCookie(c.csrfToken.name,g,c.csrfToken.attributes).setCookie(c.sessionToken.name,h,c.sessionToken.attributes).toHeaders();return{success:!0,headers:b,redirectURL:_,toResponse:()=>Response.json({success:!0,redirectURL:_},{headers:b})}}catch(t){let n=`CREDENTIALS_SIGN_IN_ERROR`,r=`An error occurred during credentials sign-in.`;e.s(t)&&(n=t.code,r=t.message);let i=new Headers(p),a={success:!1,headers:i,redirectURL:null,error:{code:n,message:r},toResponse:()=>Response.json({success:!1,redirectURL:null,error:{code:n,message:r}},{headers:i,status:401})};return t instanceof e.a?(d?.log(`INVALID_CREDENTIALS`,{severity:`warning`,structuredData:{path:`/signIn/credentials`}}),a):(d?.log(`CREDENTIALS_SIGN_IN_FAILED`,{severity:`error`,structuredData:{path:`/signIn/credentials`}}),a)}},T=async({ctx:e,request:t,headers:n,redirect:r,redirectTo:i,skipCSRFCheck:a=!1})=>{let o=await e.sessionStrategy.destroySession(new Headers(n),a),c=t;if(!c){let t=`${await v({ctx:e,headers:o})}${e.basePath}/signOut`;c=new Request(t,{headers:o})}let l=new s.HeadersBuilder(o),u=await S(c,i,e);u&&l.setHeader(`Location`,u);let d=l.toHeaders();return{headers:d,redirect:r??!1,redirectURL:u,success:!0,toResponse:()=>Response.json({success:!0,redirect:r??!1,redirectURL:u},{headers:d,status:i?302:202})}},E={httpOnly:!0,sameSite:`lax`,path:`/`,maxAge:3600*24*15},D={secure:!1,httpOnly:!0},O={secure:!0,httpOnly:!0},k={secure:!0,httpOnly:!0,path:`/`,domain:void 0},A={httpOnly:!0,maxAge:300,sameSite:`lax`,expires:new Date(Date.now()+300*1e3)},j=(e,t,n)=>(0,l.serialize)(e,t,n),M={...E,expires:new Date(0),maxAge:0,secure:!0},N=(t,n)=>{let r=t instanceof Request?t.headers.get(`Cookie`):t.get(`Cookie`);if(!r)throw new e.n(`COOKIE_NOT_FOUND`,`No cookies found. There is no active session`);let i=(0,l.parse)(r)[n];if(!i)throw new e.n(`COOKIE_NOT_FOUND`,`Cookie "${n}" not found. There is no active session`);return i},P=(e,t,r,i)=>(t.httpOnly||i?.log(`COOKIE_HTTPONLY_DISABLED`),t.domain===`*`&&(t.domain=void 0,i?.log(`COOKIE_WILDCARD_DOMAIN`)),e?r===`host`?{...E,...t,...k}:{...E,...t,...O}:(t.secure&&i?.log(`COOKIE_SECURE_DISABLED`),t.sameSite==`none`&&(t.sameSite=`lax`,i?.log(`COOKIE_SAMESITE_NONE_WITHOUT_SECURE`)),n.t.NODE_ENV===`production`&&i?.log(`COOKIE_INSECURE_IN_PRODUCTION`),r===`host`&&i?.log(`COOKIE_HOST_STRATEGY_INSECURE`),{...E,...t,...D})),F=(e,t,n,r)=>{t??=`aura-auth`;let i=e?`__Secure-`:``,a=e?`__Host-`:``;return{sessionToken:{name:`${i}${t}.${n?.sessionToken?.name??`session_token`}`,attributes:P(e,{...E,...n?.sessionToken?.attributes},n?.sessionToken?.attributes?.strategy??`secure`,r)},state:{name:`${i}${t}.${n?.state?.name??`state`}`,attributes:P(e,{...A,...n?.state?.attributes},n?.state?.attributes?.strategy??`secure`,r)},csrfToken:{name:`${a}${t}.${n?.csrfToken?.name??`csrf_token`}`,attributes:P(e,{...n?.csrfToken?.attributes,...k,sameSite:`strict`},n?.csrfToken?.attributes?.strategy??`host`,r)},redirectTo:{name:`${i}${t}.${n?.redirectTo?.name??`redirect_to`}`,attributes:P(e,{...A,...n?.redirectTo?.attributes},n?.redirectTo?.attributes?.strategy??`secure`,r)},redirectURI:{name:`${i}${t}.${n?.redirectURI?.name??`redirect_uri`}`,attributes:P(e,{...A,...n?.redirectURI?.attributes},n?.redirectURI?.attributes?.strategy??`secure`,r)},codeVerifier:{name:`${i}${t}.${n?.codeVerifier?.name??`code_verifier`}`,attributes:P(e,{...A,...n?.codeVerifier?.attributes},n?.codeVerifier?.attributes?.strategy??`secure`,r)}}},I=async({ctx:e,headers:t})=>{let n=new s.HeadersBuilder(p).setCookie(e.cookies.sessionToken.name,``,{...e.cookies.sessionToken.attributes,...M}).setCookie(e.cookies.csrfToken.name,``,{...e.cookies.csrfToken.attributes,...M}).toHeaders(),i={code:`GET_SESSION_FAILED`,message:`Failed to retrieve session. The session token may be missing, expired, or invalid.`},a={session:null,headers:n,success:!1,error:i,toResponse:()=>Response.json({success:!1,session:null},{status:401,headers:n})};try{let{session:n,headers:o}=await e.sessionStrategy.getSession(new Headers(t));if(!n)return a;let s=r.v(o,p);return{session:n,headers:s,success:!0,toResponse:()=>Response.json({success:!0,session:n,error:i},{headers:s})}}catch(t){return e?.logger?.log(`AUTH_SESSION_INVALID`,{structuredData:{error_type:r.m(t)}}),a}},L=async({ctx:t,request:n,headers:i,session:a,redirectTo:o,skipCSRFCheck:s=!1})=>{try{let{session:e,headers:c}=await t.sessionStrategy.refreshSession(new Headers(i),a,s),l=r.v(c,p),u=null;if(o){let e=n;if(!e){let n=`${await v({ctx:t,headers:c})}${t.basePath}/updateSession`;e=new Request(n,{headers:l})}u=await S(e,o,t)}return{headers:l,session:e,success:!!e,redirectURL:u,toResponse:()=>Response.json({success:!!e,session:e,redirectURL:u},{headers:l,status:e?200:401})}}catch(t){let n=`UPDATE_SESSION_INVALID`,r=`Failed to update session.`;e.s(t)&&(n=t.code,r=t.message);let i=new Headers(p);return{success:!1,headers:i,session:null,redirectURL:null,error:{code:n,message:r},toResponse:()=>Response.json({success:!1,session:null,redirectURL:null,error:{code:n,message:r}},{status:400,headers:i})}}},R=e=>({getSession:async t=>await I({ctx:e,headers:t.headers}),signIn:async(t,n)=>C(t,{ctx:e,...n}),signInCredentials:async t=>w({ctx:e,...t}),updateSession:async t=>L({ctx:e,...t,skipCSRFCheck:!0}),signOut:async t=>T({ctx:e,...t,skipCSRFCheck:!0})}),ee=(t,n)=>{let r=t?.mode??`sealed`;if(![`sealed`,`signed`,`encrypted`].includes(r))throw new e.r(`[auth] invalid JWT mode "${r}". Valid options are: "sealed", "signed", "encrypted".`);return{createToken:r===`sealed`?n.encodeJWT:r===`signed`?n.signJWS:n.encryptJWE,verifyToken:r===`sealed`?n.decodeJWT:r===`signed`?n.verifyJWS:n.decryptJWE}},z=e=>({getCookie:t=>({sessionToken:N(t,e().sessionToken.name)}),setCookie:({sessionToken:t})=>new s.HeadersBuilder(p).setCookie(e().sessionToken.name,t,e().sessionToken.attributes).toHeaders(),clear:()=>new s.HeadersBuilder(p).setCookie(e().csrfToken.name,``,{...M,...e().csrfToken.attributes}).setCookie(e().sessionToken.name,``,{...M,...e().sessionToken.attributes}).toHeaders()}),B=(e,t)=>{switch(t){case`strip`:return e.strip();case`passthrough`:return e.loose();case`strict`:return e.strict()}},V=t=>{let n=B(t.schema??a.UserIdentity,t.unknownKeys??`strip`),i=n.partial();return{parse:async(t={})=>{let i=await n.safeParseAsync(t);if(!i.success)throw new e.a(`INVALID_IDENTITY_VALIDATION_FAILED`,JSON.stringify(r.p(i.error),null,2),{cause:i.error});return i.data},parseAsPartial:async(t={})=>{let n=await i.safeParseAsync(t);if(!n.success)throw new e.a(`INVALID_IDENTITY_VALIDATION_FAILED`,JSON.stringify(r.p(n.error),null,2),{cause:n.error});return n.data}}},H=({config:t,jose:n,logger:a,cookies:o,identity:s})=>{let c=ee(t?.jwt,n),l=z(o),u=t?.jwt?.maxAge??3600*24*15,d=t?.jwt?.expirationStrategy??`absolute`,f=V(s),p=({exp:e})=>{if(!e)return null;let t=Math.floor(Date.now()/1e3);switch(d){case`fixed`:case`absolute`:return null;case`rolling`:return new Date((t+u)*1e3);case`sliding`:{let n=u*.25;return e-t<n?new Date((t+u)*1e3):null}default:return null}},m=async(t,s=!1)=>{let c=null,l=null,u=t.get(`X-CSRF-Token`);try{c=N(t,o().sessionToken.name)}catch{throw new e.i(`SESSION_TOKEN_MISSING`,`The sessionToken is missing.`)}try{l=N(t,o().csrfToken.name)}catch{throw new e.i(`CSRF_TOKEN_MISSING`,`The CSRF token is missing.`)}if(a?.log(`CSRF_TOKEN_REQUESTED`,{structuredData:{has_session:!!c,has_csrf_token:!!l,has_csrf_header:!!u,skip_csrf_check:s}}),!c)throw a?.log(`SESSION_TOKEN_MISSING`),new e.i(`SESSION_TOKEN_MISSING`,`The sessionToken is missing.`);if(s)try{await n.verifyJWS(l)}catch(t){throw a?.log(`CSRF_TOKEN_INVALID`,{structuredData:{error_type:r.m(t)}}),new e.i(`CSRF_TOKEN_INVALID`,`CSRF token verification failed`)}else{if(!l)throw a?.log(`CSRF_TOKEN_MISSING`),new e.i(`CSRF_TOKEN_MISSING`,`The CSRF token is missing.`);if(!u)throw a?.log(`CSRF_HEADER_MISSING`),new e.i(`CSRF_HEADER_MISSING`,`The CSRF header is missing.`);try{await i.o(n,l,u)}catch(t){throw a?.log(`CSRF_TOKEN_INVALID`,{structuredData:{error_type:r.m(t)}}),new e.i(`CSRF_TOKEN_INVALID`,`CSRF token verification failed`)}a?.log(`CSRF_TOKEN_VERIFIED`)}try{return await n.decodeJWT(c),!0}catch(e){return a?.log(`INVALID_JWT_TOKEN`,{structuredData:{error_type:r.m(e)}}),!1}};return{getSession:async e=>{let t=new Headers;try{let{sessionToken:n}=l.getCookie(e);if(!n)return{session:null,headers:t};let{exp:r,iat:i,jti:o,nbf:u,aud:m,iss:h,mexp:g,..._}=await c.verifyToken(n);if(!_.sub)return{session:null,headers:t};let v={user:_,expires:r?new Date(r*1e3).toISOString():``},y=p({exp:r});if(!y){let t=s.skipValidation?v.user:await f.parse(v.user);return{session:{expires:v.expires,user:t},headers:e}}let b=s.skipValidation?v.user:await f.parse(v.user),x={user:b,expires:y.toISOString()},S=d===`absolute`?i:Math.floor(Date.now()/1e3),C=await c.createToken({...b,exp:Math.floor(y.getTime()/1e3),iat:S,mexp:g});return a?.log(`SESSION_REFRESHED`,{structuredData:{strategy:`stateless`,expiresAt:y.toISOString()}}),{session:x,headers:l.setCookie({sessionToken:C})}}catch(e){return a?.log(`AUTH_SESSION_INVALID`,{structuredData:{error_type:r.m(e)}}),{session:null,headers:t}}},createSession:async e=>{s.skipValidation&&a?.log(`IDENTITY_VALIDATION_DISABLED`,{structuredData:{identity_validation_disabled:!0}});let t=s.skipValidation?e:await f.parse(e);return c.createToken(t)},refreshSession:async(e,t,n=!1)=>{try{let{sessionToken:r}=l.getCookie(e);if(!r||!await m(e,n))return{session:null,headers:l.clear()};let i=await c.verifyToken(r),{exp:a,mexp:o,sub:h,iat:g}=i,_=s.skipValidation?i:await f.parse(i),v=s.skipValidation?t.user:await f.parseAsPartial(t.user),y=t.expires?new Date(t.expires):p({exp:a})??new Date(Date.now()+u*1e3),b={user:{..._,...v,sub:h},expires:y.toISOString()},x=d===`absolute`?g:Math.floor(Date.now()/1e3),S=await c.createToken({...b.user,exp:Math.floor(y.getTime()/1e3),iat:x,mexp:o});return b.expires=new Date(b.expires).toISOString(),{session:b,headers:l.setCookie({sessionToken:S})}}catch(e){return a?.log(`AUTH_SESSION_INVALID`,{structuredData:{error_type:r.m(e)}}),{session:null,headers:l.clear()}}},revokeSession:async e=>{},destroySession:async(e,t=!1)=>(await m(e,t),l.clear())}},U=({config:t,jose:n,cookies:r,logger:i,identity:a})=>{let o=t?.strategy??`jwt`;switch(o){case`jwt`:return H({jose:n,config:t,cookies:r,logger:i,identity:a});default:throw new e.r(`[auth] unknown session strategy "${o}". Valid options are: "jwt".`)}},W=e=>{let r=n.n(`TRUSTED_PROXY_HEADERS`)===void 0?e?.trustedProxyHeaders??!1:n.i(`TRUSTED_PROXY_HEADERS`),s=o.t(e),c=e?.cookies?.prefix,l=e?.cookies?.overrides??{},u=F(!0,c,l,s),d=F(!1,c,l,s),f=i.c(e?.secret,e?.session),p={oauth:t.n(e?.oauth),credentials:e?.credentials,cookies:d,jose:f,secret:e?.secret,basePath:e?.basePath??`/auth`,trustedProxyHeaders:r,trustedOrigins:n.r(`TRUSTED_ORIGINS`).length>0?n.r(`TRUSTED_ORIGINS`):e?.trustedOrigins,logger:s,cookieConfig:{secure:u,standard:d},baseURL:e?.baseURL,identity:{schema:e?.identity?.schema??a.UserIdentity,unknownKeys:e?.identity?.unknownKeys??`strip`,skipValidation:e?.identity?.skipValidation??!1}};return p.sessionStrategy=U({cookies:()=>p.cookies,jose:p.jose,config:e?.session,logger:p.logger,identity:p.identity}),p},G=t=>n=>{if((0,s.isRouterError)(n)){let{message:e,status:r,statusText:i}=n;return t?.log(`ROUTER_INTERNAL_ERROR`),Response.json({type:`ROUTER_ERROR`,code:`ROUTER_INTERNAL_ERROR`,message:e},{status:r,statusText:i})}if((0,s.isInvalidZodSchemaError)(n))return t?.log(`INVALID_REQUEST`),Response.json({type:`ROUTER_ERROR`,code:`INVALID_REQUEST`,message:n.errors},{status:422});if(e.f(n)){let{error:e,message:r,type:i,errorURI:a}=n;return t?.log(`OAUTH_PROTOCOL_ERROR`,{structuredData:{error:e,error_description:r,error_uri:a??``}}),Response.json({type:i,message:r},{status:400})}if(e.c(n)){let{type:e,code:r,message:i}=n;return t?.log(`INVALID_OAUTH_CONFIGURATION`,{structuredData:{error:r,error_description:i}}),Response.json({type:e,message:i},{status:400})}if(e.l(n)){let{type:e,code:r,message:i}=n;return t?.log(`AUTH_SECURITY_ERROR`,{structuredData:{error:r,error_description:i}}),Response.json({type:e,code:r,message:i},{status:400})}if(e.u(n)){let{type:e,code:r,message:i}=n;return t?.log(`IDENTITY_VALIDATION_FAILED`,{structuredData:{error:r,error_description:i}}),Response.json({type:e,code:r,message:i},{status:422})}return t?.log(`SERVER_ERROR`,{structuredData:{error_type:n.name,error_message:n.message}}),Response.json({type:`SERVER_ERROR`,code:`SERVER_ERROR`,message:`An unexpected error occurred`},{status:500})},K=e=>(0,s.createEndpointConfig)(`/signIn/:oauth`,{schemas:{params:c.z.object({oauth:c.z.enum(Object.keys(e),`The OAuth provider is not supported or invalid.`)}),searchParams:c.z.object({redirect:c.z.stringbool().optional().default(!0),redirectTo:c.z.string().optional()})}}),q=e=>(0,s.createEndpoint)(`GET`,`/signIn/:oauth`,async e=>{let{toResponse:t}=await C(e.params.oauth,{ctx:e.context,request:e.request,headers:e.request.headers,redirect:e.searchParams.redirect,redirectTo:e.searchParams.redirectTo});return t()},K(e)),J=(0,s.createEndpointConfig)({schemas:{body:c.z.object({username:c.z.string(),password:c.z.string()}),searchParams:c.z.object({redirectTo:c.z.string().optional()})}}),Y=(0,s.createEndpoint)(`POST`,`/signIn/credentials`,async e=>{let t=e.body,{toResponse:n}=await w({ctx:e.context,payload:t,request:e.request,headers:e.request.headers,redirectTo:e.searchParams.redirectTo});return n()},J),X=async(e,t={},n=5e3)=>{let r=new AbortController,i=setTimeout(()=>r.abort(),n);return await fetch(e,{...t,signal:r.signal}).finally(()=>clearTimeout(i))},Z=e=>{let t=i.i(16);return{sub:e?.id??e?.sub??t,email:e?.email,name:e?.name??e?.username??e?.nickname,image:e?.image??e?.picture}},Q=async(n,i,a)=>{let o=n.userInfo,s=typeof o==`string`?o:o.url,c=typeof o==`string`?void 0:o.headers,l=typeof o==`string`?`GET`:(o.method??`GET`).toUpperCase();try{a?.log(`OAUTH_USERINFO_REQUEST_INITIATED`,{structuredData:{endpoint:s}});let o=await X(s,{method:l,headers:{"User-Agent":`Aura Auth/${r.l}`,Accept:`application/json`,Authorization:`Bearer ${i}`,...c}});if(!o.ok)throw a?.log(`OAUTH_USERINFO_INVALID_RESPONSE`),new e.o(`INVALID_REQUEST`,`Invalid userinfo response format`);let u=await o.json(),{success:d,data:f}=t.s.safeParse(u);if(d)throw a?.log(`OAUTH_USERINFO_ERROR`,{message:`Error response received from OAuth userinfo endpoint`,structuredData:{error:f.error,error_description:f.error_description??``}}),new e.o(`INVALID_REQUEST`,`An error was received from the OAuth userinfo endpoint.`);return a?.log(`OAUTH_USERINFO_SUCCESS`),n?.profile?n.profile(u):Z(u)}catch(t){throw e.f(t)?t:(a?.log(`OAUTH_USERINFO_REQUEST_FAILED`),e.d(t)?new e.o(`SERVER_ERROR`,`Failed to fetch user information from OAuth provider`,``,{cause:t}):new e.o(`SERVER_ERROR`,`Failed to fetch user information`,``,{cause:t}))}},te=async(n,r,i,a,o)=>{let{accessToken:s,clientId:c,clientSecret:l}=n;if(!c||!l||!r||!i||!a||!s)throw o?.log(`INVALID_OAUTH_CONFIGURATION`,{structuredData:{has_client_id:!!c,has_client_secret:!!l,has_access_token:!!s,has_redirect_uri:!!r,has_code:!!i,has_code_verifier:!!a}}),new e.n(`INVALID_OAUTH_CONFIGURATION`,`The OAuth provider configuration is invalid.`);let u=typeof s==`string`?s:s.url,d=typeof s==`string`?void 0:s.headers;try{o?.log(`OAUTH_ACCESS_TOKEN_REQUEST_INITIATED`,{structuredData:{has_client_id:!!c,redirect_uri:r,grant_type:`authorization_code`}});let n=await X(u,{method:`POST`,headers:{...d,Accept:`application/json`,"Content-Type":`application/x-www-form-urlencoded`},body:new URLSearchParams({client_id:c,client_secret:l,code:i,redirect_uri:r,grant_type:`authorization_code`,code_verifier:a}).toString()});if(!n.ok)throw o?.log(`INVALID_OAUTH_ACCESS_TOKEN_RESPONSE`),new e.o(`invalid_request`,`Invalid access token response`);let s=await n.json(),f=t.i.safeParse(s);if(!f.success){let{success:n,data:r}=t.r.safeParse(s);throw n?(o?.log(`OAUTH_ACCESS_TOKEN_ERROR`,{structuredData:{error:r.error,error_description:r.error_description??``}}),new e.o(`INVALID_ACCESS_TOKEN`,`Failed to retrieve access token`)):(o?.log(`INVALID_OAUTH_ACCESS_TOKEN_RESPONSE`),new e.o(`invalid_request`,`Invalid access token response format`))}return o?.log(`OAUTH_ACCESS_TOKEN_SUCCESS`),f.data}catch(t){throw o?.log(`OAUTH_ACCESS_TOKEN_REQUEST_FAILED`),t instanceof Error?new e.o(`server_error`,`Failed to communicate with OAuth provider`,``,{cause:t}):t}},ne=n=>(0,s.createEndpointConfig)(`/callback/:oauth`,{schemas:{params:c.z.object({oauth:c.z.enum(Object.keys(n),`The OAuth provider is not supported or invalid.`)}),searchParams:c.z.object({code:c.z.string(`Missing code parameter in the OAuth authorization response.`),state:c.z.string(`Missing state parameter in the OAuth authorization response.`)})},use:[n=>{let{searchParams:r,context:{logger:i}}=n,a=t.o.safeParse(r);if(a.success){let{error:t,error_description:n}=a.data,r=[`access_denied`,`server_error`].includes(t.toLowerCase())?`critical`:`warning`;throw i?.log(`OAUTH_AUTHORIZATION_ERROR`,{severity:r,structuredData:{error:t,error_description:n??``}}),new e.o(t,n||`OAuth Authorization Error`)}return n}]}),re=t=>(0,s.createEndpoint)(`GET`,`/callback/:oauth`,async t=>{let{request:n,params:{oauth:a},searchParams:{code:o,state:c},context:l}=t,{oauth:d,cookies:f,jose:p,logger:m,trustedOrigins:h}=l,g=d[a],v=N(n,f.state.name),b=N(n,f.codeVerifier.name),x=N(n,f.redirectTo.name),S=N(n,f.redirectURI.name);if(!r._(v,c))throw m?.log(`MISMATCHING_STATE`,{structuredData:{oauth_provider:a}}),new e.i(`MISMATCHING_STATE`,`The provided state passed in the OAuth response does not match the stored state.`);let C=await te(g,S,o,b,m),w=await _(n,h),T=await y(n,l);if(!r.r(x)&&!(w.length>0?r.s(x,w):r.i(x,T)))throw m?.log(`POTENTIAL_OPEN_REDIRECT_ATTACK_DETECTED`,{structuredData:{redirect_path:x,provider:a,has_trusted_origins:w.length>0,request_origin:T}}),new e.i(`POTENTIAL_OPEN_REDIRECT_ATTACK_DETECTED`,`Invalid redirect path. Potential open redirect attack detected.`);let E=await Q(g,C.access_token,m),D=await l.sessionStrategy.createSession(E),O=await i.t(p);m?.log(`OAUTH_CALLBACK_SUCCESS`,{structuredData:{provider:a}});let k=new s.HeadersBuilder(u).setHeader(`Location`,x).setCookie(f.sessionToken.name,D,f.sessionToken.attributes).setCookie(f.csrfToken.name,O,f.csrfToken.attributes).setCookie(f.state.name,``,M).setCookie(f.redirectURI.name,``,M).setCookie(f.redirectTo.name,``,M).setCookie(f.codeVerifier.name,``,M).toHeaders();return Response.json({oauth:a},{status:302,headers:k})},ne(t)),ie=(0,s.createEndpoint)(`GET`,`/session`,async e=>{let{toResponse:t}=await I({ctx:e.context,headers:e.request.headers});return t()}),ae=(0,s.createEndpointConfig)({schemas:{searchParams:c.z.object({token_type_hint:c.z.literal(`session_token`),redirectTo:c.z.string().optional()})}}),oe=(0,s.createEndpoint)(`POST`,`/signOut`,async e=>{let{toResponse:t}=await T({ctx:e.context,request:e.request,headers:e.request.headers,redirectTo:e.searchParams.redirectTo});return t()},ae),se=(e,t)=>{try{return N(e,t)}catch{return}},$=(0,s.createEndpoint)(`GET`,`/csrfToken`,async e=>{let{request:t,context:{jose:n,cookies:r,logger:a}}=e,o=se(t,r.csrfToken.name);a?.log(`CSRF_TOKEN_REQUESTED`,{structuredData:{has_token:!!o}});let s=await i.t(n,o);a?.log(`CSRF_TOKEN_ISSUED`,{structuredData:{issued:!!s}});let c=new Headers(p);return c.append(`Set-Cookie`,j(r.csrfToken.name,s,r.csrfToken.attributes)),Response.json({csrfToken:s},{headers:c})}),ce=e=>(0,s.createEndpointConfig)({schemas:{body:c.z.object({user:e.schema?.partial().optional(),expires:c.z.coerce.date().optional()})}}),le=e=>(0,s.createEndpoint)(`PATCH`,`/session`,async e=>{let{toResponse:t}=await L({ctx:e.context,headers:e.request.headers,session:{user:e.body.user,expires:e.body.expires?.toISOString()}});return t()},ce(e)),ue=e=>{let t=W(e);return{basePath:e?.basePath??`/auth`,onError:G(t.logger),context:t,use:[e=>{let n=r.h(e.request,e.context.trustedProxyHeaders);return e.context.cookies=n?t.cookieConfig.secure:t.cookieConfig.standard,e}]}},de=e=>{let t=ue(e);return{handlers:(0,s.createRouter)([q(t.context.oauth),Y,re(t.context.oauth),ie,oe,$,le(t.context.identity)],t),jose:t.context.jose,api:R(t.context)}},fe=e=>{let t=de(e);return t.handlers.ALL=async e=>{let n=e.method.toUpperCase(),r={GET:t.handlers.GET,POST:t.handlers.POST,PATCH:t.handlers.PATCH};return n in r?await r[n](e):new Response(`Method Not Allowed`,{status:405,headers:{Allow:Object.keys(r).join(`, `)}})},t};exports.createAuth=fe;
|