fix: disable open signup - invite creates users directly via DB

This commit is contained in:
2026-01-28 21:46:02 +00:00
parent 1e30d349f5
commit 702156c406
2 changed files with 42 additions and 21 deletions

View File

@@ -28,7 +28,8 @@ export const auth = betterAuth({
}, },
emailAndPassword: { emailAndPassword: {
enabled: true, enabled: true,
requireEmailVerification: false, // Enable later for production requireEmailVerification: false,
disableSignUp: true, // Registration is invite-only
}, },
session: { session: {
expiresIn: 60 * 60 * 24 * 7, // 7 days expiresIn: 60 * 60 * 24 * 7, // 7 days

View File

@@ -1,8 +1,7 @@
import { Elysia, t } from 'elysia'; import { Elysia, t } from 'elysia';
import { db } from '../db'; import { db } from '../db';
import { invites, users } from '../db/schema'; import { invites, users, accounts } from '../db/schema';
import { eq, and } from 'drizzle-orm'; import { eq, and } from 'drizzle-orm';
import { auth } from '../lib/auth';
export const inviteRoutes = new Elysia({ prefix: '/auth/invite' }) export const inviteRoutes = new Elysia({ prefix: '/auth/invite' })
// Validate invite token (public - no auth required) // Validate invite token (public - no auth required)
@@ -18,7 +17,6 @@ export const inviteRoutes = new Elysia({ prefix: '/auth/invite' })
} }
if (new Date() > invite.expiresAt) { if (new Date() > invite.expiresAt) {
// Mark as expired
await db.update(invites) await db.update(invites)
.set({ status: 'expired' }) .set({ status: 'expired' })
.where(eq(invites.id, invite.id)); .where(eq(invites.id, invite.id));
@@ -38,6 +36,7 @@ export const inviteRoutes = new Elysia({ prefix: '/auth/invite' })
}) })
// Accept invite (public - no auth required) // Accept invite (public - no auth required)
// Creates user directly in DB, bypassing Better Auth's blocked signup endpoint
.post('/:token/accept', async ({ params, body, set }: { .post('/:token/accept', async ({ params, body, set }: {
params: { token: string }; params: { token: string };
body: { password: string; name?: string }; body: { password: string; name?: string };
@@ -61,22 +60,44 @@ export const inviteRoutes = new Elysia({ prefix: '/auth/invite' })
throw new Error('Invite has expired'); throw new Error('Invite has expired');
} }
// Create user via Better Auth's sign-up // Check if user with this email already exists
const [existing] = await db.select({ id: users.id })
.from(users)
.where(eq(users.email, invite.email))
.limit(1);
if (existing) {
set.status = 409;
throw new Error('An account with this email already exists');
}
try { try {
const signUpResult = await auth.api.signUpEmail({ const now = new Date();
body: { const userId = crypto.randomUUID();
const accountId = crypto.randomUUID();
const hashedPassword = await Bun.password.hash(body.password, { algorithm: 'bcrypt', cost: 10 });
// Create user record
await db.insert(users).values({
id: userId,
email: invite.email, email: invite.email,
password: body.password,
name: body.name || invite.name, name: body.name || invite.name,
}, role: invite.role,
emailVerified: false,
createdAt: now,
updatedAt: now,
}); });
// Set the user's role from the invite // Create credential account record (how Better Auth stores email/password)
if (signUpResult?.user?.id) { await db.insert(accounts).values({
await db.update(users) id: accountId,
.set({ role: invite.role }) userId,
.where(eq(users.id, signUpResult.user.id)); accountId: userId,
} providerId: 'credential',
password: hashedPassword,
createdAt: now,
updatedAt: now,
});
// Mark invite as accepted // Mark invite as accepted
await db.update(invites) await db.update(invites)
@@ -86,12 +107,11 @@ export const inviteRoutes = new Elysia({ prefix: '/auth/invite' })
return { return {
success: true, success: true,
user: { user: {
id: signUpResult.user.id, id: userId,
email: signUpResult.user.email, email: invite.email,
name: signUpResult.user.name, name: body.name || invite.name,
role: invite.role, role: invite.role,
}, },
token: signUpResult.token,
}; };
} catch (error: any) { } catch (error: any) {
set.status = 400; set.status = 400;