const path = require('path'); const mongoose = require('mongoose'); const { User } = require('@librechat/data-schemas').createModels(mongoose); require('module-alias')({ base: path.resolve(__dirname, '..', 'api') }); const { registerUser } = require('~/server/services/AuthService'); const { askQuestion, silentExit } = require('./helpers'); const connect = require('./connect'); (async () => { await connect(); console.purple('--------------------------'); console.purple('Create a new user account!'); console.purple('--------------------------'); if (process.argv.length < 5) { console.orange('Usage: npm run create-user [--email-verified=false]'); console.orange('Note: if you do not pass in the arguments, you will be prompted for them.'); console.orange( 'If you really need to pass in the password, you can do so as the 4th argument (not recommended for security).', ); console.orange('Use --email-verified=false to set emailVerified to false. Default is true.'); console.purple('--------------------------'); } let email = ''; let password = ''; let name = ''; let username = ''; let emailVerified = true; // Parse command line arguments for (let i = 2; i < process.argv.length; i++) { if (process.argv[i].startsWith('--email-verified=')) { emailVerified = process.argv[i].split('=')[1].toLowerCase() !== 'false'; continue; } if (!email) { email = process.argv[i]; } else if (!name) { name = process.argv[i]; } else if (!username) { username = process.argv[i]; } else if (!password) { console.red('Warning: password passed in as argument, this is not secure!'); password = process.argv[i]; } } if (!email) { email = await askQuestion('Email:'); } if (!email.includes('@')) { console.red('Error: Invalid email address!'); silentExit(1); } const defaultName = email.split('@')[0]; if (!name) { name = await askQuestion('Name: (default is: ' + defaultName + ')'); if (!name) { name = defaultName; } } if (!username) { username = await askQuestion('Username: (default is: ' + defaultName + ')'); if (!username) { username = defaultName; } } if (!password) { password = await askQuestion('Password: (leave blank, to generate one)'); if (!password) { password = Math.random().toString(36).slice(-18); console.orange('Your password is: ' + password); } } // Only prompt for emailVerified if it wasn't set via CLI if (!process.argv.some((arg) => arg.startsWith('--email-verified='))) { const emailVerifiedInput = await askQuestion(`Email verified? (Y/n, default is Y): If \`y\`, the user's email will be considered verified. If \`n\`, and email service is configured, the user will be sent a verification email. If \`n\`, and email service is not configured, you must have the \`ALLOW_UNVERIFIED_EMAIL_LOGIN\` .env variable set to true, or the user will need to attempt logging in to have a verification link sent to them.`); if (emailVerifiedInput.toLowerCase() === 'n') { emailVerified = false; } } const userExists = await User.findOne({ $or: [{ email }, { username }] }); if (userExists) { console.red('Error: A user with that email or username already exists!'); silentExit(1); } const user = { email, password, name, username, confirm_password: password }; let result; try { result = await registerUser(user, { emailVerified }); } catch (error) { console.red('Error: ' + error.message); silentExit(1); } if (result.status !== 200) { console.red('Error: ' + result.message); silentExit(1); } const userCreated = await User.findOne({ $or: [{ email }, { username }] }); if (userCreated) { console.green('User created successfully!'); console.green(`Email verified: ${userCreated.emailVerified}`); silentExit(0); } })(); process.on('uncaughtException', (err) => { if (!err.message.includes('fetch failed')) { console.error('There was an uncaught error:'); console.error(err); } if (err.message.includes('fetch failed')) { return; } else { process.exit(1); } });