Completed
Push — master ( a22894...37130c )
by thomas
62:03 queued 58:09
created

CompactSignatureSerializer::doSerialize()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 8
Code Lines 5

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 6
CRAP Score 1.0156

Importance

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