This is page 55 of 67. Use http://codebase.md/better-auth/better-auth?lines=true&page={x} to view the full context. # Directory Structure ``` ├── .gitattributes ├── .github │ ├── CODEOWNERS │ ├── FUNDING.yml │ ├── ISSUE_TEMPLATE │ │ ├── bug_report.yml │ │ └── feature_request.yml │ ├── renovate.json5 │ └── workflows │ ├── ci.yml │ ├── e2e.yml │ ├── preview.yml │ └── release.yml ├── .gitignore ├── .npmrc ├── .nvmrc ├── .vscode │ └── settings.json ├── banner-dark.png ├── banner.png ├── biome.json ├── bump.config.ts ├── CODE_OF_CONDUCT.md ├── CONTRIBUTING.md ├── demo │ ├── expo-example │ │ ├── .env.example │ │ ├── .gitignore │ │ ├── app.config.ts │ │ ├── assets │ │ │ ├── bg-image.jpeg │ │ │ ├── fonts │ │ │ │ └── SpaceMono-Regular.ttf │ │ │ ├── icon.png │ │ │ └── images │ │ │ ├── adaptive-icon.png │ │ │ ├── favicon.png │ │ │ ├── logo.png │ │ │ ├── partial-react-logo.png │ │ │ ├── react-logo.png │ │ │ ├── [email protected] │ │ │ ├── [email protected] │ │ │ └── splash.png │ │ ├── babel.config.js │ │ ├── components.json │ │ ├── expo-env.d.ts │ │ ├── index.ts │ │ ├── metro.config.js │ │ ├── nativewind-env.d.ts │ │ ├── package.json │ │ ├── README.md │ │ ├── src │ │ │ ├── app │ │ │ │ ├── _layout.tsx │ │ │ │ ├── api │ │ │ │ │ └── auth │ │ │ │ │ └── [...route]+api.ts │ │ │ │ ├── dashboard.tsx │ │ │ │ ├── forget-password.tsx │ │ │ │ ├── index.tsx │ │ │ │ └── sign-up.tsx │ │ │ ├── components │ │ │ │ ├── icons │ │ │ │ │ └── google.tsx │ │ │ │ └── ui │ │ │ │ ├── avatar.tsx │ │ │ │ ├── button.tsx │ │ │ │ ├── card.tsx │ │ │ │ ├── dialog.tsx │ │ │ │ ├── input.tsx │ │ │ │ ├── separator.tsx │ │ │ │ └── text.tsx │ │ │ ├── global.css │ │ │ └── lib │ │ │ ├── auth-client.ts │ │ │ ├── auth.ts │ │ │ ├── icons │ │ │ │ ├── iconWithClassName.ts │ │ │ │ └── X.tsx │ │ │ └── utils.ts │ │ ├── tailwind.config.js │ │ └── tsconfig.json │ └── nextjs │ ├── .env.example │ ├── .gitignore │ ├── app │ │ ├── (auth) │ │ │ ├── forget-password │ │ │ │ └── page.tsx │ │ │ ├── reset-password │ │ │ │ └── page.tsx │ │ │ ├── sign-in │ │ │ │ ├── loading.tsx │ │ │ │ └── page.tsx │ │ │ └── two-factor │ │ │ ├── otp │ │ │ │ └── page.tsx │ │ │ └── page.tsx │ │ ├── accept-invitation │ │ │ └── [id] │ │ │ ├── invitation-error.tsx │ │ │ └── page.tsx │ │ ├── admin │ │ │ └── page.tsx │ │ ├── api │ │ │ └── auth │ │ │ └── [...all] │ │ │ └── route.ts │ │ ├── apps │ │ │ └── register │ │ │ └── page.tsx │ │ ├── client-test │ │ │ └── page.tsx │ │ ├── dashboard │ │ │ ├── change-plan.tsx │ │ │ ├── client.tsx │ │ │ ├── organization-card.tsx │ │ │ ├── page.tsx │ │ │ ├── upgrade-button.tsx │ │ │ └── user-card.tsx │ │ ├── device │ │ │ ├── approve │ │ │ │ └── page.tsx │ │ │ ├── denied │ │ │ │ └── page.tsx │ │ │ ├── layout.tsx │ │ │ ├── page.tsx │ │ │ └── success │ │ │ └── page.tsx │ │ ├── favicon.ico │ │ ├── features.tsx │ │ ├── fonts │ │ │ ├── GeistMonoVF.woff │ │ │ └── GeistVF.woff │ │ ├── globals.css │ │ ├── layout.tsx │ │ ├── oauth │ │ │ └── authorize │ │ │ ├── concet-buttons.tsx │ │ │ └── page.tsx │ │ ├── page.tsx │ │ └── pricing │ │ └── page.tsx │ ├── components │ │ ├── account-switch.tsx │ │ ├── blocks │ │ │ └── pricing.tsx │ │ ├── logo.tsx │ │ ├── one-tap.tsx │ │ ├── sign-in-btn.tsx │ │ ├── sign-in.tsx │ │ ├── sign-up.tsx │ │ ├── theme-provider.tsx │ │ ├── theme-toggle.tsx │ │ ├── tier-labels.tsx │ │ ├── ui │ │ │ ├── accordion.tsx │ │ │ ├── alert-dialog.tsx │ │ │ ├── alert.tsx │ │ │ ├── aspect-ratio.tsx │ │ │ ├── avatar.tsx │ │ │ ├── badge.tsx │ │ │ ├── breadcrumb.tsx │ │ │ ├── button.tsx │ │ │ ├── calendar.tsx │ │ │ ├── card.tsx │ │ │ ├── carousel.tsx │ │ │ ├── chart.tsx │ │ │ ├── checkbox.tsx │ │ │ ├── collapsible.tsx │ │ │ ├── command.tsx │ │ │ ├── context-menu.tsx │ │ │ ├── copy-button.tsx │ │ │ ├── dialog.tsx │ │ │ ├── drawer.tsx │ │ │ ├── dropdown-menu.tsx │ │ │ ├── form.tsx │ │ │ ├── hover-card.tsx │ │ │ ├── input-otp.tsx │ │ │ ├── input.tsx │ │ │ ├── label.tsx │ │ │ ├── menubar.tsx │ │ │ ├── navigation-menu.tsx │ │ │ ├── pagination.tsx │ │ │ ├── password-input.tsx │ │ │ ├── popover.tsx │ │ │ ├── progress.tsx │ │ │ ├── radio-group.tsx │ │ │ ├── resizable.tsx │ │ │ ├── scroll-area.tsx │ │ │ ├── select.tsx │ │ │ ├── separator.tsx │ │ │ ├── sheet.tsx │ │ │ ├── skeleton.tsx │ │ │ ├── slider.tsx │ │ │ ├── sonner.tsx │ │ │ ├── switch.tsx │ │ │ ├── table.tsx │ │ │ ├── tabs.tsx │ │ │ ├── tabs2.tsx │ │ │ ├── textarea.tsx │ │ │ ├── toast.tsx │ │ │ ├── toaster.tsx │ │ │ ├── toggle-group.tsx │ │ │ ├── toggle.tsx │ │ │ └── tooltip.tsx │ │ └── wrapper.tsx │ ├── components.json │ ├── hooks │ │ └── use-toast.ts │ ├── lib │ │ ├── auth-client.ts │ │ ├── auth-types.ts │ │ ├── auth.ts │ │ ├── email │ │ │ ├── invitation.tsx │ │ │ ├── resend.ts │ │ │ └── reset-password.tsx │ │ ├── metadata.ts │ │ ├── shared.ts │ │ └── utils.ts │ ├── next.config.ts │ ├── package.json │ ├── postcss.config.mjs │ ├── proxy.ts │ ├── public │ │ ├── __og.png │ │ ├── _og.png │ │ ├── favicon │ │ │ ├── android-chrome-192x192.png │ │ │ ├── android-chrome-512x512.png │ │ │ ├── apple-touch-icon.png │ │ │ ├── favicon-16x16.png │ │ │ ├── favicon-32x32.png │ │ │ ├── favicon.ico │ │ │ ├── light │ │ │ │ ├── android-chrome-192x192.png │ │ │ │ ├── android-chrome-512x512.png │ │ │ │ ├── apple-touch-icon.png │ │ │ │ ├── favicon-16x16.png │ │ │ │ ├── favicon-32x32.png │ │ │ │ ├── favicon.ico │ │ │ │ └── site.webmanifest │ │ │ └── site.webmanifest │ │ ├── logo.svg │ │ └── og.png │ ├── README.md │ ├── tailwind.config.ts │ ├── tsconfig.json │ └── turbo.json ├── docker-compose.yml ├── docs │ ├── .env.example │ ├── .gitignore │ ├── app │ │ ├── api │ │ │ ├── ai-chat │ │ │ │ └── route.ts │ │ │ ├── analytics │ │ │ │ ├── conversation │ │ │ │ │ └── route.ts │ │ │ │ ├── event │ │ │ │ │ └── route.ts │ │ │ │ └── feedback │ │ │ │ └── route.ts │ │ │ ├── chat │ │ │ │ └── route.ts │ │ │ ├── og │ │ │ │ └── route.tsx │ │ │ ├── og-release │ │ │ │ └── route.tsx │ │ │ ├── search │ │ │ │ └── route.ts │ │ │ └── support │ │ │ └── route.ts │ │ ├── blog │ │ │ ├── _components │ │ │ │ ├── _layout.tsx │ │ │ │ ├── blog-list.tsx │ │ │ │ ├── changelog-layout.tsx │ │ │ │ ├── default-changelog.tsx │ │ │ │ ├── fmt-dates.tsx │ │ │ │ ├── icons.tsx │ │ │ │ ├── stat-field.tsx │ │ │ │ └── support.tsx │ │ │ ├── [[...slug]] │ │ │ │ └── page.tsx │ │ │ └── layout.tsx │ │ ├── changelogs │ │ │ ├── _components │ │ │ │ ├── _layout.tsx │ │ │ │ ├── changelog-layout.tsx │ │ │ │ ├── default-changelog.tsx │ │ │ │ ├── fmt-dates.tsx │ │ │ │ ├── grid-pattern.tsx │ │ │ │ ├── icons.tsx │ │ │ │ └── stat-field.tsx │ │ │ ├── [[...slug]] │ │ │ │ └── page.tsx │ │ │ └── layout.tsx │ │ ├── community │ │ │ ├── _components │ │ │ │ ├── header.tsx │ │ │ │ └── stats.tsx │ │ │ └── page.tsx │ │ ├── docs │ │ │ ├── [[...slug]] │ │ │ │ ├── page.client.tsx │ │ │ │ └── page.tsx │ │ │ ├── layout.tsx │ │ │ └── lib │ │ │ └── get-llm-text.ts │ │ ├── global.css │ │ ├── layout.config.tsx │ │ ├── layout.tsx │ │ ├── llms.txt │ │ │ ├── [...slug] │ │ │ │ └── route.ts │ │ │ └── route.ts │ │ ├── not-found.tsx │ │ ├── page.tsx │ │ ├── reference │ │ │ └── route.ts │ │ ├── sitemap.xml │ │ ├── static.json │ │ │ └── route.ts │ │ └── v1 │ │ ├── _components │ │ │ └── v1-text.tsx │ │ ├── bg-line.tsx │ │ └── page.tsx │ ├── assets │ │ ├── Geist.ttf │ │ └── GeistMono.ttf │ ├── components │ │ ├── ai-chat-modal.tsx │ │ ├── anchor-scroll-fix.tsx │ │ ├── api-method-tabs.tsx │ │ ├── api-method.tsx │ │ ├── banner.tsx │ │ ├── blocks │ │ │ └── features.tsx │ │ ├── builder │ │ │ ├── beam.tsx │ │ │ ├── code-tabs │ │ │ │ ├── code-editor.tsx │ │ │ │ ├── code-tabs.tsx │ │ │ │ ├── index.tsx │ │ │ │ ├── tab-bar.tsx │ │ │ │ └── theme.ts │ │ │ ├── index.tsx │ │ │ ├── sign-in.tsx │ │ │ ├── sign-up.tsx │ │ │ ├── social-provider.tsx │ │ │ ├── store.ts │ │ │ └── tabs.tsx │ │ ├── display-techstack.tsx │ │ ├── divider-text.tsx │ │ ├── docs │ │ │ ├── docs.client.tsx │ │ │ ├── docs.tsx │ │ │ ├── layout │ │ │ │ ├── nav.tsx │ │ │ │ ├── theme-toggle.tsx │ │ │ │ ├── toc-thumb.tsx │ │ │ │ └── toc.tsx │ │ │ ├── page.client.tsx │ │ │ ├── page.tsx │ │ │ ├── shared.tsx │ │ │ └── ui │ │ │ ├── button.tsx │ │ │ ├── collapsible.tsx │ │ │ ├── popover.tsx │ │ │ └── scroll-area.tsx │ │ ├── endpoint.tsx │ │ ├── features.tsx │ │ ├── floating-ai-search.tsx │ │ ├── fork-button.tsx │ │ ├── generate-apple-jwt.tsx │ │ ├── generate-secret.tsx │ │ ├── github-stat.tsx │ │ ├── icons.tsx │ │ ├── landing │ │ │ ├── gradient-bg.tsx │ │ │ ├── grid-pattern.tsx │ │ │ ├── hero.tsx │ │ │ ├── section-svg.tsx │ │ │ ├── section.tsx │ │ │ ├── spotlight.tsx │ │ │ └── testimonials.tsx │ │ ├── logo-context-menu.tsx │ │ ├── logo.tsx │ │ ├── markdown-renderer.tsx │ │ ├── markdown.tsx │ │ ├── mdx │ │ │ ├── add-to-cursor.tsx │ │ │ └── database-tables.tsx │ │ ├── message-feedback.tsx │ │ ├── mobile-search-icon.tsx │ │ ├── nav-bar.tsx │ │ ├── nav-link.tsx │ │ ├── nav-mobile.tsx │ │ ├── promo-card.tsx │ │ ├── resource-card.tsx │ │ ├── resource-grid.tsx │ │ ├── resource-section.tsx │ │ ├── ripple.tsx │ │ ├── search-dialog.tsx │ │ ├── side-bar.tsx │ │ ├── sidebar-content.tsx │ │ ├── techstack-icons.tsx │ │ ├── theme-provider.tsx │ │ ├── theme-toggler.tsx │ │ └── ui │ │ ├── accordion.tsx │ │ ├── alert-dialog.tsx │ │ ├── alert.tsx │ │ ├── aside-link.tsx │ │ ├── aspect-ratio.tsx │ │ ├── avatar.tsx │ │ ├── background-beams.tsx │ │ ├── background-boxes.tsx │ │ ├── badge.tsx │ │ ├── breadcrumb.tsx │ │ ├── button.tsx │ │ ├── calendar.tsx │ │ ├── callout.tsx │ │ ├── card.tsx │ │ ├── carousel.tsx │ │ ├── chart.tsx │ │ ├── checkbox.tsx │ │ ├── code-block.tsx │ │ ├── collapsible.tsx │ │ ├── command.tsx │ │ ├── context-menu.tsx │ │ ├── dialog.tsx │ │ ├── drawer.tsx │ │ ├── dropdown-menu.tsx │ │ ├── dynamic-code-block.tsx │ │ ├── fade-in.tsx │ │ ├── form.tsx │ │ ├── hover-card.tsx │ │ ├── input-otp.tsx │ │ ├── input.tsx │ │ ├── label.tsx │ │ ├── menubar.tsx │ │ ├── navigation-menu.tsx │ │ ├── pagination.tsx │ │ ├── popover.tsx │ │ ├── progress.tsx │ │ ├── radio-group.tsx │ │ ├── resizable.tsx │ │ ├── scroll-area.tsx │ │ ├── select.tsx │ │ ├── separator.tsx │ │ ├── sheet.tsx │ │ ├── sidebar.tsx │ │ ├── skeleton.tsx │ │ ├── slider.tsx │ │ ├── sonner.tsx │ │ ├── sparkles.tsx │ │ ├── switch.tsx │ │ ├── table.tsx │ │ ├── tabs.tsx │ │ ├── textarea.tsx │ │ ├── toggle-group.tsx │ │ ├── toggle.tsx │ │ ├── tooltip-docs.tsx │ │ ├── tooltip.tsx │ │ └── use-copy-button.tsx │ ├── components.json │ ├── content │ │ ├── blogs │ │ │ ├── 0-supabase-auth-to-planetscale-migration.mdx │ │ │ ├── 1-3.mdx │ │ │ ├── authjs-joins-better-auth.mdx │ │ │ └── seed-round.mdx │ │ ├── changelogs │ │ │ ├── 1-2.mdx │ │ │ └── 1.0.mdx │ │ └── docs │ │ ├── adapters │ │ │ ├── community-adapters.mdx │ │ │ ├── drizzle.mdx │ │ │ ├── mongo.mdx │ │ │ ├── mssql.mdx │ │ │ ├── mysql.mdx │ │ │ ├── other-relational-databases.mdx │ │ │ ├── postgresql.mdx │ │ │ ├── prisma.mdx │ │ │ └── sqlite.mdx │ │ ├── authentication │ │ │ ├── apple.mdx │ │ │ ├── atlassian.mdx │ │ │ ├── cognito.mdx │ │ │ ├── discord.mdx │ │ │ ├── dropbox.mdx │ │ │ ├── email-password.mdx │ │ │ ├── facebook.mdx │ │ │ ├── figma.mdx │ │ │ ├── github.mdx │ │ │ ├── gitlab.mdx │ │ │ ├── google.mdx │ │ │ ├── huggingface.mdx │ │ │ ├── kakao.mdx │ │ │ ├── kick.mdx │ │ │ ├── line.mdx │ │ │ ├── linear.mdx │ │ │ ├── linkedin.mdx │ │ │ ├── microsoft.mdx │ │ │ ├── naver.mdx │ │ │ ├── notion.mdx │ │ │ ├── other-social-providers.mdx │ │ │ ├── paypal.mdx │ │ │ ├── reddit.mdx │ │ │ ├── roblox.mdx │ │ │ ├── salesforce.mdx │ │ │ ├── slack.mdx │ │ │ ├── spotify.mdx │ │ │ ├── tiktok.mdx │ │ │ ├── twitch.mdx │ │ │ ├── twitter.mdx │ │ │ ├── vk.mdx │ │ │ └── zoom.mdx │ │ ├── basic-usage.mdx │ │ ├── comparison.mdx │ │ ├── concepts │ │ │ ├── api.mdx │ │ │ ├── cli.mdx │ │ │ ├── client.mdx │ │ │ ├── cookies.mdx │ │ │ ├── database.mdx │ │ │ ├── email.mdx │ │ │ ├── hooks.mdx │ │ │ ├── oauth.mdx │ │ │ ├── plugins.mdx │ │ │ ├── rate-limit.mdx │ │ │ ├── session-management.mdx │ │ │ ├── typescript.mdx │ │ │ └── users-accounts.mdx │ │ ├── examples │ │ │ ├── astro.mdx │ │ │ ├── next-js.mdx │ │ │ ├── nuxt.mdx │ │ │ ├── remix.mdx │ │ │ └── svelte-kit.mdx │ │ ├── guides │ │ │ ├── auth0-migration-guide.mdx │ │ │ ├── browser-extension-guide.mdx │ │ │ ├── clerk-migration-guide.mdx │ │ │ ├── create-a-db-adapter.mdx │ │ │ ├── next-auth-migration-guide.mdx │ │ │ ├── optimizing-for-performance.mdx │ │ │ ├── saml-sso-with-okta.mdx │ │ │ ├── supabase-migration-guide.mdx │ │ │ └── your-first-plugin.mdx │ │ ├── installation.mdx │ │ ├── integrations │ │ │ ├── astro.mdx │ │ │ ├── convex.mdx │ │ │ ├── elysia.mdx │ │ │ ├── expo.mdx │ │ │ ├── express.mdx │ │ │ ├── fastify.mdx │ │ │ ├── hono.mdx │ │ │ ├── lynx.mdx │ │ │ ├── nestjs.mdx │ │ │ ├── next.mdx │ │ │ ├── nitro.mdx │ │ │ ├── nuxt.mdx │ │ │ ├── remix.mdx │ │ │ ├── solid-start.mdx │ │ │ ├── svelte-kit.mdx │ │ │ ├── tanstack.mdx │ │ │ └── waku.mdx │ │ ├── introduction.mdx │ │ ├── meta.json │ │ ├── plugins │ │ │ ├── 2fa.mdx │ │ │ ├── admin.mdx │ │ │ ├── anonymous.mdx │ │ │ ├── api-key.mdx │ │ │ ├── autumn.mdx │ │ │ ├── bearer.mdx │ │ │ ├── captcha.mdx │ │ │ ├── community-plugins.mdx │ │ │ ├── device-authorization.mdx │ │ │ ├── dodopayments.mdx │ │ │ ├── dub.mdx │ │ │ ├── email-otp.mdx │ │ │ ├── generic-oauth.mdx │ │ │ ├── have-i-been-pwned.mdx │ │ │ ├── jwt.mdx │ │ │ ├── last-login-method.mdx │ │ │ ├── magic-link.mdx │ │ │ ├── mcp.mdx │ │ │ ├── multi-session.mdx │ │ │ ├── oauth-proxy.mdx │ │ │ ├── oidc-provider.mdx │ │ │ ├── one-tap.mdx │ │ │ ├── one-time-token.mdx │ │ │ ├── open-api.mdx │ │ │ ├── organization.mdx │ │ │ ├── passkey.mdx │ │ │ ├── phone-number.mdx │ │ │ ├── polar.mdx │ │ │ ├── siwe.mdx │ │ │ ├── sso.mdx │ │ │ ├── stripe.mdx │ │ │ └── username.mdx │ │ └── reference │ │ ├── contributing.mdx │ │ ├── faq.mdx │ │ ├── options.mdx │ │ ├── resources.mdx │ │ ├── security.mdx │ │ └── telemetry.mdx │ ├── hooks │ │ └── use-mobile.ts │ ├── ignore-build.sh │ ├── lib │ │ ├── blog.ts │ │ ├── chat │ │ │ └── inkeep-qa-schema.ts │ │ ├── constants.ts │ │ ├── export-search-indexes.ts │ │ ├── inkeep-analytics.ts │ │ ├── is-active.ts │ │ ├── metadata.ts │ │ ├── source.ts │ │ └── utils.ts │ ├── next.config.js │ ├── package.json │ ├── postcss.config.js │ ├── proxy.ts │ ├── public │ │ ├── avatars │ │ │ └── beka.jpg │ │ ├── blogs │ │ │ ├── authjs-joins.png │ │ │ ├── seed-round.png │ │ │ └── supabase-ps.png │ │ ├── branding │ │ │ ├── better-auth-brand-assets.zip │ │ │ ├── better-auth-logo-dark.png │ │ │ ├── better-auth-logo-dark.svg │ │ │ ├── better-auth-logo-light.png │ │ │ ├── better-auth-logo-light.svg │ │ │ ├── better-auth-logo-wordmark-dark.png │ │ │ ├── better-auth-logo-wordmark-dark.svg │ │ │ ├── better-auth-logo-wordmark-light.png │ │ │ └── better-auth-logo-wordmark-light.svg │ │ ├── extension-id.png │ │ ├── favicon │ │ │ ├── android-chrome-192x192.png │ │ │ ├── android-chrome-512x512.png │ │ │ ├── apple-touch-icon.png │ │ │ ├── favicon-16x16.png │ │ │ ├── favicon-32x32.png │ │ │ ├── favicon.ico │ │ │ ├── light │ │ │ │ ├── android-chrome-192x192.png │ │ │ │ ├── android-chrome-512x512.png │ │ │ │ ├── apple-touch-icon.png │ │ │ │ ├── favicon-16x16.png │ │ │ │ ├── favicon-32x32.png │ │ │ │ ├── favicon.ico │ │ │ │ └── site.webmanifest │ │ │ └── site.webmanifest │ │ ├── images │ │ │ └── blogs │ │ │ └── better auth (1).png │ │ ├── logo.png │ │ ├── logo.svg │ │ ├── LogoDark.webp │ │ ├── LogoLight.webp │ │ ├── og.png │ │ ├── open-api-reference.png │ │ ├── people-say │ │ │ ├── code-with-antonio.jpg │ │ │ ├── dagmawi-babi.png │ │ │ ├── dax.png │ │ │ ├── dev-ed.png │ │ │ ├── egoist.png │ │ │ ├── guillermo-rauch.png │ │ │ ├── jonathan-wilke.png │ │ │ ├── josh-tried-coding.jpg │ │ │ ├── kitze.jpg │ │ │ ├── lazar-nikolov.png │ │ │ ├── nizzy.png │ │ │ ├── omar-mcadam.png │ │ │ ├── ryan-vogel.jpg │ │ │ ├── saltyatom.jpg │ │ │ ├── sebastien-chopin.png │ │ │ ├── shreyas-mididoddi.png │ │ │ ├── tech-nerd.png │ │ │ ├── theo.png │ │ │ ├── vybhav-bhargav.png │ │ │ └── xavier-pladevall.jpg │ │ ├── plus.svg │ │ ├── release-og │ │ │ ├── 1-2.png │ │ │ ├── 1-3.png │ │ │ └── changelog-og.png │ │ └── v1-og.png │ ├── README.md │ ├── scripts │ │ ├── endpoint-to-doc │ │ │ ├── index.ts │ │ │ ├── input.ts │ │ │ ├── output.mdx │ │ │ └── readme.md │ │ └── sync-orama.ts │ ├── source.config.ts │ ├── tailwind.config.js │ ├── tsconfig.json │ └── turbo.json ├── e2e │ ├── integration │ │ ├── package.json │ │ ├── playwright.config.ts │ │ ├── solid-vinxi │ │ │ ├── .gitignore │ │ │ ├── app.config.ts │ │ │ ├── e2e │ │ │ │ ├── test.spec.ts │ │ │ │ └── utils.ts │ │ │ ├── package.json │ │ │ ├── public │ │ │ │ └── favicon.ico │ │ │ ├── src │ │ │ │ ├── app.tsx │ │ │ │ ├── entry-client.tsx │ │ │ │ ├── entry-server.tsx │ │ │ │ ├── global.d.ts │ │ │ │ ├── lib │ │ │ │ │ ├── auth-client.ts │ │ │ │ │ └── auth.ts │ │ │ │ └── routes │ │ │ │ ├── [...404].tsx │ │ │ │ ├── api │ │ │ │ │ └── auth │ │ │ │ │ └── [...all].ts │ │ │ │ └── index.tsx │ │ │ └── tsconfig.json │ │ ├── test-utils │ │ │ ├── package.json │ │ │ └── src │ │ │ └── playwright.ts │ │ └── vanilla-node │ │ ├── e2e │ │ │ ├── app.ts │ │ │ ├── domain.spec.ts │ │ │ ├── postgres-js.spec.ts │ │ │ ├── test.spec.ts │ │ │ └── utils.ts │ │ ├── index.html │ │ ├── package.json │ │ ├── src │ │ │ ├── main.ts │ │ │ └── vite-env.d.ts │ │ ├── tsconfig.json │ │ └── vite.config.ts │ └── smoke │ ├── package.json │ ├── test │ │ ├── bun.spec.ts │ │ ├── cloudflare.spec.ts │ │ ├── deno.spec.ts │ │ ├── fixtures │ │ │ ├── bun-simple.ts │ │ │ ├── cloudflare │ │ │ │ ├── .gitignore │ │ │ │ ├── drizzle │ │ │ │ │ ├── 0000_clean_vector.sql │ │ │ │ │ └── meta │ │ │ │ │ ├── _journal.json │ │ │ │ │ └── 0000_snapshot.json │ │ │ │ ├── drizzle.config.ts │ │ │ │ ├── package.json │ │ │ │ ├── src │ │ │ │ │ ├── auth-schema.ts │ │ │ │ │ ├── db.ts │ │ │ │ │ └── index.ts │ │ │ │ ├── test │ │ │ │ │ ├── apply-migrations.ts │ │ │ │ │ ├── env.d.ts │ │ │ │ │ └── index.test.ts │ │ │ │ ├── tsconfig.json │ │ │ │ ├── vitest.config.ts │ │ │ │ ├── worker-configuration.d.ts │ │ │ │ └── wrangler.json │ │ │ ├── deno-simple.ts │ │ │ ├── tsconfig-decelration │ │ │ │ ├── package.json │ │ │ │ ├── src │ │ │ │ │ └── index.ts │ │ │ │ └── tsconfig.json │ │ │ ├── tsconfig-exact-optional-property-types │ │ │ │ ├── package.json │ │ │ │ ├── src │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── organization.ts │ │ │ │ │ ├── user-additional-fields.ts │ │ │ │ │ └── username.ts │ │ │ │ └── tsconfig.json │ │ │ ├── tsconfig-verbatim-module-syntax-node10 │ │ │ │ ├── package.json │ │ │ │ ├── src │ │ │ │ │ └── index.ts │ │ │ │ └── tsconfig.json │ │ │ └── vite │ │ │ ├── package.json │ │ │ ├── src │ │ │ │ ├── client.ts │ │ │ │ └── server.ts │ │ │ ├── tsconfig.json │ │ │ └── vite.config.ts │ │ ├── ssr.ts │ │ ├── typecheck.spec.ts │ │ └── vite.spec.ts │ └── tsconfig.json ├── LICENSE.md ├── package.json ├── packages │ ├── better-auth │ │ ├── package.json │ │ ├── README.md │ │ ├── src │ │ │ ├── __snapshots__ │ │ │ │ └── init.test.ts.snap │ │ │ ├── adapters │ │ │ │ ├── adapter-factory │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── test │ │ │ │ │ │ ├── __snapshots__ │ │ │ │ │ │ │ └── adapter-factory.test.ts.snap │ │ │ │ │ │ └── adapter-factory.test.ts │ │ │ │ │ └── types.ts │ │ │ │ ├── create-test-suite.ts │ │ │ │ ├── drizzle-adapter │ │ │ │ │ ├── drizzle-adapter.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ └── test │ │ │ │ │ ├── .gitignore │ │ │ │ │ ├── adapter.drizzle.mysql.test.ts │ │ │ │ │ ├── adapter.drizzle.pg.test.ts │ │ │ │ │ ├── adapter.drizzle.sqlite.test.ts │ │ │ │ │ └── generate-schema.ts │ │ │ │ ├── index.ts │ │ │ │ ├── kysely-adapter │ │ │ │ │ ├── bun-sqlite-dialect.ts │ │ │ │ │ ├── dialect.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── kysely-adapter.ts │ │ │ │ │ ├── node-sqlite-dialect.ts │ │ │ │ │ ├── test │ │ │ │ │ │ ├── adapter.kysely.mssql.test.ts │ │ │ │ │ │ ├── adapter.kysely.mysql.test.ts │ │ │ │ │ │ ├── adapter.kysely.pg.test.ts │ │ │ │ │ │ ├── adapter.kysely.sqlite.test.ts │ │ │ │ │ │ └── node-sqlite-dialect.test.ts │ │ │ │ │ └── types.ts │ │ │ │ ├── memory-adapter │ │ │ │ │ ├── adapter.memory.test.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ └── memory-adapter.ts │ │ │ │ ├── mongodb-adapter │ │ │ │ │ ├── adapter.mongo-db.test.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ └── mongodb-adapter.ts │ │ │ │ ├── prisma-adapter │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── prisma-adapter.ts │ │ │ │ │ └── test │ │ │ │ │ ├── .gitignore │ │ │ │ │ ├── base.prisma │ │ │ │ │ ├── generate-auth-config.ts │ │ │ │ │ ├── generate-prisma-schema.ts │ │ │ │ │ ├── get-prisma-client.ts │ │ │ │ │ ├── prisma.mysql.test.ts │ │ │ │ │ ├── prisma.pg.test.ts │ │ │ │ │ ├── prisma.sqlite.test.ts │ │ │ │ │ └── push-prisma-schema.ts │ │ │ │ ├── test-adapter.ts │ │ │ │ ├── test.ts │ │ │ │ ├── tests │ │ │ │ │ ├── auth-flow.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── normal.ts │ │ │ │ │ ├── number-id.ts │ │ │ │ │ ├── performance.ts │ │ │ │ │ └── transactions.ts │ │ │ │ └── utils.ts │ │ │ ├── api │ │ │ │ ├── check-endpoint-conflicts.test.ts │ │ │ │ ├── index.test.ts │ │ │ │ ├── index.ts │ │ │ │ ├── middlewares │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── origin-check.test.ts │ │ │ │ │ └── origin-check.ts │ │ │ │ ├── rate-limiter │ │ │ │ │ ├── index.ts │ │ │ │ │ └── rate-limiter.test.ts │ │ │ │ ├── routes │ │ │ │ │ ├── account.test.ts │ │ │ │ │ ├── account.ts │ │ │ │ │ ├── callback.ts │ │ │ │ │ ├── email-verification.test.ts │ │ │ │ │ ├── email-verification.ts │ │ │ │ │ ├── error.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── ok.ts │ │ │ │ │ ├── reset-password.test.ts │ │ │ │ │ ├── reset-password.ts │ │ │ │ │ ├── session-api.test.ts │ │ │ │ │ ├── session.ts │ │ │ │ │ ├── sign-in.test.ts │ │ │ │ │ ├── sign-in.ts │ │ │ │ │ ├── sign-out.test.ts │ │ │ │ │ ├── sign-out.ts │ │ │ │ │ ├── sign-up.test.ts │ │ │ │ │ ├── sign-up.ts │ │ │ │ │ ├── update-user.test.ts │ │ │ │ │ └── update-user.ts │ │ │ │ ├── to-auth-endpoints.test.ts │ │ │ │ └── to-auth-endpoints.ts │ │ │ ├── auth.test.ts │ │ │ ├── auth.ts │ │ │ ├── call.test.ts │ │ │ ├── client │ │ │ │ ├── client-ssr.test.ts │ │ │ │ ├── client.test.ts │ │ │ │ ├── config.ts │ │ │ │ ├── fetch-plugins.ts │ │ │ │ ├── index.ts │ │ │ │ ├── lynx │ │ │ │ │ ├── index.ts │ │ │ │ │ └── lynx-store.ts │ │ │ │ ├── parser.ts │ │ │ │ ├── path-to-object.ts │ │ │ │ ├── plugins │ │ │ │ │ ├── index.ts │ │ │ │ │ └── infer-plugin.ts │ │ │ │ ├── proxy.ts │ │ │ │ ├── query.ts │ │ │ │ ├── react │ │ │ │ │ ├── index.ts │ │ │ │ │ └── react-store.ts │ │ │ │ ├── session-atom.ts │ │ │ │ ├── solid │ │ │ │ │ ├── index.ts │ │ │ │ │ └── solid-store.ts │ │ │ │ ├── svelte │ │ │ │ │ └── index.ts │ │ │ │ ├── test-plugin.ts │ │ │ │ ├── types.ts │ │ │ │ ├── url.test.ts │ │ │ │ ├── vanilla.ts │ │ │ │ └── vue │ │ │ │ ├── index.ts │ │ │ │ └── vue-store.ts │ │ │ ├── cookies │ │ │ │ ├── check-cookies.ts │ │ │ │ ├── cookie-utils.ts │ │ │ │ ├── cookies.test.ts │ │ │ │ └── index.ts │ │ │ ├── crypto │ │ │ │ ├── buffer.ts │ │ │ │ ├── hash.ts │ │ │ │ ├── index.ts │ │ │ │ ├── jwt.ts │ │ │ │ ├── password.test.ts │ │ │ │ ├── password.ts │ │ │ │ └── random.ts │ │ │ ├── db │ │ │ │ ├── db.test.ts │ │ │ │ ├── field.ts │ │ │ │ ├── get-migration.ts │ │ │ │ ├── get-schema.ts │ │ │ │ ├── get-tables.test.ts │ │ │ │ ├── get-tables.ts │ │ │ │ ├── index.ts │ │ │ │ ├── internal-adapter.test.ts │ │ │ │ ├── internal-adapter.ts │ │ │ │ ├── schema.ts │ │ │ │ ├── secondary-storage.test.ts │ │ │ │ ├── to-zod.ts │ │ │ │ ├── utils.ts │ │ │ │ └── with-hooks.ts │ │ │ ├── index.ts │ │ │ ├── init.test.ts │ │ │ ├── init.ts │ │ │ ├── integrations │ │ │ │ ├── next-js.ts │ │ │ │ ├── node.ts │ │ │ │ ├── react-start.ts │ │ │ │ ├── solid-start.ts │ │ │ │ └── svelte-kit.ts │ │ │ ├── oauth2 │ │ │ │ ├── index.ts │ │ │ │ ├── link-account.test.ts │ │ │ │ ├── link-account.ts │ │ │ │ ├── state.ts │ │ │ │ └── utils.ts │ │ │ ├── plugins │ │ │ │ ├── access │ │ │ │ │ ├── access.test.ts │ │ │ │ │ ├── access.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ └── types.ts │ │ │ │ ├── additional-fields │ │ │ │ │ ├── additional-fields.test.ts │ │ │ │ │ └── client.ts │ │ │ │ ├── admin │ │ │ │ │ ├── access │ │ │ │ │ │ ├── index.ts │ │ │ │ │ │ └── statement.ts │ │ │ │ │ ├── admin.test.ts │ │ │ │ │ ├── admin.ts │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── error-codes.ts │ │ │ │ │ ├── has-permission.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── schema.ts │ │ │ │ │ └── types.ts │ │ │ │ ├── anonymous │ │ │ │ │ ├── anon.test.ts │ │ │ │ │ ├── client.ts │ │ │ │ │ └── index.ts │ │ │ │ ├── api-key │ │ │ │ │ ├── api-key.test.ts │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── rate-limit.ts │ │ │ │ │ ├── routes │ │ │ │ │ │ ├── create-api-key.ts │ │ │ │ │ │ ├── delete-all-expired-api-keys.ts │ │ │ │ │ │ ├── delete-api-key.ts │ │ │ │ │ │ ├── get-api-key.ts │ │ │ │ │ │ ├── index.ts │ │ │ │ │ │ ├── list-api-keys.ts │ │ │ │ │ │ ├── update-api-key.ts │ │ │ │ │ │ └── verify-api-key.ts │ │ │ │ │ ├── schema.ts │ │ │ │ │ └── types.ts │ │ │ │ ├── bearer │ │ │ │ │ ├── bearer.test.ts │ │ │ │ │ └── index.ts │ │ │ │ ├── captcha │ │ │ │ │ ├── captcha.test.ts │ │ │ │ │ ├── constants.ts │ │ │ │ │ ├── error-codes.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── types.ts │ │ │ │ │ ├── utils.ts │ │ │ │ │ └── verify-handlers │ │ │ │ │ ├── captchafox.ts │ │ │ │ │ ├── cloudflare-turnstile.ts │ │ │ │ │ ├── google-recaptcha.ts │ │ │ │ │ ├── h-captcha.ts │ │ │ │ │ └── index.ts │ │ │ │ ├── custom-session │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── custom-session.test.ts │ │ │ │ │ └── index.ts │ │ │ │ ├── device-authorization │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── device-authorization.test.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ └── schema.ts │ │ │ │ ├── email-otp │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── email-otp.test.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ └── utils.ts │ │ │ │ ├── generic-oauth │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── generic-oauth.test.ts │ │ │ │ │ └── index.ts │ │ │ │ ├── haveibeenpwned │ │ │ │ │ ├── haveibeenpwned.test.ts │ │ │ │ │ └── index.ts │ │ │ │ ├── index.ts │ │ │ │ ├── jwt │ │ │ │ │ ├── adapter.ts │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── jwt.test.ts │ │ │ │ │ ├── schema.ts │ │ │ │ │ ├── sign.ts │ │ │ │ │ ├── types.ts │ │ │ │ │ └── utils.ts │ │ │ │ ├── last-login-method │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── custom-prefix.test.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ └── last-login-method.test.ts │ │ │ │ ├── magic-link │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── magic-link.test.ts │ │ │ │ │ └── utils.ts │ │ │ │ ├── mcp │ │ │ │ │ ├── authorize.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ └── mcp.test.ts │ │ │ │ ├── multi-session │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ └── multi-session.test.ts │ │ │ │ ├── oauth-proxy │ │ │ │ │ ├── index.ts │ │ │ │ │ └── oauth-proxy.test.ts │ │ │ │ ├── oidc-provider │ │ │ │ │ ├── authorize.ts │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── oidc.test.ts │ │ │ │ │ ├── schema.ts │ │ │ │ │ ├── types.ts │ │ │ │ │ ├── ui.ts │ │ │ │ │ └── utils.ts │ │ │ │ ├── one-tap │ │ │ │ │ ├── client.ts │ │ │ │ │ └── index.ts │ │ │ │ ├── one-time-token │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── one-time-token.test.ts │ │ │ │ │ └── utils.ts │ │ │ │ ├── open-api │ │ │ │ │ ├── generator.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── logo.ts │ │ │ │ │ └── open-api.test.ts │ │ │ │ ├── organization │ │ │ │ │ ├── access │ │ │ │ │ │ ├── index.ts │ │ │ │ │ │ └── statement.ts │ │ │ │ │ ├── adapter.ts │ │ │ │ │ ├── call.ts │ │ │ │ │ ├── client.test.ts │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── error-codes.ts │ │ │ │ │ ├── has-permission.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── organization-hook.test.ts │ │ │ │ │ ├── organization.test.ts │ │ │ │ │ ├── organization.ts │ │ │ │ │ ├── permission.ts │ │ │ │ │ ├── routes │ │ │ │ │ │ ├── crud-access-control.test.ts │ │ │ │ │ │ ├── crud-access-control.ts │ │ │ │ │ │ ├── crud-invites.ts │ │ │ │ │ │ ├── crud-members.test.ts │ │ │ │ │ │ ├── crud-members.ts │ │ │ │ │ │ ├── crud-org.test.ts │ │ │ │ │ │ ├── crud-org.ts │ │ │ │ │ │ └── crud-team.ts │ │ │ │ │ ├── schema.ts │ │ │ │ │ ├── team.test.ts │ │ │ │ │ └── types.ts │ │ │ │ ├── passkey │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ └── passkey.test.ts │ │ │ │ ├── phone-number │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── phone-number-error.ts │ │ │ │ │ └── phone-number.test.ts │ │ │ │ ├── siwe │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── schema.ts │ │ │ │ │ ├── siwe.test.ts │ │ │ │ │ └── types.ts │ │ │ │ ├── two-factor │ │ │ │ │ ├── backup-codes │ │ │ │ │ │ └── index.ts │ │ │ │ │ ├── client.ts │ │ │ │ │ ├── constant.ts │ │ │ │ │ ├── error-code.ts │ │ │ │ │ ├── index.ts │ │ │ │ │ ├── otp │ │ │ │ │ │ └── index.ts │ │ │ │ │ ├── schema.ts │ │ │ │ │ ├── totp │ │ │ │ │ │ └── index.ts │ │ │ │ │ ├── two-factor.test.ts │ │ │ │ │ ├── types.ts │ │ │ │ │ ├── utils.ts │ │ │ │ │ └── verify-two-factor.ts │ │ │ │ └── username │ │ │ │ ├── client.ts │ │ │ │ ├── error-codes.ts │ │ │ │ ├── index.ts │ │ │ │ ├── schema.ts │ │ │ │ └── username.test.ts │ │ │ ├── social-providers │ │ │ │ └── index.ts │ │ │ ├── social.test.ts │ │ │ ├── test-utils │ │ │ │ ├── headers.ts │ │ │ │ ├── index.ts │ │ │ │ ├── state.ts │ │ │ │ └── test-instance.ts │ │ │ ├── types │ │ │ │ ├── adapter.ts │ │ │ │ ├── api.ts │ │ │ │ ├── helper.ts │ │ │ │ ├── index.ts │ │ │ │ ├── models.ts │ │ │ │ ├── plugins.ts │ │ │ │ └── types.test.ts │ │ │ └── utils │ │ │ ├── await-object.ts │ │ │ ├── boolean.ts │ │ │ ├── clone.ts │ │ │ ├── constants.ts │ │ │ ├── date.ts │ │ │ ├── ensure-utc.ts │ │ │ ├── get-request-ip.ts │ │ │ ├── hashing.ts │ │ │ ├── hide-metadata.ts │ │ │ ├── id.ts │ │ │ ├── import-util.ts │ │ │ ├── index.ts │ │ │ ├── is-atom.ts │ │ │ ├── is-promise.ts │ │ │ ├── json.ts │ │ │ ├── merger.ts │ │ │ ├── middleware-response.ts │ │ │ ├── misc.ts │ │ │ ├── password.ts │ │ │ ├── plugin-helper.ts │ │ │ ├── shim.ts │ │ │ ├── time.ts │ │ │ ├── url.ts │ │ │ └── wildcard.ts │ │ ├── tsconfig.json │ │ ├── tsdown.config.ts │ │ └── vitest.config.ts │ ├── cli │ │ ├── CHANGELOG.md │ │ ├── package.json │ │ ├── README.md │ │ ├── src │ │ │ ├── commands │ │ │ │ ├── generate.ts │ │ │ │ ├── info.ts │ │ │ │ ├── init.ts │ │ │ │ ├── login.ts │ │ │ │ ├── mcp.ts │ │ │ │ ├── migrate.ts │ │ │ │ └── secret.ts │ │ │ ├── generators │ │ │ │ ├── auth-config.ts │ │ │ │ ├── drizzle.ts │ │ │ │ ├── index.ts │ │ │ │ ├── kysely.ts │ │ │ │ ├── prisma.ts │ │ │ │ └── types.ts │ │ │ ├── index.ts │ │ │ └── utils │ │ │ ├── add-svelte-kit-env-modules.ts │ │ │ ├── check-package-managers.ts │ │ │ ├── format-ms.ts │ │ │ ├── get-config.ts │ │ │ ├── get-package-info.ts │ │ │ ├── get-tsconfig-info.ts │ │ │ └── install-dependencies.ts │ │ ├── test │ │ │ ├── __snapshots__ │ │ │ │ ├── auth-schema-mysql-enum.txt │ │ │ │ ├── auth-schema-mysql-number-id.txt │ │ │ │ ├── auth-schema-mysql-passkey-number-id.txt │ │ │ │ ├── auth-schema-mysql-passkey.txt │ │ │ │ ├── auth-schema-mysql.txt │ │ │ │ ├── auth-schema-number-id.txt │ │ │ │ ├── auth-schema-pg-enum.txt │ │ │ │ ├── auth-schema-pg-passkey.txt │ │ │ │ ├── auth-schema-sqlite-enum.txt │ │ │ │ ├── auth-schema-sqlite-number-id.txt │ │ │ │ ├── auth-schema-sqlite-passkey-number-id.txt │ │ │ │ ├── auth-schema-sqlite-passkey.txt │ │ │ │ ├── auth-schema-sqlite.txt │ │ │ │ ├── auth-schema.txt │ │ │ │ ├── migrations.sql │ │ │ │ ├── schema-mongodb.prisma │ │ │ │ ├── schema-mysql-custom.prisma │ │ │ │ ├── schema-mysql.prisma │ │ │ │ ├── schema-numberid.prisma │ │ │ │ └── schema.prisma │ │ │ ├── generate-all-db.test.ts │ │ │ ├── generate.test.ts │ │ │ ├── get-config.test.ts │ │ │ ├── info.test.ts │ │ │ └── migrate.test.ts │ │ ├── tsconfig.json │ │ ├── tsconfig.test.json │ │ └── tsdown.config.ts │ ├── core │ │ ├── package.json │ │ ├── src │ │ │ ├── api │ │ │ │ └── index.ts │ │ │ ├── async_hooks │ │ │ │ └── index.ts │ │ │ ├── context │ │ │ │ ├── endpoint-context.ts │ │ │ │ ├── index.ts │ │ │ │ └── transaction.ts │ │ │ ├── db │ │ │ │ ├── adapter │ │ │ │ │ └── index.ts │ │ │ │ ├── index.ts │ │ │ │ ├── plugin.ts │ │ │ │ ├── schema │ │ │ │ │ ├── account.ts │ │ │ │ │ ├── rate-limit.ts │ │ │ │ │ ├── session.ts │ │ │ │ │ ├── shared.ts │ │ │ │ │ ├── user.ts │ │ │ │ │ └── verification.ts │ │ │ │ └── type.ts │ │ │ ├── env │ │ │ │ ├── color-depth.ts │ │ │ │ ├── env-impl.ts │ │ │ │ ├── index.ts │ │ │ │ ├── logger.test.ts │ │ │ │ └── logger.ts │ │ │ ├── error │ │ │ │ ├── codes.ts │ │ │ │ └── index.ts │ │ │ ├── index.ts │ │ │ ├── oauth2 │ │ │ │ ├── client-credentials-token.ts │ │ │ │ ├── create-authorization-url.ts │ │ │ │ ├── index.ts │ │ │ │ ├── oauth-provider.ts │ │ │ │ ├── refresh-access-token.ts │ │ │ │ ├── utils.ts │ │ │ │ └── validate-authorization-code.ts │ │ │ ├── social-providers │ │ │ │ ├── apple.ts │ │ │ │ ├── atlassian.ts │ │ │ │ ├── cognito.ts │ │ │ │ ├── discord.ts │ │ │ │ ├── dropbox.ts │ │ │ │ ├── facebook.ts │ │ │ │ ├── figma.ts │ │ │ │ ├── github.ts │ │ │ │ ├── gitlab.ts │ │ │ │ ├── google.ts │ │ │ │ ├── huggingface.ts │ │ │ │ ├── index.ts │ │ │ │ ├── kakao.ts │ │ │ │ ├── kick.ts │ │ │ │ ├── line.ts │ │ │ │ ├── linear.ts │ │ │ │ ├── linkedin.ts │ │ │ │ ├── microsoft-entra-id.ts │ │ │ │ ├── naver.ts │ │ │ │ ├── notion.ts │ │ │ │ ├── paypal.ts │ │ │ │ ├── reddit.ts │ │ │ │ ├── roblox.ts │ │ │ │ ├── salesforce.ts │ │ │ │ ├── slack.ts │ │ │ │ ├── spotify.ts │ │ │ │ ├── tiktok.ts │ │ │ │ ├── twitch.ts │ │ │ │ ├── twitter.ts │ │ │ │ ├── vk.ts │ │ │ │ └── zoom.ts │ │ │ ├── types │ │ │ │ ├── context.ts │ │ │ │ ├── cookie.ts │ │ │ │ ├── helper.ts │ │ │ │ ├── index.ts │ │ │ │ ├── init-options.ts │ │ │ │ ├── plugin-client.ts │ │ │ │ └── plugin.ts │ │ │ └── utils │ │ │ ├── error-codes.ts │ │ │ └── index.ts │ │ ├── tsconfig.json │ │ └── tsdown.config.ts │ ├── expo │ │ ├── CHANGELOG.md │ │ ├── package.json │ │ ├── README.md │ │ ├── src │ │ │ ├── client.ts │ │ │ ├── expo.test.ts │ │ │ └── index.ts │ │ ├── tsconfig.json │ │ └── tsdown.config.ts │ ├── sso │ │ ├── package.json │ │ ├── src │ │ │ ├── client.ts │ │ │ ├── index.ts │ │ │ ├── oidc.test.ts │ │ │ └── saml.test.ts │ │ ├── tsconfig.json │ │ └── tsdown.config.ts │ ├── stripe │ │ ├── CHANGELOG.md │ │ ├── package.json │ │ ├── src │ │ │ ├── client.ts │ │ │ ├── hooks.ts │ │ │ ├── index.ts │ │ │ ├── schema.ts │ │ │ ├── stripe.test.ts │ │ │ ├── types.ts │ │ │ └── utils.ts │ │ ├── tsconfig.json │ │ ├── tsdown.config.ts │ │ └── vitest.config.ts │ └── telemetry │ ├── package.json │ ├── src │ │ ├── detectors │ │ │ ├── detect-auth-config.ts │ │ │ ├── detect-database.ts │ │ │ ├── detect-framework.ts │ │ │ ├── detect-project-info.ts │ │ │ ├── detect-runtime.ts │ │ │ └── detect-system-info.ts │ │ ├── index.ts │ │ ├── project-id.ts │ │ ├── telemetry.test.ts │ │ ├── types.ts │ │ └── utils │ │ ├── hash.ts │ │ ├── id.ts │ │ ├── import-util.ts │ │ └── package-json.ts │ ├── tsconfig.json │ └── tsdown.config.ts ├── pnpm-lock.yaml ├── pnpm-workspace.yaml ├── README.md ├── SECURITY.md ├── tsconfig.json └── turbo.json ``` # Files -------------------------------------------------------------------------------- /packages/better-auth/src/plugins/organization/routes/crud-access-control.ts: -------------------------------------------------------------------------------- ```typescript 1 | import * as z from "zod"; 2 | import { APIError } from "../../../api"; 3 | import { createAuthEndpoint } from "@better-auth/core/api"; 4 | import type { OrganizationOptions } from "../types"; 5 | import { orgSessionMiddleware } from "../call"; 6 | import { hasPermission } from "../has-permission"; 7 | import type { Member, OrganizationRole } from "../schema"; 8 | import type { User } from "../../../types"; 9 | import type { Where } from "@better-auth/core/db/adapter"; 10 | import type { AccessControl } from "../../access"; 11 | import { 12 | toZodSchema, 13 | type InferAdditionalFieldsFromPluginOptions, 14 | } from "../../../db"; 15 | import { ORGANIZATION_ERROR_CODES } from "../error-codes"; 16 | import type { GenericEndpointContext } from "@better-auth/core"; 17 | 18 | type IsExactlyEmptyObject<T> = keyof T extends never // no keys 19 | ? T extends {} // is assignable to {} 20 | ? {} extends T 21 | ? true 22 | : false // and {} is assignable to it 23 | : false 24 | : false; 25 | 26 | const normalizeRoleName = (role: string) => role.toLowerCase(); 27 | const DEFAULT_MAXIMUM_ROLES_PER_ORGANIZATION = Number.POSITIVE_INFINITY; 28 | 29 | const getAdditionalFields = < 30 | O extends OrganizationOptions, 31 | AllPartial extends boolean = false, 32 | >( 33 | options: O, 34 | shouldBePartial: AllPartial = false as AllPartial, 35 | ) => { 36 | let additionalFields = 37 | options?.schema?.organizationRole?.additionalFields || {}; 38 | if (shouldBePartial) { 39 | for (const key in additionalFields) { 40 | additionalFields[key]!.required = false; 41 | } 42 | } 43 | const additionalFieldsSchema = toZodSchema({ 44 | fields: additionalFields, 45 | isClientSide: true, 46 | }); 47 | type AdditionalFields = AllPartial extends true 48 | ? Partial<InferAdditionalFieldsFromPluginOptions<"organizationRole", O>> 49 | : InferAdditionalFieldsFromPluginOptions<"organizationRole", O>; 50 | type ReturnAdditionalFields = InferAdditionalFieldsFromPluginOptions< 51 | "organizationRole", 52 | O, 53 | false 54 | >; 55 | 56 | return { 57 | additionalFieldsSchema, 58 | $AdditionalFields: {} as AdditionalFields, 59 | $ReturnAdditionalFields: {} as ReturnAdditionalFields, 60 | }; 61 | }; 62 | 63 | export const createOrgRole = <O extends OrganizationOptions>(options: O) => { 64 | const { additionalFieldsSchema, $AdditionalFields, $ReturnAdditionalFields } = 65 | getAdditionalFields<O>(options, false); 66 | type AdditionalFields = typeof $AdditionalFields; 67 | type ReturnAdditionalFields = typeof $ReturnAdditionalFields; 68 | 69 | return createAuthEndpoint( 70 | "/organization/create-role", 71 | { 72 | method: "POST", 73 | body: z.object({ 74 | organizationId: z.string().optional().meta({ 75 | description: 76 | "The id of the organization to create the role in. If not provided, the user's active organization will be used.", 77 | }), 78 | role: z.string().meta({ 79 | description: "The name of the role to create", 80 | }), 81 | permission: z.record(z.string(), z.array(z.string())).meta({ 82 | description: "The permission to assign to the role", 83 | }), 84 | additionalFields: z 85 | .object({ ...additionalFieldsSchema.shape }) 86 | .optional(), 87 | }), 88 | metadata: { 89 | $Infer: { 90 | body: {} as { 91 | organizationId?: string | undefined; 92 | role: string; 93 | permission: Record<string, string[]>; 94 | } & (IsExactlyEmptyObject<AdditionalFields> extends true 95 | ? { additionalFields?: {} } 96 | : { additionalFields: AdditionalFields }), 97 | }, 98 | }, 99 | requireHeaders: true, 100 | use: [orgSessionMiddleware], 101 | }, 102 | async (ctx) => { 103 | const { session, user } = ctx.context.session; 104 | let roleName = ctx.body.role; 105 | const permission = ctx.body.permission; 106 | const additionalFields = ctx.body.additionalFields; 107 | 108 | const ac = options.ac; 109 | if (!ac) { 110 | ctx.context.logger.error( 111 | `[Dynamic Access Control] The organization plugin is missing a pre-defined ac instance.`, 112 | `\nPlease refer to the documentation here: https://better-auth.com/docs/plugins/organization#dynamic-access-control`, 113 | ); 114 | throw new APIError("NOT_IMPLEMENTED", { 115 | message: ORGANIZATION_ERROR_CODES.MISSING_AC_INSTANCE, 116 | }); 117 | } 118 | 119 | // Get the organization id where the role will be created. 120 | // We can verify if the org id is valid and associated with the user in the next step when we try to find the member. 121 | const organizationId = 122 | ctx.body.organizationId ?? session.activeOrganizationId; 123 | if (!organizationId) { 124 | ctx.context.logger.error( 125 | `[Dynamic Access Control] The session is missing an active organization id to create a role. Either set an active org id, or pass an organizationId in the request body.`, 126 | ); 127 | throw new APIError("BAD_REQUEST", { 128 | message: 129 | ORGANIZATION_ERROR_CODES.YOU_MUST_BE_IN_AN_ORGANIZATION_TO_CREATE_A_ROLE, 130 | }); 131 | } 132 | 133 | roleName = normalizeRoleName(roleName); 134 | 135 | await checkIfRoleNameIsTakenByPreDefinedRole({ 136 | role: roleName, 137 | organizationId, 138 | options, 139 | ctx, 140 | }); 141 | 142 | // Get the user's role associated with the organization. 143 | // This also serves as a check to ensure the org id is valid. 144 | const member = await ctx.context.adapter.findOne<Member>({ 145 | model: "member", 146 | where: [ 147 | { 148 | field: "organizationId", 149 | value: organizationId, 150 | operator: "eq", 151 | connector: "AND", 152 | }, 153 | { 154 | field: "userId", 155 | value: user.id, 156 | operator: "eq", 157 | connector: "AND", 158 | }, 159 | ], 160 | }); 161 | if (!member) { 162 | ctx.context.logger.error( 163 | `[Dynamic Access Control] The user is not a member of the organization to create a role.`, 164 | { 165 | userId: user.id, 166 | organizationId, 167 | }, 168 | ); 169 | throw new APIError("FORBIDDEN", { 170 | message: 171 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_A_MEMBER_OF_THIS_ORGANIZATION, 172 | }); 173 | } 174 | 175 | const canCreateRole = await hasPermission( 176 | { 177 | options, 178 | organizationId, 179 | permissions: { 180 | ac: ["create"], 181 | }, 182 | role: member.role, 183 | }, 184 | ctx, 185 | ); 186 | if (!canCreateRole) { 187 | ctx.context.logger.error( 188 | `[Dynamic Access Control] The user is not permitted to create a role. If this is unexpected, please make sure the role associated to that member has the "ac" resource with the "create" permission.`, 189 | { 190 | userId: user.id, 191 | organizationId, 192 | role: member.role, 193 | }, 194 | ); 195 | throw new APIError("FORBIDDEN", { 196 | message: 197 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_ALLOWED_TO_CREATE_A_ROLE, 198 | }); 199 | } 200 | 201 | const maximumRolesPerOrganization = 202 | typeof options.dynamicAccessControl?.maximumRolesPerOrganization === 203 | "function" 204 | ? await options.dynamicAccessControl.maximumRolesPerOrganization( 205 | organizationId, 206 | ) 207 | : (options.dynamicAccessControl?.maximumRolesPerOrganization ?? 208 | DEFAULT_MAXIMUM_ROLES_PER_ORGANIZATION); 209 | const rolesInDB = await ctx.context.adapter.count({ 210 | model: "organizationRole", 211 | where: [ 212 | { 213 | field: "organizationId", 214 | value: organizationId, 215 | operator: "eq", 216 | connector: "AND", 217 | }, 218 | ], 219 | }); 220 | if (rolesInDB >= maximumRolesPerOrganization) { 221 | ctx.context.logger.error( 222 | `[Dynamic Access Control] Failed to create a new role, the organization has too many roles. Maximum allowed roles is ${maximumRolesPerOrganization}.`, 223 | { 224 | organizationId, 225 | maximumRolesPerOrganization, 226 | rolesInDB, 227 | }, 228 | ); 229 | throw new APIError("BAD_REQUEST", { 230 | message: ORGANIZATION_ERROR_CODES.TOO_MANY_ROLES, 231 | }); 232 | } 233 | 234 | await checkForInvalidResources({ ac, ctx, permission }); 235 | 236 | await checkIfMemberHasPermission({ 237 | ctx, 238 | member, 239 | options, 240 | organizationId, 241 | permissionRequired: permission, 242 | user, 243 | action: "create", 244 | }); 245 | 246 | await checkIfRoleNameIsTakenByRoleInDB({ 247 | ctx, 248 | organizationId, 249 | role: roleName, 250 | }); 251 | 252 | const newRole = ac.newRole(permission); 253 | 254 | const newRoleInDB = await ctx.context.adapter.create< 255 | Omit<OrganizationRole, "permission"> & { permission: string } 256 | >({ 257 | model: "organizationRole", 258 | data: { 259 | createdAt: new Date(), 260 | organizationId, 261 | permission: JSON.stringify(permission), 262 | role: roleName, 263 | ...additionalFields, 264 | }, 265 | }); 266 | 267 | const data = { 268 | ...newRoleInDB, 269 | permission, 270 | } as OrganizationRole & ReturnAdditionalFields; 271 | return ctx.json({ 272 | success: true, 273 | roleData: data, 274 | statements: newRole.statements, 275 | }); 276 | }, 277 | ); 278 | }; 279 | 280 | export const deleteOrgRole = <O extends OrganizationOptions>(options: O) => { 281 | return createAuthEndpoint( 282 | "/organization/delete-role", 283 | { 284 | method: "POST", 285 | body: z 286 | .object({ 287 | organizationId: z.string().optional().meta({ 288 | description: 289 | "The id of the organization to create the role in. If not provided, the user's active organization will be used.", 290 | }), 291 | }) 292 | .and( 293 | z.union([ 294 | z.object({ 295 | roleName: z.string().nonempty().meta({ 296 | description: "The name of the role to delete", 297 | }), 298 | }), 299 | z.object({ 300 | roleId: z.string().nonempty().meta({ 301 | description: "The id of the role to delete", 302 | }), 303 | }), 304 | ]), 305 | ), 306 | requireHeaders: true, 307 | use: [orgSessionMiddleware], 308 | metadata: { 309 | $Infer: { 310 | body: {} as { 311 | roleName?: string | undefined; 312 | roleId?: string | undefined; 313 | organizationId?: string | undefined; 314 | }, 315 | }, 316 | }, 317 | }, 318 | async (ctx) => { 319 | const { session, user } = ctx.context.session; 320 | 321 | // We can verify if the org id is valid and associated with the user in the next step when we try to find the member. 322 | const organizationId = 323 | ctx.body.organizationId ?? session.activeOrganizationId; 324 | if (!organizationId) { 325 | ctx.context.logger.error( 326 | `[Dynamic Access Control] The session is missing an active organization id to delete a role. Either set an active org id, or pass an organizationId in the request body.`, 327 | ); 328 | throw new APIError("BAD_REQUEST", { 329 | message: ORGANIZATION_ERROR_CODES.NO_ACTIVE_ORGANIZATION, 330 | }); 331 | } 332 | 333 | // Get the user's role associated with the organization. 334 | // This also serves as a check to ensure the org id is valid. 335 | const member = await ctx.context.adapter.findOne<Member>({ 336 | model: "member", 337 | where: [ 338 | { 339 | field: "organizationId", 340 | value: organizationId, 341 | operator: "eq", 342 | connector: "AND", 343 | }, 344 | { 345 | field: "userId", 346 | value: user.id, 347 | operator: "eq", 348 | connector: "AND", 349 | }, 350 | ], 351 | }); 352 | if (!member) { 353 | ctx.context.logger.error( 354 | `[Dynamic Access Control] The user is not a member of the organization to delete a role.`, 355 | { 356 | userId: user.id, 357 | organizationId, 358 | }, 359 | ); 360 | throw new APIError("FORBIDDEN", { 361 | message: 362 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_A_MEMBER_OF_THIS_ORGANIZATION, 363 | }); 364 | } 365 | 366 | const canDeleteRole = await hasPermission( 367 | { 368 | options, 369 | organizationId, 370 | permissions: { 371 | ac: ["delete"], 372 | }, 373 | role: member.role, 374 | }, 375 | ctx, 376 | ); 377 | if (!canDeleteRole) { 378 | ctx.context.logger.error( 379 | `[Dynamic Access Control] The user is not permitted to delete a role. If this is unexpected, please make sure the role associated to that member has the "ac" resource with the "delete" permission.`, 380 | { 381 | userId: user.id, 382 | organizationId, 383 | role: member.role, 384 | }, 385 | ); 386 | throw new APIError("FORBIDDEN", { 387 | message: 388 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_ALLOWED_TO_DELETE_A_ROLE, 389 | }); 390 | } 391 | 392 | if (ctx.body.roleName) { 393 | const roleName = ctx.body.roleName; 394 | const defaultRoles = options.roles 395 | ? Object.keys(options.roles) 396 | : ["owner", "admin", "member"]; 397 | if (defaultRoles.includes(roleName)) { 398 | ctx.context.logger.error( 399 | `[Dynamic Access Control] Cannot delete a pre-defined role.`, 400 | { 401 | roleName, 402 | organizationId, 403 | defaultRoles, 404 | }, 405 | ); 406 | throw new APIError("BAD_REQUEST", { 407 | message: ORGANIZATION_ERROR_CODES.CANNOT_DELETE_A_PRE_DEFINED_ROLE, 408 | }); 409 | } 410 | } 411 | 412 | let condition: Where; 413 | if (ctx.body.roleName) { 414 | condition = { 415 | field: "role", 416 | value: ctx.body.roleName, 417 | operator: "eq", 418 | connector: "AND", 419 | }; 420 | } else if (ctx.body.roleId) { 421 | condition = { 422 | field: "id", 423 | value: ctx.body.roleId, 424 | operator: "eq", 425 | connector: "AND", 426 | }; 427 | } else { 428 | // shouldn't be able to reach here given the schema validation. 429 | // But just in case, throw an error. 430 | ctx.context.logger.error( 431 | `[Dynamic Access Control] The role name/id is not provided in the request body.`, 432 | ); 433 | throw new APIError("BAD_REQUEST", { 434 | message: ORGANIZATION_ERROR_CODES.ROLE_NOT_FOUND, 435 | }); 436 | } 437 | const existingRoleInDB = 438 | await ctx.context.adapter.findOne<OrganizationRole>({ 439 | model: "organizationRole", 440 | where: [ 441 | { 442 | field: "organizationId", 443 | value: organizationId, 444 | operator: "eq", 445 | connector: "AND", 446 | }, 447 | condition, 448 | ], 449 | }); 450 | if (!existingRoleInDB) { 451 | ctx.context.logger.error( 452 | `[Dynamic Access Control] The role name/id does not exist in the database.`, 453 | { 454 | ...("roleName" in ctx.body 455 | ? { roleName: ctx.body.roleName } 456 | : { roleId: ctx.body.roleId }), 457 | organizationId, 458 | }, 459 | ); 460 | throw new APIError("BAD_REQUEST", { 461 | message: ORGANIZATION_ERROR_CODES.ROLE_NOT_FOUND, 462 | }); 463 | } 464 | 465 | existingRoleInDB.permission = JSON.parse( 466 | existingRoleInDB.permission as never as string, 467 | ); 468 | 469 | await ctx.context.adapter.delete({ 470 | model: "organizationRole", 471 | where: [ 472 | { 473 | field: "organizationId", 474 | value: organizationId, 475 | operator: "eq", 476 | connector: "AND", 477 | }, 478 | condition, 479 | ], 480 | }); 481 | 482 | return ctx.json({ 483 | success: true, 484 | }); 485 | }, 486 | ); 487 | }; 488 | 489 | export const listOrgRoles = <O extends OrganizationOptions>(options: O) => { 490 | const { $ReturnAdditionalFields } = getAdditionalFields<O>(options, false); 491 | type ReturnAdditionalFields = typeof $ReturnAdditionalFields; 492 | 493 | return createAuthEndpoint( 494 | "/organization/list-roles", 495 | { 496 | method: "GET", 497 | use: [orgSessionMiddleware], 498 | query: z 499 | .object({ 500 | organizationId: z.string().optional().meta({ 501 | description: 502 | "The id of the organization to list roles for. If not provided, the user's active organization will be used.", 503 | }), 504 | }) 505 | .optional(), 506 | }, 507 | async (ctx) => { 508 | const { session, user } = ctx.context.session; 509 | 510 | const organizationId = 511 | ctx.query?.organizationId ?? session.activeOrganizationId; 512 | if (!organizationId) { 513 | ctx.context.logger.error( 514 | `[Dynamic Access Control] The session is missing an active organization id to list roles. Either set an active org id, or pass an organizationId in the request query.`, 515 | ); 516 | throw new APIError("BAD_REQUEST", { 517 | message: ORGANIZATION_ERROR_CODES.NO_ACTIVE_ORGANIZATION, 518 | }); 519 | } 520 | 521 | const member = await ctx.context.adapter.findOne<Member>({ 522 | model: "member", 523 | where: [ 524 | { 525 | field: "organizationId", 526 | value: organizationId, 527 | operator: "eq", 528 | connector: "AND", 529 | }, 530 | { 531 | field: "userId", 532 | value: user.id, 533 | operator: "eq", 534 | connector: "AND", 535 | }, 536 | ], 537 | }); 538 | if (!member) { 539 | ctx.context.logger.error( 540 | `[Dynamic Access Control] The user is not a member of the organization to list roles.`, 541 | { 542 | userId: user.id, 543 | organizationId, 544 | }, 545 | ); 546 | throw new APIError("FORBIDDEN", { 547 | message: 548 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_A_MEMBER_OF_THIS_ORGANIZATION, 549 | }); 550 | } 551 | 552 | const canListRoles = await hasPermission( 553 | { 554 | options, 555 | organizationId, 556 | permissions: { 557 | ac: ["read"], 558 | }, 559 | role: member.role, 560 | }, 561 | ctx, 562 | ); 563 | if (!canListRoles) { 564 | ctx.context.logger.error( 565 | `[Dynamic Access Control] The user is not permitted to list roles.`, 566 | { 567 | userId: user.id, 568 | organizationId, 569 | role: member.role, 570 | }, 571 | ); 572 | throw new APIError("FORBIDDEN", { 573 | message: ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_ALLOWED_TO_LIST_A_ROLE, 574 | }); 575 | } 576 | 577 | let roles = await ctx.context.adapter.findMany< 578 | OrganizationRole & ReturnAdditionalFields 579 | >({ 580 | model: "organizationRole", 581 | where: [ 582 | { 583 | field: "organizationId", 584 | value: organizationId, 585 | operator: "eq", 586 | connector: "AND", 587 | }, 588 | ], 589 | }); 590 | 591 | roles = roles.map((x) => ({ 592 | ...x, 593 | permission: JSON.parse(x.permission as never as string), 594 | })); 595 | 596 | return ctx.json(roles); 597 | }, 598 | ); 599 | }; 600 | 601 | export const getOrgRole = <O extends OrganizationOptions>(options: O) => { 602 | const { $ReturnAdditionalFields } = getAdditionalFields<O>(options, false); 603 | type ReturnAdditionalFields = typeof $ReturnAdditionalFields; 604 | return createAuthEndpoint( 605 | "/organization/get-role", 606 | { 607 | method: "GET", 608 | use: [orgSessionMiddleware], 609 | query: z 610 | .object({ 611 | organizationId: z.string().optional().meta({ 612 | description: 613 | "The id of the organization to read a role for. If not provided, the user's active organization will be used.", 614 | }), 615 | }) 616 | .and( 617 | z.union([ 618 | z.object({ 619 | roleName: z.string().nonempty().meta({ 620 | description: "The name of the role to read", 621 | }), 622 | }), 623 | z.object({ 624 | roleId: z.string().nonempty().meta({ 625 | description: "The id of the role to read", 626 | }), 627 | }), 628 | ]), 629 | ) 630 | .optional(), 631 | metadata: { 632 | $Infer: { 633 | query: {} as { 634 | organizationId?: string | undefined; 635 | roleName?: string | undefined; 636 | roleId?: string | undefined; 637 | }, 638 | }, 639 | }, 640 | }, 641 | async (ctx) => { 642 | const { session, user } = ctx.context.session; 643 | 644 | const organizationId = 645 | ctx.query?.organizationId ?? session.activeOrganizationId; 646 | if (!organizationId) { 647 | ctx.context.logger.error( 648 | `[Dynamic Access Control] The session is missing an active organization id to read a role. Either set an active org id, or pass an organizationId in the request query.`, 649 | ); 650 | throw new APIError("BAD_REQUEST", { 651 | message: ORGANIZATION_ERROR_CODES.NO_ACTIVE_ORGANIZATION, 652 | }); 653 | } 654 | 655 | const member = await ctx.context.adapter.findOne<Member>({ 656 | model: "member", 657 | where: [ 658 | { 659 | field: "organizationId", 660 | value: organizationId, 661 | operator: "eq", 662 | connector: "AND", 663 | }, 664 | { 665 | field: "userId", 666 | value: user.id, 667 | operator: "eq", 668 | connector: "AND", 669 | }, 670 | ], 671 | }); 672 | if (!member) { 673 | ctx.context.logger.error( 674 | `[Dynamic Access Control] The user is not a member of the organization to read a role.`, 675 | { 676 | userId: user.id, 677 | organizationId, 678 | }, 679 | ); 680 | throw new APIError("FORBIDDEN", { 681 | message: 682 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_A_MEMBER_OF_THIS_ORGANIZATION, 683 | }); 684 | } 685 | 686 | const canListRoles = await hasPermission( 687 | { 688 | options, 689 | organizationId, 690 | permissions: { 691 | ac: ["read"], 692 | }, 693 | role: member.role, 694 | }, 695 | ctx, 696 | ); 697 | if (!canListRoles) { 698 | ctx.context.logger.error( 699 | `[Dynamic Access Control] The user is not permitted to read a role.`, 700 | { 701 | userId: user.id, 702 | organizationId, 703 | role: member.role, 704 | }, 705 | ); 706 | throw new APIError("FORBIDDEN", { 707 | message: ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_ALLOWED_TO_READ_A_ROLE, 708 | }); 709 | } 710 | 711 | let condition: Where; 712 | if (ctx.query.roleName) { 713 | condition = { 714 | field: "role", 715 | value: ctx.query.roleName, 716 | operator: "eq", 717 | connector: "AND", 718 | }; 719 | } else if (ctx.query.roleId) { 720 | condition = { 721 | field: "id", 722 | value: ctx.query.roleId, 723 | operator: "eq", 724 | connector: "AND", 725 | }; 726 | } else { 727 | // shouldn't be able to reach here given the schema validation. 728 | // But just in case, throw an error. 729 | ctx.context.logger.error( 730 | `[Dynamic Access Control] The role name/id is not provided in the request query.`, 731 | ); 732 | throw new APIError("BAD_REQUEST", { 733 | message: ORGANIZATION_ERROR_CODES.ROLE_NOT_FOUND, 734 | }); 735 | } 736 | let role = await ctx.context.adapter.findOne<OrganizationRole>({ 737 | model: "organizationRole", 738 | where: [ 739 | { 740 | field: "organizationId", 741 | value: organizationId, 742 | operator: "eq", 743 | connector: "AND", 744 | }, 745 | condition, 746 | ], 747 | }); 748 | if (!role) { 749 | ctx.context.logger.error( 750 | `[Dynamic Access Control] The role name/id does not exist in the database.`, 751 | { 752 | ...("roleName" in ctx.query 753 | ? { roleName: ctx.query.roleName } 754 | : { roleId: ctx.query.roleId }), 755 | organizationId, 756 | }, 757 | ); 758 | throw new APIError("BAD_REQUEST", { 759 | message: ORGANIZATION_ERROR_CODES.ROLE_NOT_FOUND, 760 | }); 761 | } 762 | 763 | role.permission = JSON.parse(role.permission as never as string); 764 | 765 | return ctx.json(role as OrganizationRole & ReturnAdditionalFields); 766 | }, 767 | ); 768 | }; 769 | 770 | export const updateOrgRole = <O extends OrganizationOptions>(options: O) => { 771 | const { additionalFieldsSchema, $AdditionalFields, $ReturnAdditionalFields } = 772 | getAdditionalFields<O, true>(options, true); 773 | type AdditionalFields = typeof $AdditionalFields; 774 | type ReturnAdditionalFields = typeof $ReturnAdditionalFields; 775 | 776 | return createAuthEndpoint( 777 | "/organization/update-role", 778 | { 779 | method: "POST", 780 | body: z 781 | .object({ 782 | organizationId: z.string().optional().meta({ 783 | description: 784 | "The id of the organization to update the role in. If not provided, the user's active organization will be used.", 785 | }), 786 | data: z.object({ 787 | permission: z 788 | .record(z.string(), z.array(z.string())) 789 | .optional() 790 | .meta({ 791 | description: "The permission to update the role with", 792 | }), 793 | roleName: z.string().optional().meta({ 794 | description: "The name of the role to update", 795 | }), 796 | ...additionalFieldsSchema.shape, 797 | }), 798 | }) 799 | .and( 800 | z.union([ 801 | z.object({ 802 | roleName: z.string().nonempty().meta({ 803 | description: "The name of the role to update", 804 | }), 805 | }), 806 | z.object({ 807 | roleId: z.string().nonempty().meta({ 808 | description: "The id of the role to update", 809 | }), 810 | }), 811 | ]), 812 | ), 813 | metadata: { 814 | $Infer: { 815 | body: {} as { 816 | organizationId?: string | undefined; 817 | data: { 818 | permission?: Record<string, string[]> | undefined; 819 | roleName?: string | undefined; 820 | } & AdditionalFields; 821 | roleName?: string | undefined; 822 | roleId?: string | undefined; 823 | }, 824 | }, 825 | }, 826 | use: [orgSessionMiddleware], 827 | }, 828 | async (ctx) => { 829 | const { session, user } = ctx.context.session; 830 | 831 | const ac = options.ac; 832 | if (!ac) { 833 | ctx.context.logger.error( 834 | `[Dynamic Access Control] The organization plugin is missing a pre-defined ac instance.`, 835 | `\nPlease refer to the documentation here: https://better-auth.com/docs/plugins/organization#dynamic-access-control`, 836 | ); 837 | throw new APIError("NOT_IMPLEMENTED", { 838 | message: ORGANIZATION_ERROR_CODES.MISSING_AC_INSTANCE, 839 | }); 840 | } 841 | 842 | const organizationId = 843 | ctx.body.organizationId ?? session.activeOrganizationId; 844 | if (!organizationId) { 845 | ctx.context.logger.error( 846 | `[Dynamic Access Control] The session is missing an active organization id to update a role. Either set an active org id, or pass an organizationId in the request body.`, 847 | ); 848 | throw new APIError("BAD_REQUEST", { 849 | message: ORGANIZATION_ERROR_CODES.NO_ACTIVE_ORGANIZATION, 850 | }); 851 | } 852 | 853 | const member = await ctx.context.adapter.findOne<Member>({ 854 | model: "member", 855 | where: [ 856 | { 857 | field: "organizationId", 858 | value: organizationId, 859 | operator: "eq", 860 | connector: "AND", 861 | }, 862 | { 863 | field: "userId", 864 | value: user.id, 865 | operator: "eq", 866 | connector: "AND", 867 | }, 868 | ], 869 | }); 870 | if (!member) { 871 | ctx.context.logger.error( 872 | `[Dynamic Access Control] The user is not a member of the organization to update a role.`, 873 | { 874 | userId: user.id, 875 | organizationId, 876 | }, 877 | ); 878 | throw new APIError("FORBIDDEN", { 879 | message: 880 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_A_MEMBER_OF_THIS_ORGANIZATION, 881 | }); 882 | } 883 | 884 | const canUpdateRole = await hasPermission( 885 | { 886 | options, 887 | organizationId, 888 | role: member.role, 889 | permissions: { 890 | ac: ["update"], 891 | }, 892 | }, 893 | ctx, 894 | ); 895 | if (!canUpdateRole) { 896 | ctx.context.logger.error( 897 | `[Dynamic Access Control] The user is not permitted to update a role.`, 898 | ); 899 | throw new APIError("FORBIDDEN", { 900 | message: 901 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_ALLOWED_TO_UPDATE_A_ROLE, 902 | }); 903 | } 904 | 905 | let condition: Where; 906 | if (ctx.body.roleName) { 907 | condition = { 908 | field: "role", 909 | value: ctx.body.roleName, 910 | operator: "eq", 911 | connector: "AND", 912 | }; 913 | } else if (ctx.body.roleId) { 914 | condition = { 915 | field: "id", 916 | value: ctx.body.roleId, 917 | operator: "eq", 918 | connector: "AND", 919 | }; 920 | } else { 921 | // shouldn't be able to reach here given the schema validation. 922 | // But just in case, throw an error. 923 | ctx.context.logger.error( 924 | `[Dynamic Access Control] The role name/id is not provided in the request body.`, 925 | ); 926 | throw new APIError("BAD_REQUEST", { 927 | message: ORGANIZATION_ERROR_CODES.ROLE_NOT_FOUND, 928 | }); 929 | } 930 | let role = await ctx.context.adapter.findOne<OrganizationRole>({ 931 | model: "organizationRole", 932 | where: [ 933 | { 934 | field: "organizationId", 935 | value: organizationId, 936 | operator: "eq", 937 | connector: "AND", 938 | }, 939 | condition, 940 | ], 941 | }); 942 | if (!role) { 943 | ctx.context.logger.error( 944 | `[Dynamic Access Control] The role name/id does not exist in the database.`, 945 | { 946 | ...("roleName" in ctx.body 947 | ? { roleName: ctx.body.roleName } 948 | : { roleId: ctx.body.roleId }), 949 | organizationId, 950 | }, 951 | ); 952 | throw new APIError("BAD_REQUEST", { 953 | message: ORGANIZATION_ERROR_CODES.ROLE_NOT_FOUND, 954 | }); 955 | } 956 | role.permission = role.permission 957 | ? JSON.parse(role.permission as never as string) 958 | : undefined; 959 | 960 | const { 961 | permission: _, 962 | roleName: __, 963 | ...additionalFields 964 | } = ctx.body.data; 965 | 966 | let updateData: Partial<OrganizationRole> = { 967 | ...additionalFields, 968 | }; 969 | 970 | if (ctx.body.data.permission) { 971 | let newPermission = ctx.body.data.permission; 972 | 973 | await checkForInvalidResources({ ac, ctx, permission: newPermission }); 974 | 975 | await checkIfMemberHasPermission({ 976 | ctx, 977 | member, 978 | options, 979 | organizationId, 980 | permissionRequired: newPermission, 981 | user, 982 | action: "update", 983 | }); 984 | 985 | updateData.permission = newPermission; 986 | } 987 | if (ctx.body.data.roleName) { 988 | let newRoleName = ctx.body.data.roleName; 989 | 990 | newRoleName = normalizeRoleName(newRoleName); 991 | 992 | await checkIfRoleNameIsTakenByPreDefinedRole({ 993 | role: newRoleName, 994 | organizationId, 995 | options, 996 | ctx, 997 | }); 998 | await checkIfRoleNameIsTakenByRoleInDB({ 999 | role: newRoleName, 1000 | organizationId, 1001 | ctx, 1002 | }); 1003 | 1004 | updateData.role = newRoleName; 1005 | } 1006 | 1007 | // ----- 1008 | // Apply the updates 1009 | const update = { 1010 | ...updateData, 1011 | ...(updateData.permission 1012 | ? { permission: JSON.stringify(updateData.permission) } 1013 | : {}), 1014 | }; 1015 | await ctx.context.adapter.update<OrganizationRole>({ 1016 | model: "organizationRole", 1017 | where: [ 1018 | { 1019 | field: "organizationId", 1020 | value: organizationId, 1021 | operator: "eq", 1022 | connector: "AND", 1023 | }, 1024 | condition, 1025 | ], 1026 | update, 1027 | }); 1028 | 1029 | // ----- 1030 | // Return the updated role 1031 | return ctx.json({ 1032 | success: true, 1033 | roleData: { 1034 | ...role, 1035 | ...update, 1036 | permission: updateData.permission || role.permission || null, 1037 | } as OrganizationRole & ReturnAdditionalFields, 1038 | }); 1039 | }, 1040 | ); 1041 | }; 1042 | 1043 | async function checkForInvalidResources({ 1044 | ac, 1045 | ctx, 1046 | permission, 1047 | }: { 1048 | ac: AccessControl; 1049 | ctx: GenericEndpointContext; 1050 | permission: Record<string, string[]>; 1051 | }) { 1052 | const validResources = Object.keys(ac.statements); 1053 | const providedResources = Object.keys(permission); 1054 | const hasInvalidResource = providedResources.some( 1055 | (r) => !validResources.includes(r), 1056 | ); 1057 | if (hasInvalidResource) { 1058 | ctx.context.logger.error( 1059 | `[Dynamic Access Control] The provided permission includes an invalid resource.`, 1060 | { 1061 | providedResources, 1062 | validResources, 1063 | }, 1064 | ); 1065 | throw new APIError("BAD_REQUEST", { 1066 | message: ORGANIZATION_ERROR_CODES.INVALID_RESOURCE, 1067 | }); 1068 | } 1069 | } 1070 | 1071 | async function checkIfMemberHasPermission({ 1072 | ctx, 1073 | permissionRequired: permission, 1074 | options, 1075 | organizationId, 1076 | member, 1077 | user, 1078 | action, 1079 | }: { 1080 | ctx: GenericEndpointContext; 1081 | permissionRequired: Record<string, string[]>; 1082 | options: OrganizationOptions; 1083 | organizationId: string; 1084 | member: Member; 1085 | user: User; 1086 | action: "create" | "update" | "delete" | "read" | "list" | "get"; 1087 | }) { 1088 | const hasNecessaryPermissions: { 1089 | resource: { [x: string]: string[] }; 1090 | hasPermission: boolean; 1091 | }[] = []; 1092 | const permissionEntries = Object.entries(permission); 1093 | for await (const [resource, permissions] of permissionEntries) { 1094 | for await (const perm of permissions) { 1095 | hasNecessaryPermissions.push({ 1096 | resource: { [resource]: [perm] }, 1097 | hasPermission: await hasPermission( 1098 | { 1099 | options, 1100 | organizationId, 1101 | permissions: { [resource]: [perm] }, 1102 | useMemoryCache: true, 1103 | role: member.role, 1104 | }, 1105 | ctx, 1106 | ), 1107 | }); 1108 | } 1109 | } 1110 | const missingPermissions = hasNecessaryPermissions 1111 | .filter((x) => x.hasPermission === false) 1112 | .map((x) => { 1113 | const key = Object.keys(x.resource)[0]!; 1114 | return `${key}:${x.resource[key]![0]}` as const; 1115 | }); 1116 | if (missingPermissions.length > 0) { 1117 | ctx.context.logger.error( 1118 | `[Dynamic Access Control] The user is missing permissions nessesary to ${action} a role with those set of permissions.\n`, 1119 | { 1120 | userId: user.id, 1121 | organizationId, 1122 | role: member.role, 1123 | missingPermissions, 1124 | }, 1125 | ); 1126 | let errorMessage: string; 1127 | if (action === "create") 1128 | errorMessage = 1129 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_ALLOWED_TO_CREATE_A_ROLE; 1130 | else if (action === "update") 1131 | errorMessage = 1132 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_ALLOWED_TO_UPDATE_A_ROLE; 1133 | else if (action === "delete") 1134 | errorMessage = 1135 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_ALLOWED_TO_DELETE_A_ROLE; 1136 | else if (action === "read") 1137 | errorMessage = 1138 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_ALLOWED_TO_READ_A_ROLE; 1139 | else if (action === "list") 1140 | errorMessage = 1141 | ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_ALLOWED_TO_LIST_A_ROLE; 1142 | else 1143 | errorMessage = ORGANIZATION_ERROR_CODES.YOU_ARE_NOT_ALLOWED_TO_GET_A_ROLE; 1144 | 1145 | throw new APIError("FORBIDDEN", { 1146 | message: errorMessage, 1147 | missingPermissions, 1148 | }); 1149 | } 1150 | } 1151 | 1152 | async function checkIfRoleNameIsTakenByPreDefinedRole({ 1153 | options, 1154 | organizationId, 1155 | role, 1156 | ctx, 1157 | }: { 1158 | options: OrganizationOptions; 1159 | organizationId: string; 1160 | role: string; 1161 | ctx: GenericEndpointContext; 1162 | }) { 1163 | const defaultRoles = options.roles 1164 | ? Object.keys(options.roles) 1165 | : ["owner", "admin", "member"]; 1166 | if (defaultRoles.includes(role)) { 1167 | ctx.context.logger.error( 1168 | `[Dynamic Access Control] The role name "${role}" is already taken by a pre-defined role.`, 1169 | { 1170 | role, 1171 | organizationId, 1172 | defaultRoles, 1173 | }, 1174 | ); 1175 | throw new APIError("BAD_REQUEST", { 1176 | message: ORGANIZATION_ERROR_CODES.ROLE_NAME_IS_ALREADY_TAKEN, 1177 | }); 1178 | } 1179 | } 1180 | 1181 | async function checkIfRoleNameIsTakenByRoleInDB({ 1182 | organizationId, 1183 | role, 1184 | ctx, 1185 | }: { 1186 | ctx: GenericEndpointContext; 1187 | organizationId: string; 1188 | role: string; 1189 | }) { 1190 | const existingRoleInDB = await ctx.context.adapter.findOne<OrganizationRole>({ 1191 | model: "organizationRole", 1192 | where: [ 1193 | { 1194 | field: "organizationId", 1195 | value: organizationId, 1196 | operator: "eq", 1197 | connector: "AND", 1198 | }, 1199 | { 1200 | field: "role", 1201 | value: role, 1202 | operator: "eq", 1203 | connector: "AND", 1204 | }, 1205 | ], 1206 | }); 1207 | if (existingRoleInDB) { 1208 | ctx.context.logger.error( 1209 | `[Dynamic Access Control] The role name "${role}" is already taken by a role in the database.`, 1210 | { 1211 | role, 1212 | organizationId, 1213 | }, 1214 | ); 1215 | throw new APIError("BAD_REQUEST", { 1216 | message: ORGANIZATION_ERROR_CODES.ROLE_NAME_IS_ALREADY_TAKEN, 1217 | }); 1218 | } 1219 | } 1220 | ``` -------------------------------------------------------------------------------- /packages/better-auth/src/plugins/open-api/logo.ts: -------------------------------------------------------------------------------- ```typescript 1 | export 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"> 2 | <rect width="75" height="75" fill="url(#pattern0_21_12)"/> 3 | <defs> 4 | <pattern id="pattern0_21_12" patternContentUnits="objectBoundingBox" width="1" height="1"> 5 | <use xlink:href="#image0_21_12" transform="scale(0.00094697)"/> 6 | </pattern> 7 | <image id="image0_21_12" width="1056" height="1056" xlink:href=""/> 8 | </defs> 9 | </svg> 10 | `; 11 | ```