X-Git-Url: http://source.bookstackapp.com/bookstack/blobdiff_plain/ddb7f33868ea499ab8f48a7062f145e8c0fbe02f..refs/pull/5280/head:/tests/Auth/SocialAuthTest.php diff --git a/tests/Auth/SocialAuthTest.php b/tests/Auth/SocialAuthTest.php index 526c0e199..89b8fd167 100644 --- a/tests/Auth/SocialAuthTest.php +++ b/tests/Auth/SocialAuthTest.php @@ -1,20 +1,28 @@ -make(); + $user = User::factory()->make(); $this->setSettings(['registration-enabled' => 'true']); - config(['GOOGLE_APP_ID' => 'abc123', 'GOOGLE_APP_SECRET' => '123abc', 'APP_URL' => 'http://localhost']); + config(['GOOGLE_APP_ID' => 'abc123', 'GOOGLE_APP_SECRET' => '123abc']); - $mockSocialite = \Mockery::mock('Laravel\Socialite\Contracts\Factory'); - $this->app['Laravel\Socialite\Contracts\Factory'] = $mockSocialite; - $mockSocialDriver = \Mockery::mock('Laravel\Socialite\Contracts\Provider'); - $mockSocialUser = \Mockery::mock('\Laravel\Socialite\Contracts\User'); + $mockSocialite = $this->mock(Factory::class); + $mockSocialDriver = Mockery::mock(Provider::class); + $mockSocialUser = Mockery::mock(\Laravel\Socialite\Contracts\User::class); $mockSocialite->shouldReceive('driver')->twice()->with('google')->andReturn($mockSocialDriver); $mockSocialDriver->shouldReceive('redirect')->once()->andReturn(redirect('/')); @@ -37,13 +45,11 @@ class SocialAuthTest extends TestCase config([ 'GOOGLE_APP_ID' => 'abc123', 'GOOGLE_APP_SECRET' => '123abc', 'GITHUB_APP_ID' => 'abc123', 'GITHUB_APP_SECRET' => '123abc', - 'APP_URL' => 'http://localhost' ]); - $mockSocialite = \Mockery::mock('Laravel\Socialite\Contracts\Factory'); - $this->app['Laravel\Socialite\Contracts\Factory'] = $mockSocialite; - $mockSocialDriver = \Mockery::mock('Laravel\Socialite\Contracts\Provider'); - $mockSocialUser = \Mockery::mock('\Laravel\Socialite\Contracts\User'); + $mockSocialite = $this->mock(Factory::class); + $mockSocialDriver = Mockery::mock(Provider::class); + $mockSocialUser = Mockery::mock(\Laravel\Socialite\Contracts\User::class); $mockSocialUser->shouldReceive('getId')->twice()->andReturn('logintest123'); @@ -54,8 +60,8 @@ class SocialAuthTest extends TestCase // Test login routes $resp = $this->get('/login'); - $resp->assertElementExists('a#social-login-google[href$="/login/service/google"]'); - $resp = $this->followingRedirects()->get("/login/service/google"); + $this->withHtml($resp)->assertElementExists('a#social-login-google[href$="/login/service/google"]'); + $resp = $this->followingRedirects()->get('/login/service/google'); $resp->assertSee('login-form'); // Test social callback @@ -64,33 +70,85 @@ class SocialAuthTest extends TestCase $resp->assertSee(trans('errors.social_account_not_used', ['socialAccount' => 'Google'])); $resp = $this->get('/login'); - $resp->assertElementExists('a#social-login-github[href$="/login/service/github"]'); - $resp = $this->followingRedirects()->get("/login/service/github"); + $this->withHtml($resp)->assertElementExists('a#social-login-github[href$="/login/service/github"]'); + $resp = $this->followingRedirects()->get('/login/service/github'); $resp->assertSee('login-form'); - // Test social callback with matching social account - \DB::table('social_accounts')->insert([ - 'user_id' => $this->getAdmin()->id, - 'driver' => 'github', - 'driver_id' => 'logintest123' + DB::table('social_accounts')->insert([ + 'user_id' => $this->users->admin()->id, + 'driver' => 'github', + 'driver_id' => 'logintest123', ]); $resp = $this->followingRedirects()->get('/login/service/github/callback'); - $resp->assertDontSee("login-form"); + $resp->assertDontSee('login-form'); + $this->assertActivityExists(ActivityType::AUTH_LOGIN, null, 'github; (' . $this->users->admin()->id . ') ' . $this->users->admin()->name); + } + + public function test_social_account_attach() + { + config([ + 'GOOGLE_APP_ID' => 'abc123', 'GOOGLE_APP_SECRET' => '123abc', + ]); + $editor = $this->users->editor(); + + $mockSocialite = $this->mock(Factory::class); + $mockSocialDriver = Mockery::mock(Provider::class); + $mockSocialUser = Mockery::mock(\Laravel\Socialite\Contracts\User::class); + + $mockSocialUser->shouldReceive('getId')->twice()->andReturn('logintest123'); + $mockSocialUser->shouldReceive('getAvatar')->andReturn(null); + + $mockSocialite->shouldReceive('driver')->twice()->with('google')->andReturn($mockSocialDriver); + $mockSocialDriver->shouldReceive('redirect')->once()->andReturn(redirect('/login/service/google/callback')); + $mockSocialDriver->shouldReceive('user')->once()->andReturn($mockSocialUser); + + // Test login routes + $resp = $this->actingAs($editor)->followingRedirects()->get('/login/service/google'); + $resp->assertSee('Access & Security'); + + // Test social callback with matching social account + $this->assertDatabaseHas('social_accounts', [ + 'user_id' => $editor->id, + 'driver' => 'google', + 'driver_id' => 'logintest123', + ]); + } + + public function test_social_account_detach() + { + $editor = $this->users->editor(); + config([ + 'GITHUB_APP_ID' => 'abc123', 'GITHUB_APP_SECRET' => '123abc', + ]); + + $socialAccount = SocialAccount::query()->forceCreate([ + 'user_id' => $editor->id, + 'driver' => 'github', + 'driver_id' => 'logintest123', + ]); + + $resp = $this->actingAs($editor)->get('/my-account/auth'); + $this->withHtml($resp)->assertElementContains('form[action$="/login/service/github/detach"]', 'Disconnect Account'); + + $resp = $this->post('/login/service/github/detach'); + $resp->assertRedirect('/my-account/auth#social-accounts'); + $resp = $this->followRedirects($resp); + $resp->assertSee('Github account was successfully disconnected from your profile.'); + + $this->assertDatabaseMissing('social_accounts', ['id' => $socialAccount->id]); } public function test_social_autoregister() { config([ 'services.google.client_id' => 'abc123', 'services.google.client_secret' => '123abc', - 'APP_URL' => 'http://localhost' ]); - $user = factory(\BookStack\Auth\User::class)->make(); - $mockSocialite = \Mockery::mock('Laravel\Socialite\Contracts\Factory'); - $this->app['Laravel\Socialite\Contracts\Factory'] = $mockSocialite; - $mockSocialDriver = \Mockery::mock('Laravel\Socialite\Contracts\Provider'); - $mockSocialUser = \Mockery::mock('\Laravel\Socialite\Contracts\User'); + $user = User::factory()->make(); + $mockSocialite = $this->mock(Factory::class); + $mockSocialDriver = Mockery::mock(Provider::class); + $mockSocialUser = Mockery::mock(\Laravel\Socialite\Contracts\User::class); $mockSocialUser->shouldReceive('getId')->times(4)->andReturn(1); $mockSocialUser->shouldReceive('getEmail')->times(2)->andReturn($user->email); @@ -122,14 +180,13 @@ class SocialAuthTest extends TestCase { config([ 'services.google.client_id' => 'abc123', 'services.google.client_secret' => '123abc', - 'APP_URL' => 'http://localhost', 'services.google.auto_register' => true, 'services.google.auto_confirm' => true + 'services.google.auto_register' => true, 'services.google.auto_confirm' => true, ]); - $user = factory(\BookStack\Auth\User::class)->make(); - $mockSocialite = \Mockery::mock('Laravel\Socialite\Contracts\Factory'); - $this->app['Laravel\Socialite\Contracts\Factory'] = $mockSocialite; - $mockSocialDriver = \Mockery::mock('Laravel\Socialite\Contracts\Provider'); - $mockSocialUser = \Mockery::mock('\Laravel\Socialite\Contracts\User'); + $user = User::factory()->make(); + $mockSocialite = $this->mock(Factory::class); + $mockSocialDriver = Mockery::mock(Provider::class); + $mockSocialUser = Mockery::mock(\Laravel\Socialite\Contracts\User::class); $mockSocialUser->shouldReceive('getId')->times(3)->andReturn(1); $mockSocialUser->shouldReceive('getEmail')->times(2)->andReturn($user->email); @@ -156,4 +213,30 @@ class SocialAuthTest extends TestCase $this->assertStringContainsString('prompt=select_account', $resp->headers->get('Location')); } + public function test_social_registration_with_no_name_uses_email_as_name() + { + $user = User::factory()->make(['email' => 'nonameuser@example.com']); + + $this->setSettings(['registration-enabled' => 'true']); + config(['GITHUB_APP_ID' => 'abc123', 'GITHUB_APP_SECRET' => '123abc']); + + $mockSocialite = $this->mock(Factory::class); + $mockSocialDriver = Mockery::mock(Provider::class); + $mockSocialUser = Mockery::mock(\Laravel\Socialite\Contracts\User::class); + + $mockSocialite->shouldReceive('driver')->twice()->with('github')->andReturn($mockSocialDriver); + $mockSocialDriver->shouldReceive('redirect')->once()->andReturn(redirect('/')); + $mockSocialDriver->shouldReceive('user')->once()->andReturn($mockSocialUser); + + $mockSocialUser->shouldReceive('getId')->twice()->andReturn(1); + $mockSocialUser->shouldReceive('getEmail')->twice()->andReturn($user->email); + $mockSocialUser->shouldReceive('getName')->once()->andReturn(''); + $mockSocialUser->shouldReceive('getAvatar')->once()->andReturn('avatar_placeholder'); + + $this->get('/register/service/github'); + $this->get('/login/service/github/callback'); + $this->assertDatabaseHas('users', ['name' => 'nonameuser', 'email' => $user->email]); + $user = $user->whereEmail($user->email)->first(); + $this->assertDatabaseHas('social_accounts', ['user_id' => $user->id]); + } }