mirror of
https://github.com/BookStackApp/BookStack.git
synced 2025-10-26 17:31:27 +03:00
Commands: Added testing for initial admin changes
- Also changed first-admin to initial. - Updated initial handling to not require email/name to be passed, using defaults instead. - Adds missing existing email use check.
This commit is contained in:
@@ -8,7 +8,6 @@ use Illuminate\Console\Command;
|
|||||||
use Illuminate\Support\Facades\Validator;
|
use Illuminate\Support\Facades\Validator;
|
||||||
use Illuminate\Support\Str;
|
use Illuminate\Support\Str;
|
||||||
use Illuminate\Validation\Rules\Password;
|
use Illuminate\Validation\Rules\Password;
|
||||||
use Illuminate\Validation\Rules\Unique;
|
|
||||||
|
|
||||||
class CreateAdminCommand extends Command
|
class CreateAdminCommand extends Command
|
||||||
{
|
{
|
||||||
@@ -23,7 +22,7 @@ class CreateAdminCommand extends Command
|
|||||||
{--password= : The password to assign to the new admin user}
|
{--password= : The password to assign to the new admin user}
|
||||||
{--external-auth-id= : The external authentication system id for the new admin user (SAML2/LDAP/OIDC)}
|
{--external-auth-id= : The external authentication system id for the new admin user (SAML2/LDAP/OIDC)}
|
||||||
{--generate-password : Generate a random password for the new admin user}
|
{--generate-password : Generate a random password for the new admin user}
|
||||||
{--first-admin : Indicate if this should set/update the details of the initial admin user}';
|
{--initial : Indicate if this should set/update the details of the initial admin user}';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The console command description.
|
* The console command description.
|
||||||
@@ -37,12 +36,12 @@ class CreateAdminCommand extends Command
|
|||||||
*/
|
*/
|
||||||
public function handle(UserRepo $userRepo): int
|
public function handle(UserRepo $userRepo): int
|
||||||
{
|
{
|
||||||
$firstAdminOnly = $this->option('first-admin');
|
$initialAdminOnly = $this->option('initial');
|
||||||
$shouldGeneratePassword = $this->option('generate-password');
|
$shouldGeneratePassword = $this->option('generate-password');
|
||||||
$details = $this->gatherDetails($shouldGeneratePassword);
|
$details = $this->gatherDetails($shouldGeneratePassword, $initialAdminOnly);
|
||||||
|
|
||||||
$validator = Validator::make($details, [
|
$validator = Validator::make($details, [
|
||||||
'email' => ['required', 'email', 'min:5', new Unique('users', 'email')],
|
'email' => ['required', 'email', 'min:5'],
|
||||||
'name' => ['required', 'min:2'],
|
'name' => ['required', 'min:2'],
|
||||||
'password' => ['required_without:external_auth_id', Password::default()],
|
'password' => ['required_without:external_auth_id', Password::default()],
|
||||||
'external_auth_id' => ['required_without:password'],
|
'external_auth_id' => ['required_without:password'],
|
||||||
@@ -58,13 +57,20 @@ class CreateAdminCommand extends Command
|
|||||||
|
|
||||||
$adminRole = Role::getSystemRole('admin');
|
$adminRole = Role::getSystemRole('admin');
|
||||||
|
|
||||||
if ($firstAdminOnly) {
|
if ($initialAdminOnly) {
|
||||||
$handled = $this->handleFirstAdminIfExists($userRepo, $details, $shouldGeneratePassword, $adminRole);
|
$handled = $this->handleInitialAdminIfExists($userRepo, $details, $shouldGeneratePassword, $adminRole);
|
||||||
if ($handled) {
|
if ($handled !== null) {
|
||||||
return 0;
|
return $handled ? 0 : 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$emailUsed = $userRepo->getByEmail($details['email']) !== null;
|
||||||
|
if ($emailUsed) {
|
||||||
|
$this->error("Could not create admin account.");
|
||||||
|
$this->error("An account with the email address \"{$details['email']}\" already exists.");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
$user = $userRepo->createWithoutActivity($validator->validated());
|
$user = $userRepo->createWithoutActivity($validator->validated());
|
||||||
$user->attachRole($adminRole);
|
$user->attachRole($adminRole);
|
||||||
$user->email_confirmed = true;
|
$user->email_confirmed = true;
|
||||||
@@ -80,14 +86,19 @@ class CreateAdminCommand extends Command
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handle updates to the first admin if exists.
|
* Handle updates to the original admin account if it exists.
|
||||||
* Returns true if the action has been handled (user updated or already a non-default admin user) otherwise
|
* Returns true if it's been successfully handled, false if unsuccessful, or null if not handled.
|
||||||
* returns false if no action has been taken, and we therefore need to proceed with a normal account creation.
|
|
||||||
*/
|
*/
|
||||||
protected function handleFirstAdminIfExists(UserRepo $userRepo, array $data, bool $generatePassword, Role $adminRole): bool
|
protected function handleInitialAdminIfExists(UserRepo $userRepo, array $data, bool $generatePassword, Role $adminRole): bool|null
|
||||||
{
|
{
|
||||||
$defaultAdmin = $userRepo->getByEmail('admin@admin.com');
|
$defaultAdmin = $userRepo->getByEmail('admin@admin.com');
|
||||||
if ($defaultAdmin && $defaultAdmin->hasSystemRole('admin')) {
|
if ($defaultAdmin && $defaultAdmin->hasSystemRole('admin')) {
|
||||||
|
if ($defaultAdmin->email !== $data['email'] && $userRepo->getByEmail($data['email']) !== null) {
|
||||||
|
$this->error("Could not create admin account.");
|
||||||
|
$this->error("An account with the email address \"{$data['email']}\" already exists.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
$userRepo->updateWithoutActivity($defaultAdmin, $data, true);
|
$userRepo->updateWithoutActivity($defaultAdmin, $data, true);
|
||||||
if ($generatePassword) {
|
if ($generatePassword) {
|
||||||
$this->line($data['password']);
|
$this->line($data['password']);
|
||||||
@@ -101,20 +112,28 @@ class CreateAdminCommand extends Command
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function gatherDetails(bool $generatePassword): array
|
protected function gatherDetails(bool $generatePassword, bool $initialAdmin): array
|
||||||
{
|
{
|
||||||
$details = $this->snakeCaseOptions();
|
$details = $this->snakeCaseOptions();
|
||||||
|
|
||||||
if (empty($details['email'])) {
|
if (empty($details['email'])) {
|
||||||
|
if ($initialAdmin) {
|
||||||
|
$details['email'] = 'admin@example.com';
|
||||||
|
} else {
|
||||||
$details['email'] = $this->ask('Please specify an email address for the new admin user');
|
$details['email'] = $this->ask('Please specify an email address for the new admin user');
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (empty($details['name'])) {
|
if (empty($details['name'])) {
|
||||||
|
if ($initialAdmin) {
|
||||||
|
$details['name'] = 'Admin';
|
||||||
|
} else {
|
||||||
$details['name'] = $this->ask('Please specify a name for the new admin user');
|
$details['name'] = $this->ask('Please specify a name for the new admin user');
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (empty($details['password'])) {
|
if (empty($details['password'])) {
|
||||||
if (empty($details['external_auth_id'])) {
|
if (empty($details['external_auth_id'])) {
|
||||||
|
|||||||
@@ -2,8 +2,11 @@
|
|||||||
|
|
||||||
namespace Tests\Commands;
|
namespace Tests\Commands;
|
||||||
|
|
||||||
|
use BookStack\Users\Models\Role;
|
||||||
use BookStack\Users\Models\User;
|
use BookStack\Users\Models\User;
|
||||||
|
use Illuminate\Support\Facades\Artisan;
|
||||||
use Illuminate\Support\Facades\Auth;
|
use Illuminate\Support\Facades\Auth;
|
||||||
|
use Illuminate\Support\Facades\Hash;
|
||||||
use Tests\TestCase;
|
use Tests\TestCase;
|
||||||
|
|
||||||
class CreateAdminCommandTest extends TestCase
|
class CreateAdminCommandTest extends TestCase
|
||||||
@@ -60,4 +63,168 @@ class CreateAdminCommandTest extends TestCase
|
|||||||
]);
|
]);
|
||||||
$this->assertTrue(Auth::attempt(['email' => 'admintest@example.com', 'password' => 'hunter2000']));
|
$this->assertTrue(Auth::attempt(['email' => 'admintest@example.com', 'password' => 'hunter2000']));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function test_generate_password_option()
|
||||||
|
{
|
||||||
|
$this->withoutMockingConsoleOutput()
|
||||||
|
->artisan('bookstack:create-admin', [
|
||||||
|
'--email' => 'admintest@example.com',
|
||||||
|
'--name' => 'Admin Test',
|
||||||
|
'--generate-password' => true,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$output = trim(Artisan::output());
|
||||||
|
$this->assertMatchesRegularExpression('/^[a-zA-Z0-9]{32}$/', $output);
|
||||||
|
|
||||||
|
$user = User::query()->where('email', '=', 'admintest@example.com')->first();
|
||||||
|
$this->assertTrue(Hash::check($output, $user->password));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_initial_option_updates_default_admin()
|
||||||
|
{
|
||||||
|
$defaultAdmin = User::query()->where('email', '=', 'admin@admin.com')->first();
|
||||||
|
|
||||||
|
$this->artisan('bookstack:create-admin', [
|
||||||
|
'--email' => 'firstadmin@example.com',
|
||||||
|
'--name' => 'Admin Test',
|
||||||
|
'--password' => 'testing-7',
|
||||||
|
'--initial' => true,
|
||||||
|
])->expectsOutput('The default admin user has been updated with the provided details!')
|
||||||
|
->assertExitCode(0);
|
||||||
|
|
||||||
|
$defaultAdmin->refresh();
|
||||||
|
|
||||||
|
$this->assertEquals('firstadmin@example.com', $defaultAdmin->email);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_initial_option_does_not_update_if_only_non_default_admin_exists()
|
||||||
|
{
|
||||||
|
$defaultAdmin = User::query()->where('email', '=', 'admin@admin.com')->first();
|
||||||
|
$defaultAdmin->email = 'testadmin@example.com';
|
||||||
|
$defaultAdmin->save();
|
||||||
|
|
||||||
|
$this->artisan('bookstack:create-admin', [
|
||||||
|
'--email' => 'firstadmin@example.com',
|
||||||
|
'--name' => 'Admin Test',
|
||||||
|
'--password' => 'testing-7',
|
||||||
|
'--initial' => true,
|
||||||
|
])->expectsOutput('Non-default admin user already exists. Skipping creation of new admin user.')
|
||||||
|
->assertExitCode(0);
|
||||||
|
|
||||||
|
$defaultAdmin->refresh();
|
||||||
|
|
||||||
|
$this->assertEquals('testadmin@example.com', $defaultAdmin->email);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_initial_option_updates_creates_new_admin_if_none_exists()
|
||||||
|
{
|
||||||
|
$adminRole = Role::getSystemRole('admin');
|
||||||
|
$adminRole->users()->delete();
|
||||||
|
$this->assertEquals(0, $adminRole->users()->count());
|
||||||
|
|
||||||
|
$this->artisan('bookstack:create-admin', [
|
||||||
|
'--email' => 'firstadmin@example.com',
|
||||||
|
'--name' => 'My initial admin',
|
||||||
|
'--password' => 'testing-7',
|
||||||
|
'--initial' => true,
|
||||||
|
])->expectsOutput("Admin account with email \"firstadmin@example.com\" successfully created!")
|
||||||
|
->assertExitCode(0);
|
||||||
|
|
||||||
|
$this->assertEquals(1, $adminRole->users()->count());
|
||||||
|
$this->assertDatabaseHas('users', [
|
||||||
|
'email' => 'firstadmin@example.com',
|
||||||
|
'name' => 'My initial admin',
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_initial_rerun_does_not_error_but_skips()
|
||||||
|
{
|
||||||
|
$adminRole = Role::getSystemRole('admin');
|
||||||
|
$adminRole->users()->delete();
|
||||||
|
|
||||||
|
$this->artisan('bookstack:create-admin', [
|
||||||
|
'--email' => 'firstadmin@example.com',
|
||||||
|
'--name' => 'My initial admin',
|
||||||
|
'--password' => 'testing-7',
|
||||||
|
'--initial' => true,
|
||||||
|
])->expectsOutput("Admin account with email \"firstadmin@example.com\" successfully created!")
|
||||||
|
->assertExitCode(0);
|
||||||
|
|
||||||
|
$this->artisan('bookstack:create-admin', [
|
||||||
|
'--email' => 'firstadmin@example.com',
|
||||||
|
'--name' => 'My initial admin',
|
||||||
|
'--password' => 'testing-7',
|
||||||
|
'--initial' => true,
|
||||||
|
])->expectsOutput("Non-default admin user already exists. Skipping creation of new admin user.")
|
||||||
|
->assertExitCode(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_initial_option_creation_errors_if_email_already_exists()
|
||||||
|
{
|
||||||
|
$adminRole = Role::getSystemRole('admin');
|
||||||
|
$adminRole->users()->delete();
|
||||||
|
$editor = $this->users->editor();
|
||||||
|
|
||||||
|
$this->artisan('bookstack:create-admin', [
|
||||||
|
'--email' => $editor->email,
|
||||||
|
'--name' => 'My initial admin',
|
||||||
|
'--password' => 'testing-7',
|
||||||
|
'--initial' => true,
|
||||||
|
])->expectsOutput("Could not create admin account.")
|
||||||
|
->expectsOutput("An account with the email address \"{$editor->email}\" already exists.")
|
||||||
|
->assertExitCode(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_initial_option_updating_errors_if_email_already_exists()
|
||||||
|
{
|
||||||
|
$editor = $this->users->editor();
|
||||||
|
$defaultAdmin = User::query()->where('email', '=', 'admin@admin.com')->first();
|
||||||
|
$this->assertNotNull($defaultAdmin);
|
||||||
|
|
||||||
|
$this->artisan('bookstack:create-admin', [
|
||||||
|
'--email' => $editor->email,
|
||||||
|
'--name' => 'My initial admin',
|
||||||
|
'--password' => 'testing-7',
|
||||||
|
'--initial' => true,
|
||||||
|
])->expectsOutput("Could not create admin account.")
|
||||||
|
->expectsOutput("An account with the email address \"{$editor->email}\" already exists.")
|
||||||
|
->assertExitCode(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_initial_option_does_not_require_name_or_email_to_be_passed()
|
||||||
|
{
|
||||||
|
$adminRole = Role::getSystemRole('admin');
|
||||||
|
$adminRole->users()->delete();
|
||||||
|
$this->assertEquals(0, $adminRole->users()->count());
|
||||||
|
|
||||||
|
$this->artisan('bookstack:create-admin', [
|
||||||
|
'--generate-password' => true,
|
||||||
|
'--initial' => true,
|
||||||
|
])->assertExitCode(0);
|
||||||
|
|
||||||
|
$this->assertEquals(1, $adminRole->users()->count());
|
||||||
|
$this->assertDatabaseHas('users', [
|
||||||
|
'email' => 'admin@example.com',
|
||||||
|
'name' => 'Admin',
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_initial_option_updating_existing_user_with_generate_password_only_outputs_password()
|
||||||
|
{
|
||||||
|
$defaultAdmin = User::query()->where('email', '=', 'admin@admin.com')->first();
|
||||||
|
|
||||||
|
$this->withoutMockingConsoleOutput()
|
||||||
|
->artisan('bookstack:create-admin', [
|
||||||
|
'--email' => 'firstadmin@example.com',
|
||||||
|
'--name' => 'Admin Test',
|
||||||
|
'--generate-password' => true,
|
||||||
|
'--initial' => true,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$output = Artisan::output();
|
||||||
|
$this->assertMatchesRegularExpression('/^[a-zA-Z0-9]{32}$/', $output);
|
||||||
|
|
||||||
|
$defaultAdmin->refresh();
|
||||||
|
$this->assertEquals('firstadmin@example.com', $defaultAdmin->email);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user