diff --git a/app/Services/Migrators/GoogleAuthMigrator.php b/app/Services/Migrators/GoogleAuthMigrator.php index 2c4d0d7e..fcb15896 100644 --- a/app/Services/Migrators/GoogleAuthMigrator.php +++ b/app/Services/Migrators/GoogleAuthMigrator.php @@ -45,7 +45,7 @@ public function migrate(mixed $migrationPayload) : Collection $parameters['otp_type'] = GAuthValueMapping::OTP_TYPE[OtpType::name($otp_parameters->getType())]; $parameters['service'] = $otp_parameters->getIssuer(); $parameters['account'] = str_replace($parameters['service'] . ':', '', $otp_parameters->getName()); - $parameters['secret'] = Base32::encodeUpper($otp_parameters->getSecret()); + $parameters['secret'] = $this->toBase32($otp_parameters->getSecret()); $parameters['algorithm'] = GAuthValueMapping::ALGORITHM[Algorithm::name($otp_parameters->getAlgorithm())]; $parameters['digits'] = GAuthValueMapping::DIGIT_COUNT[DigitCount::name($otp_parameters->getDigits())]; $parameters['counter'] = $parameters['otp_type'] === TwoFAccount::HOTP ? $otp_parameters->getCounter() : null; @@ -73,4 +73,11 @@ public function migrate(mixed $migrationPayload) : Collection return collect($twofaccounts); } + + /** + * Encode into uppercase Base32 + */ + protected function toBase32(string $str) { + return Base32::encodeUpper($str); + } } diff --git a/app/Services/QrCodeService.php b/app/Services/QrCodeService.php index aa195cbc..9ce5cb72 100644 --- a/app/Services/QrCodeService.php +++ b/app/Services/QrCodeService.php @@ -29,7 +29,7 @@ public static function encode(string $data) Log::info('data encoded to QR code'); - return $qrcode->render($data); + return $qrcode->render('stringToEncode'); } /** diff --git a/phpunit.xml b/phpunit.xml index 8063dae4..a4a687a3 100644 --- a/phpunit.xml +++ b/phpunit.xml @@ -34,6 +34,7 @@ + diff --git a/tests/Api/v1/Controllers/GroupControllerTest.php b/tests/Api/v1/Controllers/GroupControllerTest.php index aadf3847..ad33aa6f 100644 --- a/tests/Api/v1/Controllers/GroupControllerTest.php +++ b/tests/Api/v1/Controllers/GroupControllerTest.php @@ -473,4 +473,20 @@ public function test_destroy_group_resets_user_preferences() $this->assertEquals(0, $this->user->preferences['defaultGroup']); $this->assertEquals(0, $this->user->preferences['activeGroup']); } + + /** + * @test + */ + public function test_twofaccount_is_released_on_group_destroy() + { + $this->actingAs($this->user, 'api-guard') + ->json('DELETE', '/api/v1/groups/' . $this->userGroupA->id) + ->assertNoContent(); + + $this->twofaccountA->refresh(); + $this->twofaccountB->refresh(); + + $this->assertNull($this->twofaccountA->group_id); + $this->assertNull($this->twofaccountB->group_id); + } } diff --git a/tests/Api/v1/Controllers/QrCodeControllerTest.php b/tests/Api/v1/Controllers/QrCodeControllerTest.php index 3a8d8d08..38f687a4 100644 --- a/tests/Api/v1/Controllers/QrCodeControllerTest.php +++ b/tests/Api/v1/Controllers/QrCodeControllerTest.php @@ -61,7 +61,7 @@ public function test_show_qrcode_returns_base64_image() ]) ->assertOk(); - $this->assertStringStartsWith('data:image/png;base64', $response->getData()->qrcode); + $this->assertStringStartsWith('data:image/svg+xml;base64', $response->getData()->qrcode); } /** diff --git a/tests/Feature/Services/QrCodeServiceTest.php b/tests/Feature/Services/QrCodeServiceTest.php index 80a8061d..d75a2376 100644 --- a/tests/Feature/Services/QrCodeServiceTest.php +++ b/tests/Feature/Services/QrCodeServiceTest.php @@ -17,7 +17,7 @@ class QrCodeServiceTest extends FeatureTestCase { private const STRING_TO_ENCODE = 'stringToEncode'; - private const STRING_ENCODED = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAMgAAADICAIAAAAiOjnJAAAABnRSTlMA/wD/AP83WBt9AAAACXBIWXMAAA7EAAAOxAGVKw4bAAADOklEQVR4nO3dMW7kMBAAwdPh/v/ldeBLFRCYBkm7KveaEBoMBhT1fD6fPzDt7+4F8DMJi4SwSAiLhLBICIuEsEgIi4SwSAiLhLBICIuEsEgIi4SwSPxb/YPneYp1jHs7Z7a6/tXzam+/P7WeXVafgx2LhLBICIuEsEgIi4SwSAiLxPIc682u9xOn5lKnzZNueZ5v7FgkhEVCWCSERUJYJIRFQlgkxuZYb6bmIlNznanzUrfPmer127FICIuEsEgIi4SwSAiLhLBI5HOs290y3zqNHYuEsEgIi4SwSAiLhLBICIuEOdZ/p71XeDs7FglhkRAWCWGREBYJYZEQFol8jnXa+aTT1rPqlvXbsUgIi4SwSAiLhLBICIuEsEiMzbFuOc9U34819b7hLc/zjR2LhLBICIuEsEgIi4SwSAiLxHPL+Z7a1HcP+WbHIiEsEsIiISwSwiIhLBLCIrE8x6rPCU2tpz5f9eaWc1r1Ou1YJIRFQlgkhEVCWCSERUJYJPLzWKd912/Xuavb3xM0x+IIwiIhLBLCIiEsEsIiISwSy/dj7ZpL1fOn+nxS/X/rc2ar7FgkhEVCWCSERUJYJIRFQlgk8u8VrjrtPb4p9VxtVf37diwSwiIhLBLCIiEsEsIiISwS+f1Yu+6dmlKvf8ppczs7FglhkRAWCWGREBYJYZEQFolf973C0+5bX7XrPcRVdiwSwiIhLBLCIiEsEsIiISwSx32vcMqu+9lP+7+7zmnZsUgIi4SwSAiLhLBICIuEsEiM3Y91y/cH699Z/f2357Y6l6qfg+8VcgRhkRAWCWGREBYJYZEQFon8nvf6XvJd6vNP9fcHvVfIlYRFQlgkhEVCWCSERUJYJI77XmHttPvQp9az6x78N3YsEsIiISwSwiIhLBLCIiEsEr9ujjXltHvCTvt+oh2LhLBICIuEsEgIi4SwSAiLRD7HOu19wDf1veqn3V9V/74di4SwSAiLhLBICIuEsEgIi8TYHOu080mrpt7LO+3+qqn75VfZsUgIi4SwSAiLhLBICIuEsEg8t5yX4i52LBLCIiEsEsIiISwSwiIhLBLCIiEsEsIiISwSwiIhLBLCIiEsEl+BmgChoSs9XAAAAABJRU5ErkJggg=='; + private const STRING_ENCODED = 'data:image/svg+xml;base64,<?xml version="1.0" encoding="UTF-8"?>
<svg xmlns="http://www.w3.org/2000/svg" class="qr-svg qrcode" viewBox="0 0 25 25" preserveAspectRatio="xMidYMid">
<path class="qr-data light qrcode" fill="#fff" d="M12 2 h1 v1 h-1Z M13 2 h1 v1 h-1Z M12 3 h1 v1 h-1Z M13 3 h1 v1 h-1Z M11 4 h1 v1 h-1Z M13 4 h1 v1 h-1Z M11 5 h1 v1 h-1Z M12 5 h1 v1 h-1Z M14 5 h1 v1 h-1Z M14 6 h1 v1 h-1Z M11 7 h1 v1 h-1Z M12 7 h1 v1 h-1Z M13 7 h1 v1 h-1Z M14 7 h1 v1 h-1Z M13 9 h1 v1 h-1Z M14 9 h1 v1 h-1Z M11 10 h1 v1 h-1Z M14 10 h1 v1 h-1Z M2 11 h1 v1 h-1Z M3 11 h1 v1 h-1Z M5 11 h1 v1 h-1Z M6 11 h1 v1 h-1Z M7 11 h1 v1 h-1Z M9 11 h1 v1 h-1Z M12 11 h1 v1 h-1Z M13 11 h1 v1 h-1Z M14 11 h1 v1 h-1Z M3 12 h1 v1 h-1Z M5 12 h1 v1 h-1Z M6 12 h1 v1 h-1Z M9 12 h1 v1 h-1Z M11 12 h1 v1 h-1Z M13 12 h1 v1 h-1Z M14 12 h1 v1 h-1Z M16 12 h1 v1 h-1Z M18 12 h1 v1 h-1Z M20 12 h1 v1 h-1Z M3 13 h1 v1 h-1Z M5 13 h1 v1 h-1Z M10 13 h1 v1 h-1Z M11 13 h1 v1 h-1Z M13 13 h1 v1 h-1Z M14 13 h1 v1 h-1Z M15 13 h1 v1 h-1Z M16 13 h1 v1 h-1Z M17 13 h1 v1 h-1Z M20 13 h1 v1 h-1Z M21 13 h1 v1 h-1Z M4 14 h1 v1 h-1Z M7 14 h1 v1 h-1Z M12 14 h1 v1 h-1Z M15 14 h1 v1 h-1Z M16 14 h1 v1 h-1Z M17 14 h1 v1 h-1Z M18 14 h1 v1 h-1Z M20 14 h1 v1 h-1Z M21 14 h1 v1 h-1Z M22 14 h1 v1 h-1Z M11 15 h1 v1 h-1Z M12 15 h1 v1 h-1Z M18 15 h1 v1 h-1Z M20 15 h1 v1 h-1Z M22 15 h1 v1 h-1Z M11 16 h1 v1 h-1Z M14 16 h1 v1 h-1Z M15 16 h1 v1 h-1Z M16 16 h1 v1 h-1Z M19 16 h1 v1 h-1Z M21 16 h1 v1 h-1Z M22 16 h1 v1 h-1Z M14 17 h1 v1 h-1Z M16 17 h1 v1 h-1Z M17 17 h1 v1 h-1Z M21 17 h1 v1 h-1Z M13 18 h1 v1 h-1Z M15 18 h1 v1 h-1Z M16 18 h1 v1 h-1Z M18 18 h1 v1 h-1Z M22 18 h1 v1 h-1Z M11 19 h1 v1 h-1Z M12 19 h1 v1 h-1Z M13 19 h1 v1 h-1Z M15 19 h1 v1 h-1Z M17 19 h1 v1 h-1Z M18 19 h1 v1 h-1Z M19 19 h1 v1 h-1Z M22 19 h1 v1 h-1Z M11 20 h1 v1 h-1Z M12 20 h1 v1 h-1Z M15 20 h1 v1 h-1Z M17 20 h1 v1 h-1Z M18 20 h1 v1 h-1Z M21 20 h1 v1 h-1Z M22 20 h1 v1 h-1Z M13 21 h1 v1 h-1Z M17 21 h1 v1 h-1Z M19 21 h1 v1 h-1Z M20 21 h1 v1 h-1Z M21 21 h1 v1 h-1Z M11 22 h1 v1 h-1Z
M13 22 h1 v1 h-1Z M14 22 h1 v1 h-1Z M18 22 h1 v1 h-1Z M19 22 h1 v1 h-1Z M21 22 h1 v1 h-1Z M22 22 h1 v1 h-1Z"/>
<path class="qr-finder light qrcode" fill="#fff" d="M3 3 h1 v1 h-1Z M4 3 h1 v1 h-1Z M5 3 h1 v1 h-1Z M6 3 h1 v1 h-1Z M7 3 h1 v1 h-1Z M17 3 h1 v1 h-1Z M18 3 h1 v1 h-1Z M19 3 h1 v1 h-1Z M20 3 h1 v1 h-1Z M21 3 h1 v1 h-1Z M3 4 h1 v1 h-1Z M7 4 h1 v1 h-1Z M17 4 h1 v1 h-1Z M21 4 h1 v1 h-1Z M3 5 h1 v1 h-1Z M7 5 h1 v1 h-1Z M17 5 h1 v1 h-1Z M21 5 h1 v1 h-1Z M3 6 h1 v1 h-1Z M7 6 h1 v1 h-1Z M17 6 h1 v1 h-1Z M21 6 h1 v1 h-1Z M3 7 h1 v1 h-1Z M4 7 h1 v1 h-1Z M5 7 h1 v1 h-1Z M6 7 h1 v1 h-1Z M7 7 h1 v1 h-1Z M17 7 h1 v1 h-1Z M18 7 h1 v1 h-1Z M19 7 h1 v1 h-1Z M20 7 h1 v1 h-1Z M21 7 h1 v1 h-1Z M3 17 h1 v1 h-1Z M4 17 h1 v1 h-1Z M5 17 h1 v1 h-1Z M6 17 h1 v1 h-1Z M7 17 h1 v1 h-1Z M3 18 h1 v1 h-1Z M7 18 h1 v1 h-1Z M3 19 h1 v1 h-1Z M7 19 h1 v1 h-1Z M3 20 h1 v1 h-1Z M7 20 h1 v1 h-1Z M3 21 h1 v1 h-1Z M4 21 h1 v1 h-1Z M5 21 h1 v1 h-1Z M6 21 h1 v1 h-1Z M7 21 h1 v1 h-1Z"/>
<path class="qr-separator light qrcode" fill="#fff" d="M9 2 h1 v1 h-1Z M15 2 h1 v1 h-1Z M9 3 h1 v1 h-1Z M15 3 h1 v1 h-1Z M9 4 h1 v1 h-1Z M15 4 h1 v1 h-1Z M9 5 h1 v1 h-1Z M15 5 h1 v1 h-1Z M9 6 h1 v1 h-1Z M15 6 h1 v1 h-1Z M9 7 h1 v1 h-1Z M15 7 h1 v1 h-1Z M9 8 h1 v1 h-1Z M15 8 h1 v1 h-1Z M2 9 h1 v1 h-1Z M3 9 h1 v1 h-1Z M4 9 h1 v1 h-1Z M5 9 h1 v1 h-1Z M6 9 h1 v1 h-1Z M7 9 h1 v1 h-1Z M8 9 h1 v1 h-1Z M9 9 h1 v1 h-1Z M15 9 h1 v1 h-1Z M16 9 h1 v1 h-1Z M17 9 h1 v1 h-1Z M18 9 h1 v1 h-1Z M19 9 h1 v1 h-1Z M20 9 h1 v1 h-1Z M21 9 h1 v1 h-1Z M22 9 h1 v1 h-1Z M2 15 h1 v1 h-1Z M3 15 h1 v1 h-1Z M4 15 h1 v1 h-1Z M5 15 h1 v1 h-1Z M6 15 h1 v1 h-1Z M7 15 h1 v1 h-1Z M8 15 h1 v1 h-1Z M9 15 h1 v1 h-1Z M9 16 h1 v1 h-1Z M9 17 h1 v1 h-1Z M9 18 h1 v1 h-1Z M9 19 h1 v1 h-1Z M9 20 h1 v1 h-1Z M9 21 h1 v1 h-1Z M9 22 h1 v1 h-1Z"/>
<path class="qr-timing light qrcode" fill="#fff" d="M11 8 h1 v1 h-1Z M13 8 h1 v1 h-1Z M8 11 h1 v1 h-1Z M8 13 h1 v1 h-1Z"/>
<path class="qr-format light qrcode" fill="#fff" d="M10 3 h1 v1 h-1Z M10 7 h1 v1 h-1Z M10 9 h1 v1 h-1Z M6 10 h1 v1 h-1Z M7 10 h1 v1 h-1Z M9 10 h1 v1 h-1Z M16 10 h1 v1 h-1Z M17 10 h1 v1 h-1Z M21 10 h1 v1 h-1Z M10 16 h1 v1 h-1Z M10 17 h1 v1 h-1Z M10 18 h1 v1 h-1Z"/>
<path class="qr-quietzone light qrcode" fill="#fff" d="M0 0 h1 v1 h-1Z M1 0 h1 v1 h-1Z M2 0 h1 v1 h-1Z M3 0 h1 v1 h-1Z M4 0 h1 v1 h-1Z M5 0 h1 v1 h-1Z M6 0 h1 v1 h-1Z M7 0 h1 v1 h-1Z M8 0 h1 v1 h-1Z M9 0 h1 v1 h-1Z M10 0 h1 v1 h-1Z M11 0 h1 v1 h-1Z M12 0 h1 v1 h-1Z M13 0 h1 v1 h-1Z M14 0 h1 v1 h-1Z M15 0 h1 v1 h-1Z M16 0 h1 v1 h-1Z M17 0 h1 v1 h-1Z M18 0 h1 v1 h-1Z M19 0 h1 v1 h-1Z M20 0 h1 v1 h-1Z M21 0 h1 v1 h-1Z M22 0 h1 v1 h-1Z M23 0 h1 v1 h-1Z M24 0 h1 v1 h-1Z M0 1 h1 v1 h-1Z M1 1 h1 v1 h-1Z M2 1 h1 v1 h-1Z M3 1 h1 v1 h-1Z M4 1 h1 v1 h-1Z M5 1 h1 v1 h-1Z M6 1 h1 v1 h-1Z M7 1 h1 v1 h-1Z M8 1 h1 v1 h-1Z M9 1 h1 v1 h-1Z M10 1 h1 v1 h-1Z M11 1 h1 v1 h-1Z M12 1 h1 v1 h-1Z M13 1 h1 v1 h-1Z M14 1 h1 v1 h-1Z M15 1 h1 v1 h-1Z M16 1 h1 v1 h-1Z M17 1 h1 v1 h-1Z M18 1 h1 v1 h-1Z M19 1 h1 v1 h-1Z M20 1 h1 v1 h-1Z M21 1 h1 v1 h-1Z M22 1 h1 v1 h-1Z M23 1 h1 v1 h-1Z M24 1 h1 v1 h-1Z M0 2 h1 v1 h-1Z M1 2 h1 v1 h-1Z M23 2 h1 v1 h-1Z M24 2 h1 v1 h-1Z M0 3 h1 v1 h-1Z M1 3 h1 v1 h-1Z M23 3 h1 v1 h-1Z M24 3 h1 v1 h-1Z M0 4 h1 v1 h-1Z M1 4 h1 v1 h-1Z M23 4 h1 v1 h-1Z M24 4 h1 v1 h-1Z M0 5 h1 v1 h-1Z M1 5 h1 v1 h-1Z M23 5 h1 v1 h-1Z M24 5 h1 v1 h-1Z M0 6 h1 v1 h-1Z M1 6 h1 v1 h-1Z M23 6 h1 v1 h-1Z M24 6 h1 v1 h-1Z M0 7 h1 v1 h-1Z M1 7 h1 v1 h-1Z M23 7 h1 v1 h-1Z M24 7 h1 v1 h-1Z M0 8 h1 v1 h-1Z M1 8 h1 v1 h-1Z M23 8 h1 v1 h-1Z M24 8 h1 v1 h-1Z M0 9 h1 v1 h-1Z M1 9 h1 v1 h-1Z M23 9 h1 v1 h-1Z M24 9 h1 v1 h-1Z M0 10 h1 v1 h-1Z M1 10 h1 v1 h-1Z M23 10 h1 v1 h-1Z M24 10 h1 v1 h-1Z M0 11 h1 v1 h-1Z M1 11 h1 v1 h-1Z M23 11 h1 v1 h-1Z M24 11 h1 v1 h-1Z M0 12 h1 v1 h-1Z M1 12 h1 v1 h-1Z M23 12 h1 v1 h-1Z M24 12 h1 v1 h-1Z M0 13 h1 v1 h-1Z M1 13 h1 v1 h-1Z M23 13 h1 v1 h-1Z M24 13 h1 v1 h-1Z M0 14 h1 v1 h-1Z M1 14 h1 v1 h-1Z
M23 14 h1 v1 h-1Z M24 14 h1 v1 h-1Z M0 15 h1 v1 h-1Z M1 15 h1 v1 h-1Z M23 15 h1 v1 h-1Z M24 15 h1 v1 h-1Z M0 16 h1 v1 h-1Z M1 16 h1 v1 h-1Z M23 16 h1 v1 h-1Z M24 16 h1 v1 h-1Z M0 17 h1 v1 h-1Z M1 17 h1 v1 h-1Z M23 17 h1 v1 h-1Z M24 17 h1 v1 h-1Z M0 18 h1 v1 h-1Z M1 18 h1 v1 h-1Z M23 18 h1 v1 h-1Z M24 18 h1 v1 h-1Z M0 19 h1 v1 h-1Z M1 19 h1 v1 h-1Z M23 19 h1 v1 h-1Z M24 19 h1 v1 h-1Z M0 20 h1 v1 h-1Z M1 20 h1 v1 h-1Z M23 20 h1 v1 h-1Z M24 20 h1 v1 h-1Z M0 21 h1 v1 h-1Z M1 21 h1 v1 h-1Z M23 21 h1 v1 h-1Z M24 21 h1 v1 h-1Z M0 22 h1 v1 h-1Z M1 22 h1 v1 h-1Z M23 22 h1 v1 h-1Z M24 22 h1 v1 h-1Z M0 23 h1 v1 h-1Z M1 23 h1 v1 h-1Z M2 23 h1 v1 h-1Z M3 23 h1 v1 h-1Z M4 23 h1 v1 h-1Z M5 23 h1 v1 h-1Z M6 23 h1 v1 h-1Z M7 23 h1 v1 h-1Z M8 23 h1 v1 h-1Z M9 23 h1 v1 h-1Z M10 23 h1 v1 h-1Z M11 23 h1 v1 h-1Z M12 23 h1 v1 h-1Z M13 23 h1 v1 h-1Z M14 23 h1 v1 h-1Z M15 23 h1 v1 h-1Z M16 23 h1 v1 h-1Z M17 23 h1 v1 h-1Z M18 23 h1 v1 h-1Z M19 23 h1 v1 h-1Z M20 23 h1 v1 h-1Z M21 23 h1 v1 h-1Z M22 23 h1 v1 h-1Z M23 23 h1 v1 h-1Z M24 23 h1 v1 h-1Z M0 24 h1 v1 h-1Z M1 24 h1 v1 h-1Z M2 24 h1 v1 h-1Z M3 24 h1 v1 h-1Z M4 24 h1 v1 h-1Z M5 24 h1 v1 h-1Z M6 24 h1 v1 h-1Z M7 24 h1 v1 h-1Z M8 24 h1 v1 h-1Z M9 24 h1 v1 h-1Z M10 24 h1 v1 h-1Z M11 24 h1 v1 h-1Z M12 24 h1 v1 h-1Z M13 24 h1 v1 h-1Z M14 24 h1 v1 h-1Z M15 24 h1 v1 h-1Z M16 24 h1 v1 h-1Z M17 24 h1 v1 h-1Z M18 24 h1 v1 h-1Z M19 24 h1 v1 h-1Z M20 24 h1 v1 h-1Z M21 24 h1 v1 h-1Z M22 24 h1 v1 h-1Z M23 24 h1 v1 h-1Z M24 24 h1 v1 h-1Z"/>
<path class="qr-darkmodule dark qrcode" fill="#000" d="M10 15 h1 v1 h-1Z"/>
<path class="qr-data-dark dark qrcode" fill="#000" d="M11 2 h1 v1 h-1Z M14 2 h1 v1 h-1Z M11 3 h1 v1 h-1Z M14 3 h1 v1 h-1Z M12 4 h1 v1 h-1Z M14 4 h1 v1 h-1Z M13 5 h1 v1 h-1Z M11 6 h1 v1 h-1Z M12 6 h1 v1 h-1Z M13 6 h1 v1 h-1Z M11 9 h1 v1 h-1Z M12 9 h1 v1 h-1Z M12 10 h1 v1 h-1Z M13 10 h1 v1 h-1Z M4 11 h1 v1 h-1Z M10 11 h1 v1 h-1Z M11 11 h1 v1 h-1Z M15 11 h1 v1 h-1Z M16 11 h1 v1 h-1Z M17 11 h1 v1 h-1Z M18 11 h1 v1 h-1Z M19 11 h1 v1 h-1Z M20 11 h1 v1 h-1Z M21 11 h1 v1 h-1Z M22 11 h1 v1 h-1Z M2 12 h1 v1 h-1Z M4 12 h1 v1 h-1Z M7 12 h1 v1 h-1Z M10 12 h1 v1 h-1Z M12 12 h1 v1 h-1Z M15 12 h1 v1 h-1Z M17 12 h1 v1 h-1Z M19 12 h1 v1 h-1Z M21 12 h1 v1 h-1Z M22 12 h1 v1 h-1Z M2 13 h1 v1 h-1Z M4 13 h1 v1 h-1Z M6 13 h1 v1 h-1Z M7 13 h1 v1 h-1Z M9 13 h1 v1 h-1Z M12 13 h1 v1 h-1Z M18 13 h1 v1 h-1Z M19 13 h1 v1 h-1Z M22 13 h1 v1 h-1Z M2 14 h1 v1 h-1Z M3 14 h1 v1 h-1Z M5 14 h1 v1 h-1Z M6 14 h1 v1 h-1Z M9 14 h1 v1 h-1Z M10 14 h1 v1 h-1Z M11 14 h1 v1 h-1Z M13 14 h1 v1 h-1Z M14 14 h1 v1 h-1Z M19 14 h1 v1 h-1Z M13 15 h1 v1 h-1Z M14 15 h1 v1 h-1Z M15 15 h1 v1 h-1Z M16 15 h1 v1 h-1Z M17 15 h1 v1 h-1Z M19 15 h1 v1 h-1Z M21 15 h1 v1 h-1Z M12 16 h1 v1 h-1Z M13 16 h1 v1 h-1Z M17 16 h1 v1 h-1Z M18 16 h1 v1 h-1Z M20 16 h1 v1 h-1Z M11 17 h1 v1 h-1Z M12 17 h1 v1 h-1Z M13 17 h1 v1 h-1Z M15 17 h1 v1 h-1Z M18 17 h1 v1 h-1Z M19 17 h1 v1 h-1Z M20 17 h1 v1 h-1Z M22 17 h1 v1 h-1Z M11 18 h1 v1 h-1Z M12 18 h1 v1 h-1Z M14 18 h1 v1 h-1Z M17 18 h1 v1 h-1Z M19 18 h1 v1 h-1Z M20 18 h1 v1 h-1Z M21 18 h1 v1 h-1Z M14 19 h1 v1 h-1Z M16 19 h1 v1 h-1Z M20 19 h1 v1 h-1Z M21 19 h1 v1 h-1Z M13 20 h1 v1 h-1Z M14 20 h1 v1 h-1Z M16 20 h1 v1 h-1Z M19 20 h1 v1 h-1Z M20 20 h1 v1 h-1Z M11 21 h1 v1 h-1Z M12 21 h1 v1 h-1Z M14 21 h1 v1 h-1Z M15 21 h1 v1 h-1Z M16 21 h1 v1 h-1Z M18 21 h1 v1 h-1Z M22 21 h1 v1 h-1Z M12 22 h1 v1 h-1Z M15 22 h1 v1 h-1Z M16 22 h1 v1 h-1Z
M17 22 h1 v1 h-1Z M20 22 h1 v1 h-1Z"/>
<path class="qr-finder-dark dark qrcode" fill="#000" d="M2 2 h1 v1 h-1Z M3 2 h1 v1 h-1Z M4 2 h1 v1 h-1Z M5 2 h1 v1 h-1Z M6 2 h1 v1 h-1Z M7 2 h1 v1 h-1Z M8 2 h1 v1 h-1Z M16 2 h1 v1 h-1Z M17 2 h1 v1 h-1Z M18 2 h1 v1 h-1Z M19 2 h1 v1 h-1Z M20 2 h1 v1 h-1Z M21 2 h1 v1 h-1Z M22 2 h1 v1 h-1Z M2 3 h1 v1 h-1Z M8 3 h1 v1 h-1Z M16 3 h1 v1 h-1Z M22 3 h1 v1 h-1Z M2 4 h1 v1 h-1Z M8 4 h1 v1 h-1Z M16 4 h1 v1 h-1Z M22 4 h1 v1 h-1Z M2 5 h1 v1 h-1Z M8 5 h1 v1 h-1Z M16 5 h1 v1 h-1Z M22 5 h1 v1 h-1Z M2 6 h1 v1 h-1Z M8 6 h1 v1 h-1Z M16 6 h1 v1 h-1Z M22 6 h1 v1 h-1Z M2 7 h1 v1 h-1Z M8 7 h1 v1 h-1Z M16 7 h1 v1 h-1Z M22 7 h1 v1 h-1Z M2 8 h1 v1 h-1Z M3 8 h1 v1 h-1Z M4 8 h1 v1 h-1Z M5 8 h1 v1 h-1Z M6 8 h1 v1 h-1Z M7 8 h1 v1 h-1Z M8 8 h1 v1 h-1Z M16 8 h1 v1 h-1Z M17 8 h1 v1 h-1Z M18 8 h1 v1 h-1Z M19 8 h1 v1 h-1Z M20 8 h1 v1 h-1Z M21 8 h1 v1 h-1Z M22 8 h1 v1 h-1Z M2 16 h1 v1 h-1Z M3 16 h1 v1 h-1Z M4 16 h1 v1 h-1Z M5 16 h1 v1 h-1Z M6 16 h1 v1 h-1Z M7 16 h1 v1 h-1Z M8 16 h1 v1 h-1Z M2 17 h1 v1 h-1Z M8 17 h1 v1 h-1Z M2 18 h1 v1 h-1Z M8 18 h1 v1 h-1Z M2 19 h1 v1 h-1Z M8 19 h1 v1 h-1Z M2 20 h1 v1 h-1Z M8 20 h1 v1 h-1Z M2 21 h1 v1 h-1Z M8 21 h1 v1 h-1Z M2 22 h1 v1 h-1Z M3 22 h1 v1 h-1Z M4 22 h1 v1 h-1Z M5 22 h1 v1 h-1Z M6 22 h1 v1 h-1Z M7 22 h1 v1 h-1Z M8 22 h1 v1 h-1Z"/>
<path class="qr-timing-dark dark qrcode" fill="#000" d="M10 8 h1 v1 h-1Z M12 8 h1 v1 h-1Z M14 8 h1 v1 h-1Z M8 10 h1 v1 h-1Z M8 12 h1 v1 h-1Z M8 14 h1 v1 h-1Z"/>
<path class="qr-format-dark dark qrcode" fill="#000" d="M10 2 h1 v1 h-1Z M10 4 h1 v1 h-1Z M10 5 h1 v1 h-1Z M10 6 h1 v1 h-1Z M2 10 h1 v1 h-1Z M3 10 h1 v1 h-1Z M4 10 h1 v1 h-1Z M5 10 h1 v1 h-1Z M10 10 h1 v1 h-1Z M15 10 h1 v1 h-1Z M18 10 h1 v1 h-1Z M19 10 h1 v1 h-1Z M20 10 h1 v1 h-1Z M22 10 h1 v1 h-1Z M10 19 h1 v1 h-1Z M10 20 h1 v1 h-1Z M10 21 h1 v1 h-1Z M10 22 h1 v1 h-1Z"/>
<path class="qr-finder-dot dark qrcode" fill="#000" d="M4 4 h1 v1 h-1Z M5 4 h1 v1 h-1Z M6 4 h1 v1 h-1Z M18 4 h1 v1 h-1Z M19 4 h1 v1 h-1Z M20 4 h1 v1 h-1Z M4 5 h1 v1 h-1Z M5 5 h1 v1 h-1Z M6 5 h1 v1 h-1Z M18 5 h1 v1 h-1Z M19 5 h1 v1 h-1Z M20 5 h1 v1 h-1Z M4 6 h1 v1 h-1Z M5 6 h1 v1 h-1Z M6 6 h1 v1 h-1Z M18 6 h1 v1 h-1Z M19 6 h1 v1 h-1Z M20 6 h1 v1 h-1Z M4 18 h1 v1 h-1Z M5 18 h1 v1 h-1Z M6 18 h1 v1 h-1Z M4 19 h1 v1 h-1Z M5 19 h1 v1 h-1Z M6 19 h1 v1 h-1Z M4 20 h1 v1 h-1Z M5 20 h1 v1 h-1Z M6 20 h1 v1 h-1Z"/>
</svg>
'; private const DECODED_IMAGE = 'otpauth://totp/test@test.com?secret=A4GRFHVIRBGY7UIW'; diff --git a/tests/Unit/Listeners/DissociateTwofaccountFromGroupTest.php b/tests/Unit/Listeners/DissociateTwofaccountFromGroupTest.php index 447d1b69..55e96ff5 100644 --- a/tests/Unit/Listeners/DissociateTwofaccountFromGroupTest.php +++ b/tests/Unit/Listeners/DissociateTwofaccountFromGroupTest.php @@ -4,14 +4,8 @@ use App\Events\GroupDeleting; use App\Listeners\DissociateTwofaccountFromGroup; -use App\Models\Group; -use App\Models\TwoFAccount; use Illuminate\Support\Facades\Event; -use Mockery\MockInterface; use PHPUnit\Framework\Attributes\CoversClass; -use PHPUnit\Framework\Attributes\PreserveGlobalState; -use PHPUnit\Framework\Attributes\RequiresPhp; -use PHPUnit\Framework\Attributes\RunInSeparateProcess; use Tests\TestCase; /** @@ -20,27 +14,6 @@ #[CoversClass(DissociateTwofaccountFromGroup::class)] class DissociateTwofaccountFromGroupTest extends TestCase { - /** - * @test - */ - #[RunInSeparateProcess] - #[PreserveGlobalState(false)] - #[RequiresPhp('< 8.3.0')] - public function test_twofaccount_is_released_on_group_deletion() - { - $this->mock('alias:' . TwoFAccount::class, function (MockInterface $twoFAccount) { - $twoFAccount->shouldReceive('where->update') - ->once() - ->andReturn(1); - }); - - $group = Group::factory()->make(); - $event = new GroupDeleting($group); - $listener = new DissociateTwofaccountFromGroup(); - - $this->assertNull($listener->handle($event)); - } - /** * @test */ diff --git a/tests/Unit/MigratorTest.php b/tests/Unit/MigratorTest.php index 4a8459d7..0fc7bae4 100644 --- a/tests/Unit/MigratorTest.php +++ b/tests/Unit/MigratorTest.php @@ -18,12 +18,8 @@ use Illuminate\Support\Facades\Storage; use Mockery; use Mockery\MockInterface; -use ParagonIE\ConstantTime\Base32; use PHPUnit\Framework\Attributes\CoversClass; use PHPUnit\Framework\Attributes\DataProvider; -use PHPUnit\Framework\Attributes\PreserveGlobalState; -use PHPUnit\Framework\Attributes\RequiresPhp; -use PHPUnit\Framework\Attributes\RunInSeparateProcess; use PHPUnit\Framework\Attributes\UsesClass; use Tests\Data\MigrationTestData; use Tests\Data\OtpTestData; @@ -142,6 +138,20 @@ public function setUp() : void $this->fakeTwofaccount->id = TwoFAccount::FAKE_ID; } + /** + * Clean up the testing environment before the next test. + * + * @return void + * + * @throws \Mockery\Exception\InvalidCountException + */ + protected function tearDown() : void + { + $this->forgetMock(SettingService::class); + + parent::tearDown(); + } + /** * @test */ @@ -332,17 +342,14 @@ public static function migrationWithInvalidAccountsProvider() /** * @test */ - #[RunInSeparateProcess] - #[PreserveGlobalState(false)] - #[RequiresPhp('< 8.3.0')] public function test_migrate_gauth_returns_fake_accounts() { - $this->mock('alias:' . Base32::class, function (MockInterface $baseEncoder) { - $baseEncoder->shouldReceive('encodeUpper') + $migrator = $this->partialMock(GoogleAuthMigrator::class, function (MockInterface $migrator) { + $migrator->shouldAllowMockingProtectedMethods()->shouldReceive('toBase32') ->andThrow(new \Exception()); }); - $migrator = new GoogleAuthMigrator(); + /** @disregard Undefined function */ $accounts = $migrator->migrate(MigrationTestData::GOOGLE_AUTH_MIGRATION_URI); $this->assertContainsOnlyInstancesOf(TwoFAccount::class, $accounts); @@ -352,6 +359,8 @@ public function test_migrate_gauth_returns_fake_accounts() // in the migration payload) so we do not use get() to retrieve items $this->assertEquals($this->fakeTwofaccount->id, $accounts->first()->id); $this->assertEquals($this->fakeTwofaccount->id, $accounts->last()->id); + + $this->forgetMock(GoogleAuthMigrator::class); } /** @@ -548,9 +557,4 @@ public static function encryptedMigrationDataProvider() ], ]; } - - protected function tearDown() : void - { - Mockery::close(); - } } diff --git a/tests/Unit/TwoFAccountModelTest.php b/tests/Unit/TwoFAccountModelTest.php index 50c0798c..c3501b61 100644 --- a/tests/Unit/TwoFAccountModelTest.php +++ b/tests/Unit/TwoFAccountModelTest.php @@ -11,9 +11,6 @@ use Mockery\MockInterface; use PHPUnit\Framework\Attributes\CoversClass; use PHPUnit\Framework\Attributes\DataProvider; -use PHPUnit\Framework\Attributes\PreserveGlobalState; -use PHPUnit\Framework\Attributes\RequiresPhp; -use PHPUnit\Framework\Attributes\RunInSeparateProcess; use Tests\ModelTestCase; /** @@ -63,6 +60,7 @@ public function test_sensitive_attributes_are_stored_encrypted(string $attribute ]); $this->assertEquals('STRING==', Crypt::decryptString($twofaccount->getAttributes()[$attribute])); + $this->forgetMock(SettingService::class); } /** @@ -98,6 +96,7 @@ public function test_sensitive_attributes_are_returned_clear(string $attribute) $twofaccount = TwoFAccount::factory()->make(); $this->assertEquals($twofaccount->getAttributes()[$attribute], $twofaccount->$attribute); + $this->forgetMock(SettingService::class); } /** @@ -118,14 +117,12 @@ public function test_indecipherable_attributes_returns_masked_value(string $attr $twofaccount = TwoFAccount::factory()->make(); $this->assertEquals(__('errors.indecipherable'), $twofaccount->$attribute); + $this->forgetMock(SettingService::class); } /** * @test */ - #[RunInSeparateProcess] - #[PreserveGlobalState(false)] - #[RequiresPhp('< 8.3.0')] public function test_secret_is_uppercased_and_padded_at_setup() { $settingService = $this->mock(SettingService::class, function (MockInterface $settingService) { @@ -134,7 +131,7 @@ public function test_secret_is_uppercased_and_padded_at_setup() ->andReturn(false); }); - $helpers = $this->mock('alias:' . Helpers::class, function (MockInterface $helpers) { + $helpers = $this->mock(Helpers::class, function (MockInterface $helpers) { $helpers->shouldReceive('PadToBase32Format') ->andReturn('YYYY===='); }); @@ -144,6 +141,7 @@ public function test_secret_is_uppercased_and_padded_at_setup() ]); $this->assertEquals('YYYY====', $twofaccount->secret); + $this->forgetMock(SettingService::class); } /**