#
tokens: 47827/50000 2/1094 files (page 55/67)
lines: on (toggle) GitHub
raw markdown copy reset
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 | 
```
Page 55/67FirstPrevNextLast