2022-09-30 13:56:11 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Services\Migrators;
|
|
|
|
|
2022-10-07 18:58:48 +02:00
|
|
|
use App\Services\Migrators\Migrator;
|
|
|
|
use Illuminate\Support\Collection;
|
|
|
|
use App\Models\TwoFAccount;
|
|
|
|
use Illuminate\Support\Facades\Log;
|
|
|
|
use Illuminate\Support\Arr;
|
|
|
|
use Illuminate\Support\Str;
|
|
|
|
use App\Exceptions\InvalidMigrationDataException;
|
2022-09-30 13:56:11 +02:00
|
|
|
|
2022-10-07 18:58:48 +02:00
|
|
|
class PlainTextMigrator extends Migrator
|
2022-09-30 13:56:11 +02:00
|
|
|
{
|
2022-10-07 18:58:48 +02:00
|
|
|
|
2022-09-30 13:56:11 +02:00
|
|
|
/**
|
2022-10-07 18:58:48 +02:00
|
|
|
* Convert migration data to a TwoFAccounts collection.
|
2022-09-30 13:56:11 +02:00
|
|
|
*
|
|
|
|
* @param mixed $migrationPayload
|
|
|
|
* @return \Illuminate\Support\Collection The converted accounts
|
|
|
|
*/
|
|
|
|
public function migrate(mixed $migrationPayload) : Collection
|
|
|
|
{
|
2022-10-07 18:58:48 +02:00
|
|
|
$otpauthURIs = preg_split('~\R~', $migrationPayload);
|
|
|
|
$otpauthURIs = Arr::where($otpauthURIs, function ($value, $key) {
|
|
|
|
return Str::startsWith($value, ['otpauth://totp/', 'otpauth://hotp/']);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (count($otpauthURIs) < 1) {
|
|
|
|
Log::error('No valid OtpAuth URI found in the migration');
|
|
|
|
throw new InvalidMigrationDataException('migration');
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($otpauthURIs as $key => $uri) {
|
|
|
|
|
|
|
|
try {
|
|
|
|
$twofaccounts[$key] = new TwoFAccount;
|
|
|
|
$twofaccounts[$key]->fillWithURI($uri);
|
|
|
|
}
|
|
|
|
catch (\Exception $exception) {
|
|
|
|
|
|
|
|
Log::error(sprintf('Cannot instanciate a TwoFAccount object with OTP parameters from imported item #%s', $key));
|
|
|
|
Log::error($exception->getMessage());
|
|
|
|
|
|
|
|
// The token failed to generate a valid account so we create a fake account to be returned.
|
|
|
|
$fakeAccount = new TwoFAccount();
|
|
|
|
$fakeAccount->id = -2;
|
|
|
|
$fakeAccount->otp_type = substr($uri, 10, 4);
|
|
|
|
// Only basic fields are filled to limit the risk of another exception.
|
2022-10-11 11:20:07 +02:00
|
|
|
$fakeAccount->account = __('twofaccounts.import.invalid_account');
|
|
|
|
$fakeAccount->service = filter_input(INPUT_GET, 'issuer', FILTER_SANITIZE_ENCODED) ?? __('twofaccounts.import.invalid_service');
|
2022-10-07 18:58:48 +02:00
|
|
|
// The secret field is used to pass the error, not very clean but will do the job for now.
|
|
|
|
$fakeAccount->secret = $exception->getMessage();
|
|
|
|
|
|
|
|
$twofaccounts[$key] = $fakeAccount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return collect($twofaccounts);
|
2022-09-30 13:56:11 +02:00
|
|
|
}
|
|
|
|
}
|