2023-12-09 17:22:24 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Tests\Feature\Http\Auth;
|
|
|
|
|
|
|
|
use App\Facades\Settings;
|
|
|
|
use App\Http\Controllers\Auth\SocialiteController;
|
|
|
|
use App\Models\User;
|
|
|
|
use Illuminate\Support\Facades\DB;
|
|
|
|
use Laravel\Socialite\Facades\Socialite;
|
|
|
|
use PHPUnit\Framework\Attributes\CoversClass;
|
2024-06-26 14:29:13 +02:00
|
|
|
use PHPUnit\Framework\Attributes\Test;
|
2023-12-09 17:22:24 +01:00
|
|
|
use Tests\FeatureTestCase;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* SocialiteControllerTest test class
|
|
|
|
*/
|
|
|
|
#[CoversClass(SocialiteController::class)]
|
|
|
|
class SocialiteControllerTest extends FeatureTestCase
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @var \App\Models\User|\Illuminate\Contracts\Auth\Authenticatable
|
|
|
|
*/
|
|
|
|
protected $user;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var \Laravel\Socialite\Two\User
|
|
|
|
*/
|
|
|
|
protected $socialiteUser;
|
|
|
|
|
|
|
|
private const USER_OAUTH_ID = '12345';
|
|
|
|
|
|
|
|
private const USER_OAUTH_PROVIDER = 'github';
|
|
|
|
|
|
|
|
private const USER_NAME = 'John';
|
|
|
|
|
|
|
|
private const USER_NICKNAME = 'Jo';
|
|
|
|
|
|
|
|
private const USER_EMAIL = 'john@provider.com';
|
|
|
|
|
|
|
|
public function setUp() : void
|
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
|
|
|
|
DB::table('users')->delete();
|
|
|
|
$this->user = User::factory()->create([
|
|
|
|
'name' => self::USER_NAME,
|
|
|
|
'email' => self::USER_EMAIL,
|
|
|
|
'password' => 'password',
|
|
|
|
'is_admin' => 1,
|
|
|
|
'oauth_id' => self::USER_OAUTH_ID,
|
|
|
|
'oauth_provider' => self::USER_OAUTH_PROVIDER,
|
|
|
|
]);
|
|
|
|
|
2023-12-20 16:55:58 +01:00
|
|
|
$this->socialiteUser = new \Laravel\Socialite\Two\User;
|
|
|
|
$this->socialiteUser->id = self::USER_OAUTH_ID;
|
|
|
|
$this->socialiteUser->name = self::USER_NAME;
|
|
|
|
$this->socialiteUser->email = self::USER_EMAIL;
|
2023-12-09 17:22:24 +01:00
|
|
|
$this->socialiteUser->nickname = self::USER_NICKNAME;
|
|
|
|
}
|
2023-12-20 16:55:58 +01:00
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2023-12-09 17:22:24 +01:00
|
|
|
public function test_redirect_redirects_to_provider_url()
|
|
|
|
{
|
|
|
|
Settings::set('enableSso', true);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/redirect/github');
|
|
|
|
|
|
|
|
$response->assertRedirectContains('https://github.com/login/oauth/authorize');
|
|
|
|
}
|
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2023-12-09 17:22:24 +01:00
|
|
|
public function test_redirect_returns_error_when_registrations_are_disabled()
|
|
|
|
{
|
|
|
|
Settings::set('enableSso', false);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/redirect/github');
|
|
|
|
|
|
|
|
$response->assertRedirect('/error?err=sso_disabled');
|
|
|
|
}
|
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2023-12-09 17:22:24 +01:00
|
|
|
public function test_callback_authenticates_the_user()
|
|
|
|
{
|
|
|
|
Socialite::shouldReceive('driver->user')
|
|
|
|
->andReturn($this->socialiteUser);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/callback/github', ['driver' => 'github']);
|
|
|
|
|
|
|
|
$this->assertAuthenticatedAs($this->user, 'web-guard');
|
|
|
|
}
|
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2023-12-09 17:22:24 +01:00
|
|
|
public function test_callback_redirects_authenticated_user_to_accounts()
|
|
|
|
{
|
|
|
|
Socialite::shouldReceive('driver->user')
|
|
|
|
->andReturn($this->socialiteUser);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/callback/github', ['driver' => 'github']);
|
|
|
|
|
|
|
|
$response->assertRedirect('/accounts');
|
|
|
|
}
|
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2023-12-09 17:22:24 +01:00
|
|
|
public function test_callback_updates_user_informations()
|
|
|
|
{
|
2023-12-20 16:55:58 +01:00
|
|
|
$socialiteUpdatedUser = new \Laravel\Socialite\Two\User;
|
|
|
|
$socialiteUpdatedUser->id = self::USER_OAUTH_ID;
|
|
|
|
$socialiteUpdatedUser->email = 'new_email';
|
2023-12-09 17:22:24 +01:00
|
|
|
$socialiteUpdatedUser->nickname = 'new_nickname';
|
|
|
|
|
|
|
|
Socialite::shouldReceive('driver->user')
|
|
|
|
->andReturn($socialiteUpdatedUser);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/callback/github', ['driver' => 'github']);
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('users', [
|
|
|
|
'oauth_id' => self::USER_OAUTH_ID,
|
|
|
|
'oauth_provider' => self::USER_OAUTH_PROVIDER,
|
|
|
|
'email' => 'new_email',
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2023-12-09 17:22:24 +01:00
|
|
|
public function test_callback_updates_username_with_fallback_value()
|
|
|
|
{
|
2023-12-20 16:55:58 +01:00
|
|
|
$socialiteUpdatedUser = new \Laravel\Socialite\Two\User;
|
|
|
|
$socialiteUpdatedUser->id = self::USER_OAUTH_ID;
|
|
|
|
$socialiteUpdatedUser->name = 'new_name';
|
2023-12-09 17:22:24 +01:00
|
|
|
$socialiteUpdatedUser->email = 'new_email';
|
|
|
|
|
|
|
|
Socialite::shouldReceive('driver->user')
|
|
|
|
->andReturn($socialiteUpdatedUser);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/callback/github', ['driver' => 'github']);
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('users', [
|
|
|
|
'oauth_id' => self::USER_OAUTH_ID,
|
|
|
|
'oauth_provider' => self::USER_OAUTH_PROVIDER,
|
|
|
|
'email' => 'new_email',
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2023-12-09 17:22:24 +01:00
|
|
|
public function test_callback_registers_new_user()
|
|
|
|
{
|
2023-12-20 16:55:58 +01:00
|
|
|
$newSocialiteUser = new \Laravel\Socialite\Two\User;
|
|
|
|
$newSocialiteUser->id = 'new_id';
|
|
|
|
$newSocialiteUser->name = 'jane';
|
2023-12-09 17:22:24 +01:00
|
|
|
$newSocialiteUser->email = 'jane@provider.com';
|
|
|
|
|
|
|
|
Socialite::shouldReceive('driver->user')
|
|
|
|
->andReturn($newSocialiteUser);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/callback/github', ['driver' => 'github']);
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('users', [
|
|
|
|
'oauth_id' => 'new_id',
|
|
|
|
'oauth_provider' => self::USER_OAUTH_PROVIDER,
|
|
|
|
'email' => 'jane@provider.com',
|
|
|
|
'is_admin' => 0,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2023-12-14 15:39:14 +01:00
|
|
|
public function test_callback_registers_new_user_with_existing_name()
|
|
|
|
{
|
2023-12-20 16:55:58 +01:00
|
|
|
$socialiteUserWithSameName = new \Laravel\Socialite\Two\User;
|
|
|
|
$socialiteUserWithSameName->id = 'socialiteUserWithSameNameId';
|
|
|
|
$socialiteUserWithSameName->name = self::USER_NAME;
|
|
|
|
$socialiteUserWithSameName->email = 'socialiteuserwithsamename@example.com';
|
2023-12-14 15:39:14 +01:00
|
|
|
$socialiteUserWithSameName->nickname = self::USER_NICKNAME;
|
|
|
|
|
|
|
|
Socialite::shouldReceive('driver->user')
|
|
|
|
->andReturn($socialiteUserWithSameName);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/callback/github', ['driver' => 'github']);
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('users', [
|
|
|
|
'oauth_id' => 'socialiteUserWithSameNameId',
|
|
|
|
'oauth_provider' => self::USER_OAUTH_PROVIDER,
|
|
|
|
'email' => 'socialiteuserwithsamename@example.com',
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2023-12-09 17:22:24 +01:00
|
|
|
public function test_callback_always_registers_first_user_as_admin()
|
|
|
|
{
|
|
|
|
DB::table('users')->delete();
|
|
|
|
Settings::set('disableRegistration', true);
|
|
|
|
Settings::set('enableSso', false);
|
|
|
|
|
|
|
|
Socialite::shouldReceive('driver->user')
|
|
|
|
->andReturn($this->socialiteUser);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/callback/github', ['driver' => 'github']);
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('users', [
|
|
|
|
'oauth_id' => self::USER_OAUTH_ID,
|
|
|
|
'oauth_provider' => self::USER_OAUTH_PROVIDER,
|
|
|
|
'is_admin' => 1,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2023-12-14 15:39:14 +01:00
|
|
|
public function test_callback_returns_error_when_email_is_already_used()
|
|
|
|
{
|
|
|
|
$userWithSameEmail = User::factory()->create([
|
|
|
|
'name' => 'userWithSameEmail',
|
|
|
|
'email' => 'other@example.com',
|
|
|
|
'password' => 'password',
|
|
|
|
]);
|
|
|
|
|
2023-12-20 16:55:58 +01:00
|
|
|
$socialiteUserWithSameEmail = new \Laravel\Socialite\Two\User;
|
|
|
|
$socialiteUserWithSameEmail->id = '666';
|
|
|
|
$socialiteUserWithSameEmail->name = 'socialiteUserWithSameEmail';
|
|
|
|
$socialiteUserWithSameEmail->email = 'other@example.com';
|
2023-12-14 15:39:14 +01:00
|
|
|
$socialiteUserWithSameEmail->nickname = self::USER_NICKNAME;
|
|
|
|
|
|
|
|
Socialite::shouldReceive('driver->user')
|
|
|
|
->andReturn($socialiteUserWithSameEmail);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/callback/github', ['driver' => 'github']);
|
|
|
|
|
|
|
|
$response->assertRedirect('/error?err=sso_email_already_used');
|
|
|
|
$this->assertDatabaseMissing('users', [
|
|
|
|
'oauth_id' => '666',
|
|
|
|
'oauth_provider' => self::USER_OAUTH_PROVIDER,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2024-03-08 15:05:49 +01:00
|
|
|
public function test_callback_redirects_to_error_when_registrations_are_closed()
|
2023-12-09 17:22:24 +01:00
|
|
|
{
|
|
|
|
Settings::set('disableRegistration', true);
|
2024-03-08 15:05:49 +01:00
|
|
|
Settings::set('keepSsoRegistrationEnabled', false);
|
2023-12-09 17:22:24 +01:00
|
|
|
|
2023-12-20 16:55:58 +01:00
|
|
|
$newSocialiteUser = new \Laravel\Socialite\Two\User;
|
|
|
|
$newSocialiteUser->id = 'rejected_id';
|
|
|
|
$newSocialiteUser->name = 'jane';
|
2023-12-09 17:22:24 +01:00
|
|
|
$newSocialiteUser->email = 'jane@provider.com';
|
|
|
|
|
|
|
|
Socialite::shouldReceive('driver->user')
|
|
|
|
->andReturn($newSocialiteUser);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/callback/github', ['driver' => 'github']);
|
|
|
|
|
2023-12-14 15:39:14 +01:00
|
|
|
$response->assertRedirect('/error?err=sso_no_register');
|
2023-12-09 17:22:24 +01:00
|
|
|
}
|
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2024-03-08 15:05:49 +01:00
|
|
|
public function test_callback_skips_registration_when_all_registrations_are_closed()
|
2023-12-09 17:22:24 +01:00
|
|
|
{
|
|
|
|
Settings::set('disableRegistration', true);
|
2024-03-08 15:05:49 +01:00
|
|
|
Settings::set('keepSsoRegistrationEnabled', false);
|
2023-12-09 17:22:24 +01:00
|
|
|
|
2023-12-20 16:55:58 +01:00
|
|
|
$newSocialiteUser = new \Laravel\Socialite\Two\User;
|
|
|
|
$newSocialiteUser->id = 'rejected_id';
|
|
|
|
$newSocialiteUser->name = 'jane';
|
2023-12-09 17:22:24 +01:00
|
|
|
$newSocialiteUser->email = 'jane@provider.com';
|
|
|
|
|
|
|
|
Socialite::shouldReceive('driver->user')
|
|
|
|
->andReturn($newSocialiteUser);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/callback/github', ['driver' => 'github']);
|
|
|
|
|
|
|
|
$this->assertDatabaseMissing('users', [
|
|
|
|
'oauth_id' => 'rejected_id',
|
|
|
|
'oauth_provider' => self::USER_OAUTH_PROVIDER,
|
|
|
|
]);
|
|
|
|
}
|
2024-03-08 15:05:49 +01:00
|
|
|
|
2024-06-26 14:29:13 +02:00
|
|
|
#[Test]
|
2024-03-08 15:05:49 +01:00
|
|
|
public function test_callback_registers_new_user_when_sso_registrations_are_enabled()
|
|
|
|
{
|
|
|
|
Settings::set('disableRegistration', true);
|
|
|
|
Settings::set('keepSsoRegistrationEnabled', true);
|
|
|
|
|
|
|
|
$newSocialiteUser = new \Laravel\Socialite\Two\User;
|
|
|
|
$newSocialiteUser->id = 'new_id';
|
|
|
|
$newSocialiteUser->name = 'jane';
|
|
|
|
$newSocialiteUser->email = 'jane@provider.com';
|
|
|
|
|
|
|
|
Socialite::shouldReceive('driver->user')
|
|
|
|
->andReturn($newSocialiteUser);
|
|
|
|
|
|
|
|
$response = $this->get('/socialite/callback/github', ['driver' => 'github']);
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('users', [
|
|
|
|
'oauth_id' => 'new_id',
|
|
|
|
'oauth_provider' => self::USER_OAUTH_PROVIDER,
|
|
|
|
'email' => 'jane@provider.com',
|
|
|
|
'is_admin' => 0,
|
|
|
|
]);
|
|
|
|
}
|
2023-12-09 17:22:24 +01:00
|
|
|
}
|