@zpg6-test-pkgs/better-auth
Version:
The most comprehensive authentication library for TypeScript.
494 lines (488 loc) • 46.1 kB
JavaScript
import { ZodObject, ZodType, ZodOptional, z, ZodString, ZodNumber, ZodBoolean, ZodArray } from 'zod/v4';
import { getEndpoints } from '../../api/index.mjs';
import '../../shared/better-auth.n2KFGwjY.mjs';
import '../../shared/better-auth.CMQ3rA-I.mjs';
import '../../shared/better-auth.BjBlybv-.mjs';
import '@better-auth/utils/random';
import { APIError } from 'better-call';
import '@better-auth/utils/hash';
import '@noble/ciphers/chacha';
import '@noble/ciphers/utils';
import '@noble/ciphers/webcrypto';
import '@better-auth/utils/base64';
import 'jose';
import '@noble/hashes/scrypt';
import '@better-auth/utils';
import '@better-auth/utils/hex';
import '@noble/hashes/utils';
import '../../shared/better-auth.B4Qoxdgc.mjs';
import { g as getAuthTables } from '../../shared/better-auth.DORkW_Ge.mjs';
import 'kysely';
import '../../shared/better-auth.D7aTFyWE.mjs';
import { a as createAuthEndpoint } from '../../shared/better-auth.BfeJWAMn.mjs';
import '@better-auth/utils/hmac';
import '@better-auth/utils/binary';
import '../../shared/better-auth.DF-MUmVw.mjs';
import '../../shared/better-auth.DdzSJf-n.mjs';
import '../../shared/better-auth.CW6D9eSx.mjs';
import '../../shared/better-auth.BZZKN1g7.mjs';
import '../../shared/better-auth.CuS_eDdK.mjs';
import '../../shared/better-auth.O2VtDkDK.mjs';
import '../../crypto/index.mjs';
import '@better-fetch/fetch';
import 'jose/errors';
import '../../shared/better-auth.BUPPRXfK.mjs';
import 'defu';
function getTypeFromZodType(zodType) {
if (zodType instanceof ZodString) {
return "string";
} else if (zodType instanceof ZodNumber) {
return "number";
} else if (zodType instanceof ZodBoolean) {
return "boolean";
} else if (zodType instanceof ZodObject) {
return "object";
} else if (zodType instanceof ZodArray) {
return "array";
}
return "string";
}
function getFieldSchema(field) {
const schema = {
type: field.type === "date" ? "string" : field.type
};
if (field.defaultValue !== void 0) {
schema.default = typeof field.defaultValue === "function" ? "Generated at runtime" : field.defaultValue;
}
if (field.input === false) {
schema.readOnly = true;
}
return schema;
}
function getParameters(options) {
const parameters = [];
if (options.metadata?.openapi?.parameters) {
parameters.push(...options.metadata.openapi.parameters);
return parameters;
}
if (options.query instanceof ZodObject) {
Object.entries(options.query.shape).forEach(([key, value]) => {
if (value instanceof ZodType) {
parameters.push({
name: key,
in: "query",
schema: {
type: getTypeFromZodType(value),
..."minLength" in value && value.minLength ? {
minLength: value.minLength
} : {},
description: value.description
}
});
}
});
}
return parameters;
}
function getRequestBody(options) {
if (options.metadata?.openapi?.requestBody) {
return options.metadata.openapi.requestBody;
}
if (!options.body) return void 0;
if (options.body instanceof ZodObject || options.body instanceof ZodOptional) {
const shape = options.body.shape;
if (!shape) return void 0;
const properties = {};
const required = [];
Object.entries(shape).forEach(([key, value]) => {
if (value instanceof ZodType) {
properties[key] = {
type: getTypeFromZodType(value),
description: value.description
};
if (!(value instanceof z.ZodOptional)) {
required.push(key);
}
}
});
return {
required: options.body instanceof ZodOptional ? false : options.body ? true : false,
content: {
"application/json": {
schema: {
type: "object",
properties,
required
}
}
}
};
}
return void 0;
}
function getResponse(responses) {
return {
"400": {
content: {
"application/json": {
schema: {
type: "object",
properties: {
message: {
type: "string"
}
},
required: ["message"]
}
}
},
description: "Bad Request. Usually due to missing parameters, or invalid parameters."
},
"401": {
content: {
"application/json": {
schema: {
type: "object",
properties: {
message: {
type: "string"
}
},
required: ["message"]
}
}
},
description: "Unauthorized. Due to missing or invalid authentication."
},
"403": {
content: {
"application/json": {
schema: {
type: "object",
properties: {
message: {
type: "string"
}
}
}
}
},
description: "Forbidden. You do not have permission to access this resource or to perform this action."
},
"404": {
content: {
"application/json": {
schema: {
type: "object",
properties: {
message: {
type: "string"
}
}
}
}
},
description: "Not Found. The requested resource was not found."
},
"429": {
content: {
"application/json": {
schema: {
type: "object",
properties: {
message: {
type: "string"
}
}
}
}
},
description: "Too Many Requests. You have exceeded the rate limit. Try again later."
},
"500": {
content: {
"application/json": {
schema: {
type: "object",
properties: {
message: {
type: "string"
}
}
}
}
},
description: "Internal Server Error. This is a problem with the server that you cannot fix."
},
...responses
};
}
function toOpenApiPath(path) {
return path.split("/").map((part) => part.startsWith(":") ? `{${part.slice(1)}}` : part).join("/");
}
async function generator(ctx, options) {
const baseEndpoints = getEndpoints(ctx, {
...options,
plugins: []
});
const tables = getAuthTables(options);
const models = Object.entries(tables).reduce((acc, [key, value]) => {
const modelName = key.charAt(0).toUpperCase() + key.slice(1);
const fields = value.fields;
const required = [];
const properties = {
id: { type: "string" }
};
Object.entries(fields).forEach(([fieldKey, fieldValue]) => {
if (!fieldValue) return;
properties[fieldKey] = getFieldSchema(fieldValue);
if (fieldValue.required && fieldValue.input !== false) {
required.push(fieldKey);
}
});
acc[modelName] = {
type: "object",
properties,
...required.length > 0 ? { required } : {}
};
return acc;
}, {});
const components = {
schemas: {
...models
}
};
const paths = {};
Object.entries(baseEndpoints.api).forEach(([_, value]) => {
if (ctx.options.disabledPaths?.includes(value.path)) return;
const options2 = value.options;
if (options2.metadata?.SERVER_ONLY) return;
const path = toOpenApiPath(value.path);
if (options2.method === "GET") {
paths[path] = {
get: {
tags: ["Default", ...options2.metadata?.openapi?.tags || []],
description: options2.metadata?.openapi?.description,
operationId: options2.metadata?.openapi?.operationId,
security: [
{
bearerAuth: []
}
],
parameters: getParameters(options2),
responses: getResponse(options2.metadata?.openapi?.responses)
}
};
}
if (options2.method === "POST") {
const body = getRequestBody(options2);
paths[path] = {
post: {
tags: ["Default", ...options2.metadata?.openapi?.tags || []],
description: options2.metadata?.openapi?.description,
operationId: options2.metadata?.openapi?.operationId,
security: [
{
bearerAuth: []
}
],
parameters: getParameters(options2),
...body ? { requestBody: body } : {
requestBody: {
//set body none
content: {
"application/json": {
schema: {
type: "object",
properties: {}
}
}
}
}
},
responses: getResponse(options2.metadata?.openapi?.responses)
}
};
}
});
for (const plugin of options.plugins || []) {
if (plugin.id === "open-api") {
continue;
}
const pluginEndpoints = getEndpoints(ctx, {
...options,
plugins: [plugin]
});
const api = Object.keys(pluginEndpoints.api).map((key) => {
if (baseEndpoints.api[key] === void 0) {
return pluginEndpoints.api[key];
}
return null;
}).filter((x) => x !== null);
Object.entries(api).forEach(([key, value]) => {
if (ctx.options.disabledPaths?.includes(value.path)) return;
const options2 = value.options;
if (options2.metadata?.SERVER_ONLY) return;
const path = toOpenApiPath(value.path);
if (options2.method === "GET") {
paths[path] = {
get: {
tags: options2.metadata?.openapi?.tags || [
plugin.id.charAt(0).toUpperCase() + plugin.id.slice(1)
],
description: options2.metadata?.openapi?.description,
operationId: options2.metadata?.openapi?.operationId,
security: [
{
bearerAuth: []
}
],
parameters: getParameters(options2),
responses: getResponse(options2.metadata?.openapi?.responses)
}
};
}
if (options2.method === "POST") {
paths[path] = {
post: {
tags: options2.metadata?.openapi?.tags || [
plugin.id.charAt(0).toUpperCase() + plugin.id.slice(1)
],
description: options2.metadata?.openapi?.description,
operationId: options2.metadata?.openapi?.operationId,
security: [
{
bearerAuth: []
}
],
parameters: getParameters(options2),
requestBody: getRequestBody(options2),
responses: getResponse(options2.metadata?.openapi?.responses)
}
};
}
});
}
const res = {
openapi: "3.1.1",
info: {
title: "Better Auth",
description: "API Reference for your Better Auth Instance",
version: "1.1.0"
},
components: {
...components,
securitySchemes: {
apiKeyCookie: {
type: "apiKey",
in: "cookie",
name: "apiKeyCookie",
description: "API Key authentication via cookie"
},
bearerAuth: {
type: "http",
scheme: "bearer",
description: "Bearer token authentication"
}
}
},
security: [
{
apiKeyCookie: [],
bearerAuth: []
}
],
servers: [
{
url: ctx.baseURL
}
],
tags: [
{
name: "Default",
description: "Default endpoints that are included with Better Auth by default. These endpoints are not part of any plugin."
}
],
paths
};
return res;
}
const logo = `<svg width="75" height="75" viewBox="0 0 75 75" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
<rect width="75" height="75" fill="url(#pattern0_21_12)"/>
<defs>
<pattern id="pattern0_21_12" patternContentUnits="objectBoundingBox" width="1" height="1">
<use xlink:href="#image0_21_12" transform="scale(0.00094697)"/>
</pattern>
<image id="image0_21_12" width="1056" height="1056" xlink:href=""/>
</defs>
</svg>
`;
const getHTML = (apiReference) => `<!doctype html>
<html>
<head>
<title>Scalar API Reference</title>
<meta charset="utf-8" />
<meta
name="viewport"
content="width=device-width, initial-scale=1" />
</head>
<body>
<script
id="api-reference"
type="application/json">
${JSON.stringify(apiReference)}
<\/script>
<script>
var configuration = {
favicon: "data:image/svg+xml;utf8,${encodeURIComponent(logo)}",
theme: "saturn",
metaData: {
title: "Better Auth API",
description: "API Reference for your Better Auth Instance",
}
}
document.getElementById('api-reference').dataset.configuration =
JSON.stringify(configuration)
<\/script>
<script src="https://cdn.jsdelivr.net/npm/@scalar/api-reference"><\/script>
</body>
</html>`;
const openAPI = (options) => {
const path = options?.path ?? "/reference";
return {
id: "open-api",
endpoints: {
generateOpenAPISchema: createAuthEndpoint(
"/open-api/generate-schema",
{
method: "GET"
},
async (ctx) => {
const schema = await generator(ctx.context, ctx.context.options);
return ctx.json(schema);
}
),
openAPIReference: createAuthEndpoint(
path,
{
method: "GET",
metadata: {
isAction: false
}
},
async (ctx) => {
if (options?.disableDefaultReference) {
throw new APIError("NOT_FOUND");
}
const schema = await generator(ctx.context, ctx.context.options);
return new Response(getHTML(schema), {
headers: {
"Content-Type": "text/html"
}
});
}
)
}
};
};
export { openAPI };