Completed
Push — master ( 545d02...a22894 )
by thomas
14:41
created

CompactSignatureSerializer::__construct()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 4
Code Lines 2

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 3
CRAP Score 1.0156

Importance

Changes 0
Metric Value
cc 1
eloc 2
nc 1
nop 1
dl 0
loc 4
ccs 3
cts 4
cp 0.75
crap 1.0156
rs 10
c 0
b 0
f 0
1
<?php
2
3
namespace BitWasp\Bitcoin\Crypto\EcAdapter\Impl\PhpEcc\Serializer\Signature;
4
5
use BitWasp\Bitcoin\Crypto\EcAdapter\Impl\PhpEcc\Adapter\EcAdapter;
6
use BitWasp\Bitcoin\Crypto\EcAdapter\Serializer\Signature\CompactSignatureSerializerInterface;
7
use BitWasp\Bitcoin\Crypto\EcAdapter\Signature\CompactSignatureInterface;
8
use BitWasp\Buffertools\BufferInterface;
9
use BitWasp\Buffertools\Parser;
10
use BitWasp\Buffertools\Exceptions\ParserOutOfRange;
11
use BitWasp\Bitcoin\Crypto\EcAdapter\Impl\PhpEcc\Signature\CompactSignature;
12
use BitWasp\Buffertools\TemplateFactory;
13
14
class CompactSignatureSerializer implements CompactSignatureSerializerInterface
15
{
16
17
    /**
18
     * @var EcAdapter
19
     */
20
    private $ecAdapter;
21
22
    /**
23
     * @param EcAdapter $adapter
24
     */
25 11
    public function __construct(EcAdapter $adapter)
26
    {
27 11
        $this->ecAdapter = $adapter;
28 11
    }
29
30
    /**
31
     * @return \BitWasp\Buffertools\Template
32
     */
33 8
    private function getTemplate()
34
    {
35 8
        return (new TemplateFactory())
36 8
            ->uint8()
37 8
            ->uint256()
38 8
            ->uint256()
39 8
            ->getTemplate();
40
    }
41
42
    /**
43
     * @param CompactSignature $signature
44
     * @return BufferInterface
45
     */
46 6
    private function doSerialize(CompactSignature $signature)
47
    {
48 6
        return $this->getTemplate()->write([
49 6
            $signature->getFlags(),
50 6
            $signature->getR(),
51 6
            $signature->getS()
52
        ]);
53
    }
54
55
    /**
56
     * @param CompactSignatureInterface $signature
57
     * @return BufferInterface
58
     */
59 6
    public function serialize(CompactSignatureInterface $signature)
60
    {
61
        /** @var CompactSignature $signature */
62 6
        return $this->doSerialize($signature);
63
    }
64
65
    /**
66
     * @param Parser $parser
67
     * @return CompactSignature
68
     * @throws ParserOutOfRange
69
     */
70 8
    public function fromParser(Parser $parser)
71
    {
72 8
        $math = $this->ecAdapter->getMath();
73
74
        try {
75 8
            list ($byte, $r, $s) = $this->getTemplate()->parse($parser);
76
77 7
            $recoveryFlags = $math->sub($byte, 27);
78 7
            if ($recoveryFlags < 0 || $recoveryFlags > 7) {
79
                throw new \InvalidArgumentException('invalid signature type');
80
            }
81
82 7
            $isCompressed = $math->cmp($math->bitwiseAnd($recoveryFlags, 4), 0) !== 0;
83 7
            $recoveryId = $recoveryFlags - ($isCompressed ? 4 : 0);
84 1
        } catch (ParserOutOfRange $e) {
85 1
            throw new ParserOutOfRange('Failed to extract full signature from parser');
86
        }
87
88 7
        return new CompactSignature($this->ecAdapter, $r, $s, $recoveryId, $isCompressed);
89
    }
90
91
    /**
92
     * @param $string
93
     * @return CompactSignature
94
     * @throws ParserOutOfRange
95
     */
96 8
    public function parse($string)
97
    {
98 8
        return $this->fromParser(new Parser($string, $this->ecAdapter->getMath()));
99
    }
100
}
101