mirror of
https://github.com/nicotsx/zerobyte.git
synced 2026-04-18 13:57:52 -04:00
87 lines
2.8 KiB
TypeScript
87 lines
2.8 KiB
TypeScript
import { input, select } from "@inquirer/prompts";
|
|
import { Command } from "commander";
|
|
import { eq } from "drizzle-orm";
|
|
import { toMessage } from "~/server/utils/errors";
|
|
import { db } from "../../db/db";
|
|
import { sessionsTable, usersTable } from "../../db/schema";
|
|
|
|
const listUsers = () => {
|
|
return db.select({ id: usersTable.id, username: usersTable.username }).from(usersTable);
|
|
};
|
|
|
|
const changeUsername = async (oldUsername: string, newUsername: string) => {
|
|
const [user] = await db.select().from(usersTable).where(eq(usersTable.username, oldUsername));
|
|
|
|
if (!user) {
|
|
throw new Error(`User "${oldUsername}" not found`);
|
|
}
|
|
|
|
const normalizedUsername = newUsername.toLowerCase().trim();
|
|
|
|
const [existingUser] = await db.select().from(usersTable).where(eq(usersTable.username, normalizedUsername));
|
|
if (existingUser) {
|
|
throw new Error(`Username "${newUsername}" is already taken`);
|
|
}
|
|
|
|
const usernameRegex = /^[a-z0-9_]{3,30}$/;
|
|
if (!usernameRegex.test(normalizedUsername)) {
|
|
throw new Error(
|
|
`Invalid username "${newUsername}". Usernames must be 3-30 characters long and can only contain lowercase letters, numbers, and underscores.`,
|
|
);
|
|
}
|
|
|
|
db.transaction((tx) => {
|
|
tx.update(usersTable).set({ username: normalizedUsername }).where(eq(usersTable.id, user.id)).run();
|
|
tx.delete(sessionsTable).where(eq(sessionsTable.userId, user.id)).run();
|
|
});
|
|
};
|
|
|
|
export const changeUsernameCommand = new Command("change-username")
|
|
.description("Change username for a user")
|
|
.option("-u, --username <username>", "Current username of the account")
|
|
.option("-n, --new-username <new-username>", "New username for the account")
|
|
.action(async (options) => {
|
|
console.info("\n👤 Zerobyte Change Username\n");
|
|
|
|
let username = options.username;
|
|
let newUsername = options.newUsername;
|
|
|
|
try {
|
|
if (!username) {
|
|
const users = await listUsers();
|
|
|
|
if (users.length === 0) {
|
|
console.error("No users found in the database.");
|
|
return;
|
|
}
|
|
|
|
username = await select({
|
|
message: "Select a user to change username for:",
|
|
choices: users.map((u) => ({
|
|
name: u.username,
|
|
value: u.username,
|
|
})),
|
|
});
|
|
}
|
|
|
|
if (!newUsername) {
|
|
newUsername = await input({
|
|
message: "Enter the new username:",
|
|
validate: (val) => {
|
|
const usernameRegex = /^[a-z0-9_]{3,30}$/;
|
|
return usernameRegex.test(val)
|
|
? true
|
|
: "Username must be 3-30 characters and contain only lowercase letters, numbers, or underscores";
|
|
},
|
|
});
|
|
newUsername = newUsername.toLowerCase().trim();
|
|
}
|
|
|
|
await changeUsername(username, newUsername);
|
|
console.info(`\n✅ Username for "${username}" has been changed to "${newUsername}" successfully.`);
|
|
} catch (error) {
|
|
console.error(`\n❌ Error: ${toMessage(error)}`);
|
|
process.exit(1);
|
|
}
|
|
});
|