2022-07-05 10:10:24 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Tests\Feature\Models;
|
|
|
|
|
|
|
|
use App\Models\TwoFAccount;
|
2022-12-09 10:52:17 +01:00
|
|
|
use Tests\Data\OtpTestData;
|
2022-11-22 15:15:52 +01:00
|
|
|
use Tests\FeatureTestCase;
|
2022-12-09 10:52:17 +01:00
|
|
|
use Illuminate\Support\Facades\Storage;
|
|
|
|
use Illuminate\Http\Testing\FileFactory;
|
|
|
|
use Illuminate\Support\Facades\Http;
|
|
|
|
use App\Helpers\Helpers;
|
|
|
|
use Mockery\MockInterface;
|
|
|
|
use Tests\Data\HttpRequestTestData;
|
2022-07-05 10:10:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @covers \App\Models\TwoFAccount
|
|
|
|
*/
|
|
|
|
class TwoFAccountModelTest extends FeatureTestCase
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* App\Models\TwoFAccount $customTotpTwofaccount
|
|
|
|
*/
|
|
|
|
protected $customTotpTwofaccount;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* App\Models\TwoFAccount $customTotpTwofaccount
|
|
|
|
*/
|
|
|
|
protected $customHotpTwofaccount;
|
|
|
|
|
2022-12-09 10:52:17 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
const ICON_NAME = 'oDBngpjQaQAgLtHqGuYiPRqftCXv6Sj4hSAXARpA.png';
|
|
|
|
|
2022-07-05 10:10:24 +02:00
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
2022-12-09 10:52:17 +01:00
|
|
|
public function setUp(): void
|
2022-07-05 10:10:24 +02:00
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
|
2022-11-22 15:15:52 +01:00
|
|
|
$this->customTotpTwofaccount = new TwoFAccount;
|
2022-07-05 10:10:24 +02:00
|
|
|
$this->customTotpTwofaccount->legacy_uri = OtpTestData::TOTP_FULL_CUSTOM_URI;
|
2022-11-22 15:15:52 +01:00
|
|
|
$this->customTotpTwofaccount->service = OtpTestData::SERVICE;
|
|
|
|
$this->customTotpTwofaccount->account = OtpTestData::ACCOUNT;
|
|
|
|
$this->customTotpTwofaccount->icon = OtpTestData::ICON;
|
|
|
|
$this->customTotpTwofaccount->otp_type = 'totp';
|
|
|
|
$this->customTotpTwofaccount->secret = OtpTestData::SECRET;
|
|
|
|
$this->customTotpTwofaccount->digits = OtpTestData::DIGITS_CUSTOM;
|
|
|
|
$this->customTotpTwofaccount->algorithm = OtpTestData::ALGORITHM_CUSTOM;
|
|
|
|
$this->customTotpTwofaccount->period = OtpTestData::PERIOD_CUSTOM;
|
|
|
|
$this->customTotpTwofaccount->counter = null;
|
2022-07-05 10:10:24 +02:00
|
|
|
$this->customTotpTwofaccount->save();
|
|
|
|
|
2022-11-22 15:15:52 +01:00
|
|
|
$this->customHotpTwofaccount = new TwoFAccount;
|
2022-07-05 10:10:24 +02:00
|
|
|
$this->customHotpTwofaccount->legacy_uri = OtpTestData::HOTP_FULL_CUSTOM_URI;
|
2022-11-22 15:15:52 +01:00
|
|
|
$this->customHotpTwofaccount->service = OtpTestData::SERVICE;
|
|
|
|
$this->customHotpTwofaccount->account = OtpTestData::ACCOUNT;
|
|
|
|
$this->customHotpTwofaccount->icon = OtpTestData::ICON;
|
|
|
|
$this->customHotpTwofaccount->otp_type = 'hotp';
|
|
|
|
$this->customHotpTwofaccount->secret = OtpTestData::SECRET;
|
|
|
|
$this->customHotpTwofaccount->digits = OtpTestData::DIGITS_CUSTOM;
|
|
|
|
$this->customHotpTwofaccount->algorithm = OtpTestData::ALGORITHM_CUSTOM;
|
|
|
|
$this->customHotpTwofaccount->period = null;
|
|
|
|
$this->customHotpTwofaccount->counter = OtpTestData::COUNTER_CUSTOM;
|
2022-07-05 10:10:24 +02:00
|
|
|
$this->customHotpTwofaccount->save();
|
|
|
|
|
2022-11-22 15:15:52 +01:00
|
|
|
$this->customSteamTotpTwofaccount = new TwoFAccount;
|
2022-07-13 14:58:09 +02:00
|
|
|
$this->customSteamTotpTwofaccount->legacy_uri = OtpTestData::STEAM_TOTP_URI;
|
2022-11-22 15:15:52 +01:00
|
|
|
$this->customSteamTotpTwofaccount->service = OtpTestData::STEAM;
|
|
|
|
$this->customSteamTotpTwofaccount->account = OtpTestData::ACCOUNT;
|
|
|
|
$this->customSteamTotpTwofaccount->otp_type = 'steamtotp';
|
|
|
|
$this->customSteamTotpTwofaccount->secret = OtpTestData::STEAM_SECRET;
|
|
|
|
$this->customSteamTotpTwofaccount->digits = OtpTestData::DIGITS_STEAM;
|
|
|
|
$this->customSteamTotpTwofaccount->algorithm = OtpTestData::ALGORITHM_DEFAULT;
|
|
|
|
$this->customSteamTotpTwofaccount->period = OtpTestData::PERIOD_DEFAULT;
|
|
|
|
$this->customSteamTotpTwofaccount->counter = null;
|
2022-07-13 14:58:09 +02:00
|
|
|
$this->customSteamTotpTwofaccount->save();
|
2022-07-05 10:10:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-11-22 15:15:52 +01:00
|
|
|
* @test
|
2022-12-09 10:52:17 +01:00
|
|
|
*
|
|
|
|
* @runInSeparateProcess
|
|
|
|
* @preserveGlobalState disabled
|
2022-11-22 15:15:52 +01:00
|
|
|
*/
|
2022-07-05 10:10:24 +02:00
|
|
|
public function test_fill_with_custom_totp_uri_returns_correct_value()
|
|
|
|
{
|
2022-12-09 10:52:17 +01:00
|
|
|
$this->mock('alias:' . Helpers::class, function (MockInterface $helper) {
|
|
|
|
$helper->shouldReceive('getUniqueFilename')
|
|
|
|
->andReturn(self::ICON_NAME);
|
|
|
|
|
|
|
|
$helper->shouldReceive('isValidImage')
|
|
|
|
->andReturn(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
$file = (new FileFactory)->image(self::ICON_NAME, 10, 10);
|
|
|
|
|
|
|
|
Http::preventStrayRequests();
|
|
|
|
Http::fake([
|
|
|
|
'https://en.opensuse.org/images/4/44/Button-filled-colour.png' => Http::response($file->tempFile, 200),
|
|
|
|
]);
|
|
|
|
|
|
|
|
Storage::fake('imagesLink');
|
|
|
|
Storage::fake('icons');
|
|
|
|
|
2022-07-05 10:10:24 +02:00
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithURI(OtpTestData::TOTP_FULL_CUSTOM_URI);
|
|
|
|
|
2022-12-09 10:52:17 +01:00
|
|
|
Storage::disk('icons')->assertExists(self::ICON_NAME);
|
|
|
|
Storage::disk('imagesLink')->assertMissing(self::ICON_NAME);
|
|
|
|
|
2022-07-05 10:10:24 +02:00
|
|
|
$this->assertEquals('totp', $twofaccount->otp_type);
|
|
|
|
$this->assertEquals(OtpTestData::TOTP_FULL_CUSTOM_URI, $twofaccount->legacy_uri);
|
|
|
|
$this->assertEquals(OtpTestData::SERVICE, $twofaccount->service);
|
|
|
|
$this->assertEquals(OtpTestData::ACCOUNT, $twofaccount->account);
|
|
|
|
$this->assertEquals(OtpTestData::SECRET, $twofaccount->secret);
|
|
|
|
$this->assertEquals(OtpTestData::DIGITS_CUSTOM, $twofaccount->digits);
|
|
|
|
$this->assertEquals(OtpTestData::PERIOD_CUSTOM, $twofaccount->period);
|
|
|
|
$this->assertEquals(null, $twofaccount->counter);
|
|
|
|
$this->assertEquals(OtpTestData::ALGORITHM_CUSTOM, $twofaccount->algorithm);
|
2022-12-09 10:52:17 +01:00
|
|
|
$this->assertEquals(self::ICON_NAME, $twofaccount->icon);
|
2022-07-05 10:10:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_fill_with_basic_totp_uri_returns_default_value()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithURI(OtpTestData::TOTP_SHORT_URI);
|
|
|
|
|
|
|
|
$this->assertEquals('totp', $twofaccount->otp_type);
|
|
|
|
$this->assertEquals(OtpTestData::TOTP_SHORT_URI, $twofaccount->legacy_uri);
|
|
|
|
$this->assertEquals(OtpTestData::ACCOUNT, $twofaccount->account);
|
|
|
|
$this->assertEquals(null, $twofaccount->service);
|
|
|
|
$this->assertEquals(OtpTestData::SECRET, $twofaccount->secret);
|
|
|
|
$this->assertEquals(OtpTestData::DIGITS_DEFAULT, $twofaccount->digits);
|
|
|
|
$this->assertEquals(OtpTestData::PERIOD_DEFAULT, $twofaccount->period);
|
|
|
|
$this->assertEquals(null, $twofaccount->counter);
|
|
|
|
$this->assertEquals(OtpTestData::ALGORITHM_DEFAULT, $twofaccount->algorithm);
|
|
|
|
$this->assertEquals(null, $twofaccount->icon);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
2022-12-09 10:52:17 +01:00
|
|
|
*
|
|
|
|
* @runInSeparateProcess
|
|
|
|
* @preserveGlobalState disabled
|
2022-07-05 10:10:24 +02:00
|
|
|
*/
|
|
|
|
public function test_fill_with_custom_hotp_uri_returns_correct_value()
|
|
|
|
{
|
2022-12-09 10:52:17 +01:00
|
|
|
$this->mock('alias:' . Helpers::class, function (MockInterface $helper) {
|
|
|
|
$helper->shouldReceive('getUniqueFilename')
|
|
|
|
->andReturn(self::ICON_NAME);
|
|
|
|
|
|
|
|
$helper->shouldReceive('isValidImage')
|
|
|
|
->andReturn(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
$file = (new FileFactory)->image(self::ICON_NAME, 10, 10);
|
|
|
|
|
|
|
|
Http::preventStrayRequests();
|
|
|
|
Http::fake([
|
|
|
|
'https://en.opensuse.org/images/4/44/Button-filled-colour.png' => Http::response($file->tempFile, 200),
|
|
|
|
]);
|
|
|
|
|
|
|
|
Storage::fake('imagesLink');
|
|
|
|
Storage::fake('icons');
|
|
|
|
|
2022-07-05 10:10:24 +02:00
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithURI(OtpTestData::HOTP_FULL_CUSTOM_URI);
|
|
|
|
|
2022-12-09 10:52:17 +01:00
|
|
|
Storage::disk('icons')->assertExists(self::ICON_NAME);
|
|
|
|
Storage::disk('imagesLink')->assertMissing(self::ICON_NAME);
|
|
|
|
|
2022-07-05 10:10:24 +02:00
|
|
|
$this->assertEquals('hotp', $twofaccount->otp_type);
|
|
|
|
$this->assertEquals(OtpTestData::HOTP_FULL_CUSTOM_URI, $twofaccount->legacy_uri);
|
|
|
|
$this->assertEquals(OtpTestData::SERVICE, $twofaccount->service);
|
|
|
|
$this->assertEquals(OtpTestData::ACCOUNT, $twofaccount->account);
|
|
|
|
$this->assertEquals(OtpTestData::SECRET, $twofaccount->secret);
|
|
|
|
$this->assertEquals(OtpTestData::DIGITS_CUSTOM, $twofaccount->digits);
|
|
|
|
$this->assertEquals(null, $twofaccount->period);
|
|
|
|
$this->assertEquals(OtpTestData::COUNTER_CUSTOM, $twofaccount->counter);
|
|
|
|
$this->assertEquals(OtpTestData::ALGORITHM_CUSTOM, $twofaccount->algorithm);
|
2022-12-09 10:52:17 +01:00
|
|
|
$this->assertEquals(self::ICON_NAME, $twofaccount->icon);
|
2022-07-05 10:10:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_fill_with_basic_hotp_uri_returns_default_value()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithURI(OtpTestData::HOTP_SHORT_URI);
|
|
|
|
|
|
|
|
$this->assertEquals('hotp', $twofaccount->otp_type);
|
|
|
|
$this->assertEquals(OtpTestData::HOTP_SHORT_URI, $twofaccount->legacy_uri);
|
|
|
|
$this->assertEquals(null, $twofaccount->service);
|
|
|
|
$this->assertEquals(OtpTestData::ACCOUNT, $twofaccount->account);
|
|
|
|
$this->assertEquals(OtpTestData::SECRET, $twofaccount->secret);
|
|
|
|
$this->assertEquals(OtpTestData::DIGITS_DEFAULT, $twofaccount->digits);
|
|
|
|
$this->assertEquals(null, $twofaccount->period);
|
|
|
|
$this->assertEquals(OtpTestData::COUNTER_DEFAULT, $twofaccount->counter);
|
|
|
|
$this->assertEquals(OtpTestData::ALGORITHM_DEFAULT, $twofaccount->algorithm);
|
|
|
|
$this->assertEquals(null, $twofaccount->icon);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_filled_with_uri_persists_correct_values_to_db()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithURI(OtpTestData::TOTP_SHORT_URI);
|
|
|
|
$twofaccount->save();
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('twofaccounts', [
|
2022-11-22 15:15:52 +01:00
|
|
|
'otp_type' => 'totp',
|
|
|
|
'legacy_uri' => OtpTestData::TOTP_SHORT_URI,
|
|
|
|
'service' => null,
|
|
|
|
'account' => OtpTestData::ACCOUNT,
|
|
|
|
'secret' => OtpTestData::SECRET,
|
|
|
|
'digits' => OtpTestData::DIGITS_DEFAULT,
|
|
|
|
'period' => OtpTestData::PERIOD_DEFAULT,
|
|
|
|
'counter' => null,
|
|
|
|
'algorithm' => OtpTestData::ALGORITHM_DEFAULT,
|
|
|
|
'icon' => null,
|
2022-07-05 10:10:24 +02:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_fill_with_invalid_uri_returns_ValidationException()
|
|
|
|
{
|
|
|
|
$this->expectException(\Illuminate\Validation\ValidationException::class);
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithURI(OtpTestData::INVALID_OTPAUTH_URI);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_fill_with_uri_without_label_returns_ValidationException()
|
|
|
|
{
|
|
|
|
$this->expectException(\Illuminate\Validation\ValidationException::class);
|
|
|
|
$twofaccount = new TwoFAccount;
|
2022-11-22 15:15:52 +01:00
|
|
|
$twofaccount->fillWithURI('otpauth://totp/?secret=' . OtpTestData::SECRET);
|
2022-07-05 10:10:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_create_custom_totp_from_parameters_returns_correct_value()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_FULL_VALID_PARAMETERS_FOR_CUSTOM_TOTP);
|
|
|
|
|
|
|
|
$this->assertEquals('totp', $twofaccount->otp_type);
|
|
|
|
$this->assertEquals(OtpTestData::SERVICE, $twofaccount->service);
|
|
|
|
$this->assertEquals(OtpTestData::ACCOUNT, $twofaccount->account);
|
|
|
|
$this->assertEquals(OtpTestData::SECRET, $twofaccount->secret);
|
|
|
|
$this->assertEquals(OtpTestData::DIGITS_CUSTOM, $twofaccount->digits);
|
|
|
|
$this->assertEquals(OtpTestData::PERIOD_CUSTOM, $twofaccount->period);
|
|
|
|
$this->assertEquals(null, $twofaccount->counter);
|
|
|
|
$this->assertEquals(OtpTestData::ALGORITHM_CUSTOM, $twofaccount->algorithm);
|
2022-11-22 15:15:52 +01:00
|
|
|
$this->assertStringEndsWith('.png', $twofaccount->icon);
|
2022-07-05 10:10:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_create_basic_totp_from_parameters_returns_correct_value()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_MINIMUM_VALID_PARAMETERS_FOR_TOTP);
|
|
|
|
|
|
|
|
$this->assertEquals('totp', $twofaccount->otp_type);
|
|
|
|
$this->assertEquals(null, $twofaccount->service);
|
|
|
|
$this->assertEquals(OtpTestData::ACCOUNT, $twofaccount->account);
|
|
|
|
$this->assertEquals(OtpTestData::SECRET, $twofaccount->secret);
|
|
|
|
$this->assertEquals(OtpTestData::DIGITS_DEFAULT, $twofaccount->digits);
|
|
|
|
$this->assertEquals(OtpTestData::PERIOD_DEFAULT, $twofaccount->period);
|
|
|
|
$this->assertEquals(null, $twofaccount->counter);
|
|
|
|
$this->assertEquals(OtpTestData::ALGORITHM_DEFAULT, $twofaccount->algorithm);
|
|
|
|
$this->assertEquals(null, $twofaccount->icon);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_create_custom_hotp_from_parameters_returns_correct_value()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_FULL_VALID_PARAMETERS_FOR_CUSTOM_HOTP);
|
|
|
|
|
|
|
|
$this->assertEquals('hotp', $twofaccount->otp_type);
|
|
|
|
$this->assertEquals(OtpTestData::SERVICE, $twofaccount->service);
|
|
|
|
$this->assertEquals(OtpTestData::ACCOUNT, $twofaccount->account);
|
|
|
|
$this->assertEquals(OtpTestData::SECRET, $twofaccount->secret);
|
|
|
|
$this->assertEquals(OtpTestData::DIGITS_CUSTOM, $twofaccount->digits);
|
|
|
|
$this->assertEquals(null, $twofaccount->period);
|
|
|
|
$this->assertEquals(OtpTestData::COUNTER_CUSTOM, $twofaccount->counter);
|
|
|
|
$this->assertEquals(OtpTestData::ALGORITHM_CUSTOM, $twofaccount->algorithm);
|
2022-11-22 15:15:52 +01:00
|
|
|
$this->assertStringEndsWith('.png', $twofaccount->icon);
|
2022-07-05 10:10:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_create_basic_hotp_from_parameters_returns_correct_value()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_MINIMUM_VALID_PARAMETERS_FOR_HOTP);
|
|
|
|
|
|
|
|
$this->assertEquals('hotp', $twofaccount->otp_type);
|
|
|
|
$this->assertEquals(null, $twofaccount->service);
|
|
|
|
$this->assertEquals(OtpTestData::ACCOUNT, $twofaccount->account);
|
|
|
|
$this->assertEquals(OtpTestData::SECRET, $twofaccount->secret);
|
|
|
|
$this->assertEquals(OtpTestData::DIGITS_DEFAULT, $twofaccount->digits);
|
|
|
|
$this->assertEquals(null, $twofaccount->period);
|
|
|
|
$this->assertEquals(OtpTestData::COUNTER_DEFAULT, $twofaccount->counter);
|
|
|
|
$this->assertEquals(OtpTestData::ALGORITHM_DEFAULT, $twofaccount->algorithm);
|
|
|
|
$this->assertEquals(null, $twofaccount->icon);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_create_from_parameters_persists_correct_values_to_db()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_MINIMUM_VALID_PARAMETERS_FOR_TOTP);
|
|
|
|
$twofaccount->save();
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('twofaccounts', [
|
2022-11-22 15:15:52 +01:00
|
|
|
'otp_type' => 'totp',
|
|
|
|
'legacy_uri' => OtpTestData::TOTP_SHORT_URI,
|
|
|
|
'service' => null,
|
|
|
|
'account' => OtpTestData::ACCOUNT,
|
|
|
|
'secret' => OtpTestData::SECRET,
|
|
|
|
'digits' => OtpTestData::DIGITS_DEFAULT,
|
|
|
|
'period' => OtpTestData::PERIOD_DEFAULT,
|
|
|
|
'counter' => null,
|
|
|
|
'algorithm' => OtpTestData::ALGORITHM_DEFAULT,
|
|
|
|
'icon' => null,
|
2022-07-05 10:10:24 +02:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_create_from_unsupported_parameters_returns_unsupportedOtpTypeException()
|
|
|
|
{
|
|
|
|
$this->expectException(\App\Exceptions\UnsupportedOtpTypeException::class);
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_PARAMETERS_FOR_UNSUPPORTED_OTP_TYPE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_create_from_invalid_parameters_type_returns_InvalidOtpParameterException()
|
|
|
|
{
|
|
|
|
$this->expectException(\App\Exceptions\InvalidOtpParameterException::class);
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithOtpParameters([
|
2022-11-22 15:15:52 +01:00
|
|
|
'account' => OtpTestData::ACCOUNT,
|
|
|
|
'otp_type' => 'totp',
|
|
|
|
'digits' => 'notsupported',
|
2022-07-05 10:10:24 +02:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_create_from_invalid_parameters_returns_InvalidOtpParameterException()
|
|
|
|
{
|
|
|
|
$this->expectException(\App\Exceptions\InvalidOtpParameterException::class);
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithOtpParameters([
|
|
|
|
'account' => OtpTestData::ACCOUNT,
|
|
|
|
'otp_type' => 'totp',
|
|
|
|
'algorithm' => 'notsupported',
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_update_totp_returns_updated_model()
|
|
|
|
{
|
|
|
|
$twofaccount = $this->customTotpTwofaccount;
|
|
|
|
$twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_MINIMUM_VALID_PARAMETERS_FOR_TOTP);
|
|
|
|
|
|
|
|
$this->assertEquals('totp', $twofaccount->otp_type);
|
|
|
|
$this->assertEquals(null, $twofaccount->service);
|
|
|
|
$this->assertEquals(OtpTestData::ACCOUNT, $twofaccount->account);
|
|
|
|
$this->assertEquals(OtpTestData::SECRET, $twofaccount->secret);
|
|
|
|
$this->assertEquals(OtpTestData::DIGITS_DEFAULT, $twofaccount->digits);
|
|
|
|
$this->assertEquals(OtpTestData::PERIOD_DEFAULT, $twofaccount->period);
|
|
|
|
$this->assertEquals(null, $twofaccount->counter);
|
|
|
|
$this->assertEquals(OtpTestData::ALGORITHM_DEFAULT, $twofaccount->algorithm);
|
|
|
|
$this->assertEquals(null, $twofaccount->counter);
|
|
|
|
$this->assertEquals(null, $twofaccount->icon);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_update_hotp_returns_updated_model()
|
|
|
|
{
|
|
|
|
$twofaccount = $this->customTotpTwofaccount;
|
|
|
|
$twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_MINIMUM_VALID_PARAMETERS_FOR_HOTP);
|
|
|
|
|
|
|
|
$this->assertEquals('hotp', $twofaccount->otp_type);
|
|
|
|
$this->assertEquals(null, $twofaccount->service);
|
|
|
|
$this->assertEquals(OtpTestData::ACCOUNT, $twofaccount->account);
|
|
|
|
$this->assertEquals(OtpTestData::SECRET, $twofaccount->secret);
|
|
|
|
$this->assertEquals(OtpTestData::DIGITS_DEFAULT, $twofaccount->digits);
|
|
|
|
$this->assertEquals(null, $twofaccount->period);
|
|
|
|
$this->assertEquals(OtpTestData::COUNTER_DEFAULT, $twofaccount->counter);
|
|
|
|
$this->assertEquals(OtpTestData::ALGORITHM_DEFAULT, $twofaccount->algorithm);
|
|
|
|
$this->assertEquals(null, $twofaccount->counter);
|
|
|
|
$this->assertEquals(null, $twofaccount->icon);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_update_totp_persists_updated_model()
|
|
|
|
{
|
|
|
|
$twofaccount = $this->customTotpTwofaccount;
|
|
|
|
$twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_MINIMUM_VALID_PARAMETERS_FOR_TOTP);
|
|
|
|
$twofaccount->save();
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('twofaccounts', [
|
2022-11-22 15:15:52 +01:00
|
|
|
'otp_type' => 'totp',
|
|
|
|
'service' => null,
|
|
|
|
'account' => OtpTestData::ACCOUNT,
|
|
|
|
'secret' => OtpTestData::SECRET,
|
|
|
|
'digits' => OtpTestData::DIGITS_DEFAULT,
|
|
|
|
'period' => OtpTestData::PERIOD_DEFAULT,
|
|
|
|
'counter' => null,
|
|
|
|
'algorithm' => OtpTestData::ALGORITHM_DEFAULT,
|
|
|
|
'icon' => null,
|
2022-07-05 10:10:24 +02:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
2022-12-09 10:52:17 +01:00
|
|
|
*
|
|
|
|
* @runInSeparateProcess
|
|
|
|
* @preserveGlobalState disabled
|
2022-07-05 10:10:24 +02:00
|
|
|
*/
|
|
|
|
public function test_getOTP_for_totp_returns_the_same_password()
|
|
|
|
{
|
2022-12-09 10:52:17 +01:00
|
|
|
$this->mock('alias:' . Helpers::class, function (MockInterface $helper) {
|
|
|
|
$helper->shouldReceive('getUniqueFilename')
|
|
|
|
->andReturn(self::ICON_NAME);
|
|
|
|
|
|
|
|
$helper->shouldReceive('isValidImage')
|
|
|
|
->andReturn(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
Http::preventStrayRequests();
|
|
|
|
Http::fake([
|
|
|
|
'https://en.opensuse.org/images/4/44/Button-filled-colour.png' => Http::response(HttpRequestTestData::ICON_PNG, 200),
|
|
|
|
]);
|
|
|
|
|
|
|
|
Storage::fake('imagesLink');
|
|
|
|
Storage::fake('icons');
|
|
|
|
|
2022-07-05 10:10:24 +02:00
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
|
|
|
|
$otp_from_model = $this->customTotpTwofaccount->getOTP();
|
2022-11-22 15:15:52 +01:00
|
|
|
$otp_from_uri = $twofaccount->fillWithURI(OtpTestData::TOTP_FULL_CUSTOM_URI)->getOTP();
|
2022-07-05 10:10:24 +02:00
|
|
|
|
|
|
|
if ($otp_from_model->generated_at === $otp_from_uri->generated_at) {
|
|
|
|
$this->assertEquals($otp_from_model, $otp_from_uri);
|
|
|
|
}
|
|
|
|
|
2022-11-22 15:15:52 +01:00
|
|
|
$otp_from_model = $this->customTotpTwofaccount->getOTP();
|
2022-07-05 10:10:24 +02:00
|
|
|
$otp_from_parameters = $twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_FULL_VALID_PARAMETERS_FOR_CUSTOM_TOTP)->getOTP();
|
|
|
|
|
|
|
|
if ($otp_from_model->generated_at === $otp_from_parameters->generated_at) {
|
|
|
|
$this->assertEquals($otp_from_model, $otp_from_parameters);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
2022-12-09 10:52:17 +01:00
|
|
|
*
|
|
|
|
* @runInSeparateProcess
|
|
|
|
* @preserveGlobalState disabled
|
2022-07-05 10:10:24 +02:00
|
|
|
*/
|
|
|
|
public function test_getOTP_for_hotp_returns_the_same_password()
|
|
|
|
{
|
2022-12-09 10:52:17 +01:00
|
|
|
$this->mock('alias:' . Helpers::class, function (MockInterface $helper) {
|
|
|
|
$helper->shouldReceive('getUniqueFilename')
|
|
|
|
->andReturn(self::ICON_NAME);
|
|
|
|
|
|
|
|
$helper->shouldReceive('isValidImage')
|
|
|
|
->andReturn(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
Http::preventStrayRequests();
|
|
|
|
Http::fake([
|
|
|
|
'https://en.opensuse.org/images/4/44/Button-filled-colour.png' => Http::response(HttpRequestTestData::ICON_PNG, 200),
|
|
|
|
]);
|
|
|
|
|
|
|
|
Storage::fake('imagesLink');
|
|
|
|
Storage::fake('icons');
|
|
|
|
|
2022-07-05 10:10:24 +02:00
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
|
|
|
|
$otp_from_model = $this->customHotpTwofaccount->getOTP();
|
2022-11-22 15:15:52 +01:00
|
|
|
$otp_from_uri = $twofaccount->fillWithURI(OtpTestData::HOTP_FULL_CUSTOM_URI)->getOTP();
|
2022-07-05 10:10:24 +02:00
|
|
|
|
|
|
|
$this->assertEquals($otp_from_model, $otp_from_uri);
|
|
|
|
|
|
|
|
$otp_from_parameters = $twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_FULL_VALID_PARAMETERS_FOR_CUSTOM_HOTP)->getOTP();
|
|
|
|
|
|
|
|
$this->assertEquals($otp_from_model, $otp_from_parameters);
|
|
|
|
}
|
|
|
|
|
2022-07-13 14:58:09 +02:00
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_getOTP_for_steamtotp_returns_the_same_password()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
|
|
|
|
$otp_from_model = $this->customSteamTotpTwofaccount->getOTP();
|
2022-11-22 15:15:52 +01:00
|
|
|
$otp_from_uri = $twofaccount->fillWithURI(OtpTestData::STEAM_TOTP_URI)->getOTP();
|
2022-07-13 14:58:09 +02:00
|
|
|
|
|
|
|
if ($otp_from_model->generated_at === $otp_from_uri->generated_at) {
|
|
|
|
$this->assertEquals($otp_from_model, $otp_from_uri);
|
|
|
|
}
|
|
|
|
|
2022-11-22 15:15:52 +01:00
|
|
|
$otp_from_model = $this->customSteamTotpTwofaccount->getOTP();
|
2022-07-13 14:58:09 +02:00
|
|
|
$otp_from_parameters = $twofaccount->fillWithOtpParameters(OtpTestData::ARRAY_OF_FULL_VALID_PARAMETERS_FOR_STEAM_TOTP)->getOTP();
|
|
|
|
|
|
|
|
if ($otp_from_model->generated_at === $otp_from_parameters->generated_at) {
|
|
|
|
$this->assertEquals($otp_from_model, $otp_from_parameters);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-05 10:10:24 +02:00
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_getOTP_for_totp_with_invalid_secret_returns_InvalidSecretException()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
|
|
|
|
$this->expectException(\App\Exceptions\InvalidSecretException::class);
|
2022-11-22 15:15:52 +01:00
|
|
|
$otp_from_uri = $twofaccount->fillWithURI('otpauth://totp/' . OtpTestData::ACCOUNT . '?secret=0')->getOTP();
|
2022-07-05 10:10:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_getOTP_for_totp_with_undecipherable_secret_returns_UndecipherableException()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
|
|
|
|
$this->expectException(\App\Exceptions\UndecipherableException::class);
|
|
|
|
$otp_from_uri = $twofaccount->fillWithOtpParameters([
|
2022-11-22 15:15:52 +01:00
|
|
|
'account' => OtpTestData::ACCOUNT,
|
|
|
|
'otp_type' => 'totp',
|
|
|
|
'secret' => __('errors.indecipherable'),
|
2022-07-05 10:10:24 +02:00
|
|
|
])->getOTP();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_getURI_for_custom_totp_model_returns_uri()
|
|
|
|
{
|
|
|
|
$uri = $this->customTotpTwofaccount->getURI();
|
2022-11-22 15:15:52 +01:00
|
|
|
|
2022-07-05 10:10:24 +02:00
|
|
|
$this->assertStringContainsString('otpauth://totp/', $uri);
|
|
|
|
$this->assertStringContainsString(OtpTestData::SERVICE, $uri);
|
|
|
|
$this->assertStringContainsString(OtpTestData::ACCOUNT, $uri);
|
2022-11-22 15:15:52 +01:00
|
|
|
$this->assertStringContainsString('secret=' . OtpTestData::SECRET, $uri);
|
|
|
|
$this->assertStringContainsString('digits=' . OtpTestData::DIGITS_CUSTOM, $uri);
|
|
|
|
$this->assertStringContainsString('period=' . OtpTestData::PERIOD_CUSTOM, $uri);
|
|
|
|
$this->assertStringContainsString('algorithm=' . OtpTestData::ALGORITHM_CUSTOM, $uri);
|
2022-07-05 10:10:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_getURI_for_custom_hotp_model_returns_uri()
|
|
|
|
{
|
|
|
|
$uri = $this->customHotpTwofaccount->getURI();
|
2022-11-22 15:15:52 +01:00
|
|
|
|
2022-07-05 10:10:24 +02:00
|
|
|
$this->assertStringContainsString('otpauth://hotp/', $uri);
|
|
|
|
$this->assertStringContainsString(OtpTestData::SERVICE, $uri);
|
|
|
|
$this->assertStringContainsString(OtpTestData::ACCOUNT, $uri);
|
2022-11-22 15:15:52 +01:00
|
|
|
$this->assertStringContainsString('secret=' . OtpTestData::SECRET, $uri);
|
|
|
|
$this->assertStringContainsString('digits=' . OtpTestData::DIGITS_CUSTOM, $uri);
|
|
|
|
$this->assertStringContainsString('counter=' . OtpTestData::COUNTER_CUSTOM, $uri);
|
|
|
|
$this->assertStringContainsString('algorithm=' . OtpTestData::ALGORITHM_CUSTOM, $uri);
|
2022-07-05 10:10:24 +02:00
|
|
|
}
|
2022-12-09 10:52:17 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*
|
|
|
|
* @runInSeparateProcess
|
|
|
|
* @preserveGlobalState disabled
|
|
|
|
*/
|
|
|
|
public function test_fill_succeed_when_image_fetching_fails()
|
|
|
|
{
|
|
|
|
$this->mock('alias:' . Helpers::class, function (MockInterface $helper) {
|
|
|
|
$helper->shouldReceive('getUniqueFilename')
|
|
|
|
->andReturn(self::ICON_NAME);
|
|
|
|
});
|
|
|
|
|
|
|
|
Http::preventStrayRequests();
|
|
|
|
|
|
|
|
Storage::fake('imagesLink');
|
|
|
|
Storage::fake('icons');
|
|
|
|
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->fillWithURI(OtpTestData::TOTP_FULL_CUSTOM_URI);
|
|
|
|
|
|
|
|
Storage::disk('icons')->assertMissing(self::ICON_NAME);
|
|
|
|
Storage::disk('imagesLink')->assertMissing(self::ICON_NAME);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_saving_totp_without_period_set_default_one()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->service = OtpTestData::SERVICE;
|
|
|
|
$twofaccount->account = OtpTestData::ACCOUNT;
|
|
|
|
$twofaccount->otp_type = TwoFAccount::TOTP;
|
|
|
|
$twofaccount->secret = OtpTestData::SECRET;
|
|
|
|
|
|
|
|
$twofaccount->save();
|
|
|
|
|
|
|
|
$account = TwoFAccount::find($twofaccount->id);
|
|
|
|
|
|
|
|
$this->assertEquals(TwoFAccount::DEFAULT_PERIOD, $account->period);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_saving_hotp_without_counter_set_default_one()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->service = OtpTestData::SERVICE;
|
|
|
|
$twofaccount->account = OtpTestData::ACCOUNT;
|
|
|
|
$twofaccount->otp_type = TwoFAccount::HOTP;
|
|
|
|
$twofaccount->secret = OtpTestData::SECRET;
|
|
|
|
|
|
|
|
$twofaccount->save();
|
|
|
|
|
|
|
|
$account = TwoFAccount::find($twofaccount->id);
|
|
|
|
|
|
|
|
$this->assertEquals(TwoFAccount::DEFAULT_COUNTER, $account->counter);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_equals_returns_true()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->legacy_uri = OtpTestData::TOTP_FULL_CUSTOM_URI;
|
|
|
|
$twofaccount->service = OtpTestData::SERVICE;
|
|
|
|
$twofaccount->account = OtpTestData::ACCOUNT;
|
|
|
|
$twofaccount->icon = OtpTestData::ICON;
|
|
|
|
$twofaccount->otp_type = 'totp';
|
|
|
|
$twofaccount->secret = OtpTestData::SECRET;
|
|
|
|
$twofaccount->digits = OtpTestData::DIGITS_CUSTOM;
|
|
|
|
$twofaccount->algorithm = OtpTestData::ALGORITHM_CUSTOM;
|
|
|
|
$twofaccount->period = OtpTestData::PERIOD_CUSTOM;
|
|
|
|
$twofaccount->counter = null;
|
|
|
|
$twofaccount->save();
|
|
|
|
|
|
|
|
$this->assertTrue($twofaccount->equals($this->customTotpTwofaccount));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
*/
|
|
|
|
public function test_equals_returns_false()
|
|
|
|
{
|
|
|
|
$twofaccount = new TwoFAccount;
|
|
|
|
$twofaccount->legacy_uri = OtpTestData::TOTP_FULL_CUSTOM_URI;
|
|
|
|
$twofaccount->service = OtpTestData::SERVICE;
|
|
|
|
$twofaccount->account = OtpTestData::ACCOUNT;
|
|
|
|
$twofaccount->icon = OtpTestData::ICON;
|
|
|
|
$twofaccount->otp_type = 'totp';
|
|
|
|
$twofaccount->secret = OtpTestData::SECRET;
|
|
|
|
$twofaccount->digits = OtpTestData::DIGITS_CUSTOM;
|
|
|
|
$twofaccount->algorithm = OtpTestData::ALGORITHM_CUSTOM;
|
|
|
|
$twofaccount->period = OtpTestData::PERIOD_CUSTOM;
|
|
|
|
$twofaccount->counter = null;
|
|
|
|
$twofaccount->save();
|
|
|
|
|
|
|
|
$this->assertFalse($twofaccount->equals($this->customHotpTwofaccount));
|
|
|
|
}
|
2022-11-22 15:15:52 +01:00
|
|
|
}
|