UNPKG

@zpg6-test-pkgs/better-auth

Version:

The most comprehensive authentication library for TypeScript.

494 lines (488 loc) 46.1 kB
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 };