fosrl.pangolin/server/setup/setupServerAdmin.ts

90 lines
2.7 KiB
TypeScript
Raw Normal View History

2025-01-01 21:41:31 -05:00
import { generateId, invalidateAllSessions } from "@server/auth/sessions/app";
2024-12-25 15:54:32 -05:00
import { hashPassword, verifyPassword } from "@server/auth/password";
2025-01-01 21:41:31 -05:00
import config from "@server/lib/config";
2024-12-25 15:54:32 -05:00
import db from "@server/db";
import { users } from "@server/db/schema";
import logger from "@server/logger";
import { eq } from "drizzle-orm";
import moment from "moment";
import { fromError } from "zod-validation-error";
2025-01-01 21:41:31 -05:00
import { passwordSchema } from "@server/auth/passwordSchema";
2024-12-25 15:54:32 -05:00
export async function setupServerAdmin() {
const {
server_admin: { email, password }
} = config.getRawConfig().users;
2024-12-25 15:54:32 -05:00
const parsed = passwordSchema.safeParse(password);
if (!parsed.success) {
throw Error(
`Invalid server admin password: ${fromError(parsed.error).toString()}`
);
}
const passwordHash = await hashPassword(password);
await db.transaction(async (trx) => {
try {
const [existing] = await trx
.select()
.from(users)
.where(eq(users.email, email));
if (existing) {
const passwordChanged = !(await verifyPassword(
password,
existing.passwordHash
));
if (passwordChanged) {
await trx
.update(users)
.set({ passwordHash })
2024-12-26 11:43:02 -05:00
.where(eq(users.userId, existing.userId));
2024-12-25 15:54:32 -05:00
// this isn't using the transaction, but it's probably fine
await invalidateAllSessions(existing.userId);
logger.info(`Server admin (${email}) password updated`);
}
if (existing.serverAdmin) {
2024-12-26 11:43:02 -05:00
logger.info(`Server admin (${email}) already exists`)
2024-12-25 15:54:32 -05:00
return;
}
await trx.update(users).set({ serverAdmin: false });
await trx
.update(users)
.set({
serverAdmin: true
})
.where(eq(users.email, email));
2024-12-26 11:43:02 -05:00
logger.info(`Server admin (${email}) set`);
2024-12-25 15:54:32 -05:00
return;
}
const userId = generateId(15);
2025-01-11 19:47:07 -05:00
await trx.update(users).set({ serverAdmin: false });
2024-12-25 15:54:32 -05:00
await db.insert(users).values({
userId: userId,
email: email,
passwordHash,
dateCreated: moment().toISOString(),
serverAdmin: true,
emailVerified: true
});
logger.info(`Server admin (${email}) created`);
} catch (e) {
logger.error(e);
trx.rollback();
}
});
}