Code Coverage
 
Lines
Functions and Methods
Classes and Traits
Total
14.29% covered (danger)
14.29%
4 / 28
50.00% covered (danger)
50.00%
1 / 2
CRAP
0.00% covered (danger)
0.00%
0 / 1
VerifyOtpInteractor
14.29% covered (danger)
14.29%
4 / 28
50.00% covered (danger)
50.00%
1 / 2
28.67
0.00% covered (danger)
0.00%
0 / 1
 supports
100.00% covered (success)
100.00%
4 / 4
100.00% covered (success)
100.00%
1 / 1
3
 interactWith
0.00% covered (danger)
0.00%
0 / 24
0.00% covered (danger)
0.00%
0 / 1
12
1<?php
2namespace Apie\Console\Helpers;
3
4use Apie\Core\Context\ApieContext;
5use Apie\Core\ContextConstants;
6use Apie\Core\Metadata\MetadataInterface;
7use Apie\OtpValueObjects\VerifyOTP;
8use LogicException;
9use ReflectionClass;
10use Symfony\Component\Console\Helper\HelperSet;
11use Symfony\Component\Console\Helper\QuestionHelper;
12use Symfony\Component\Console\Input\InputInterface;
13use Symfony\Component\Console\Output\OutputInterface;
14use Symfony\Component\Console\Question\Question;
15
16final class VerifyOtpInteractor implements InputInteractorInterface
17{
18    public function supports(MetadataInterface $metadata): bool
19    {
20        $class = $metadata->toClass();
21        if (!$class) {
22            return false;
23        }
24        return class_exists(VerifyOTP::class) && $class->isSubclassOf(VerifyOTP::class);
25    }
26    public function interactWith(
27        MetadataInterface $metadata,
28        HelperSet $helperSet,
29        InputInterface $input,
30        OutputInterface $output,
31        ApieContext $context
32    ): mixed {
33        $helper = $helperSet->get('question');
34        assert($helper instanceof QuestionHelper);
35
36        $resource = $context->getContext(ContextConstants::RESOURCE);
37        $otpClass = $metadata->toClass();
38        assert($otpClass !== null);
39        $property = $otpClass->getMethod('getOtpReference')->invoke(null);
40        $label = $otpClass->getMethod('getOtpLabel')->invoke(null, $resource);
41        $secret = $property->getValue($resource);
42        assert(is_callable([$secret, 'verify']));
43
44        $output->writeln('Open your authenticator application and add this code manually:');
45        $output->writeln('Name: ' . $label);
46        $output->writeln('Secret code: ' . $secret);
47        $output->writeln('Type: ' . preg_replace('/Secret$/', '', (new ReflectionClass($secret))->getShortName()));
48
49        if (is_callable([$secret, 'getQrCodeUri'])) {
50            $output->writeln('Or scan this QR code: ' . $secret->getQrCodeUri($label));
51        }
52
53        $question = new Question('Please enter the code shown in your authenticator application: ');
54        $question->setValidator(function ($input) use ($metadata, $secret) {
55            $otpInstance = $metadata->toClass()->getMethod('fromNative')->invoke(null, $input);
56            assert($otpInstance instanceof VerifyOTP);
57            if (!$secret->verify($otpInstance)) {
58                throw new LogicException('Code is not valid!');
59            }
60            return $otpInstance->toNative();
61        });
62        return (string) $helper->ask($input, $output, $question);
63    }
64}