Completed
Push — master ( 2e6493...4c3f4c )
by thomas
13:58
created

Locator::getPeerList()   B

Complexity

Conditions 3
Paths 2

Size

Total Lines 30
Code Lines 17

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 18
CRAP Score 3

Importance

Changes 0
Metric Value
dl 0
loc 30
ccs 18
cts 18
cp 1
rs 8.8571
c 0
b 0
f 0
cc 3
eloc 17
nc 2
nop 1
crap 3
1
<?php
2
3
namespace BitWasp\Bitcoin\Networking\Peer;
4
5
use BitWasp\Bitcoin\Networking\DnsSeeds\DnsSeedList;
6
use BitWasp\Bitcoin\Networking\Ip\Ipv4;
7
use BitWasp\Bitcoin\Networking\Services;
8
use BitWasp\Bitcoin\Networking\Structure\NetworkAddress;
9
use BitWasp\Bitcoin\Networking\Structure\NetworkAddressInterface;
10
use React\Dns\Resolver\Resolver;
11
use React\Promise\Deferred;
12
13
class Locator
14
{
15
16
    /**
17
     * @var Resolver
18
     */
19
    private $dns;
20
21
    /**
22
     * @var DnsSeedList
23
     */
24
    private $seeds;
25
26
    /**
27
     * @var NetworkAddressInterface[]
28
     */
29
    private $knownAddresses = [];
30
31
    /**
32
     * Locator constructor.
33
     * @param DnsSeedList $list
34
     * @param Resolver $dns
35
     */
36 9
    public function __construct(DnsSeedList $list, Resolver $dns)
37
    {
38 9
        $this->seeds = $list;
39 9
        $this->dns = $dns;
40 9
    }
41
42
    /**
43
     * @param bool $randomize - return a randomized list of dns seeds
44
     * @return string[]
45
     */
46
    public static function dnsSeedHosts($randomize = true)
47
    {
48
        $seeds = [
49
            'seed.bitcoin.sipa.be',
50
            'dnsseed.bluematt.me',
51
            'dnsseed.bitcoin.dashjr.org',
52
            'seed.bitcoinstats.com',
53
            'bitseed.xf2.org',
54
            'seed.bitnodes.io',
55
            "seed.bitcoin.jonasschnelli.ch"
56
        ];
57
58
        if ($randomize) {
59
            shuffle($seeds);
60
        }
61
62
        return $seeds;
63
    }
64
65
    /**
66
     * @param int $numSeeds
67
     * @return \React\Promise\Promise|\React\Promise\PromiseInterface
68
     */
69 6
    private function getPeerList($numSeeds = 1)
70
    {
71 6
        $peerList = new Deferred();
72
73
        // Take $numSeeds
74 6
        $seedHosts = $this->seeds->getHosts();
75 6
        shuffle($seedHosts);
76 6
        $seeds = array_slice($seedHosts, 0, min($numSeeds, count($seedHosts)));
77
78
        // Connect to $numSeeds peers
79
        /** @var Peer[] $vNetAddr */
80 6
        $vNetAddr = [];
81 6
        $c = 0;
82 6
        foreach ($seeds as $seed) {
83 6
            $this->dns
84 6
                ->resolve($seed)
85
                ->then(function ($ipList) use (&$vNetAddr, $peerList, &$numSeeds, &$c) {
86 5
                    $vNetAddr = array_merge($vNetAddr, $ipList);
87 5
                    if ($numSeeds === ++$c) {
88 5
                        $peerList->resolve($vNetAddr);
89 3
                    }
90
                }, function ($error) use ($peerList) {
91 1
                    $peerList->reject($error);
92 6
                })
93
            ;
94 4
        }
95
96
        // Compile the list of lists of peers into $this->knownAddresses
97 6
        return $peerList->promise();
98
    }
99
100
    /**
101
     * Connect to $numSeeds DNS seeds
102
     *
103
     * @param int $numSeeds
104
     * @return \React\Promise\Promise|\React\Promise\PromiseInterface
105
     */
106 6
    public function queryDnsSeeds($numSeeds = 1)
107
    {
108 6
        $deferred = new Deferred();
109 4
        $this
110 6
            ->getPeerList($numSeeds)
111 6
            ->then(
112
                function (array $vPeerVAddrs) use ($deferred) {
113 5
                    shuffle($vPeerVAddrs);
114
115
                    /** @var NetworkAddressInterface[] $addresses */
116 5
                    $addresses = [];
117 5
                    foreach ($vPeerVAddrs as $ip) {
118 5
                        $addresses[] = new NetworkAddress(
119 5
                            Services::NETWORK,
120 5
                            new Ipv4($ip),
121 2
                            8333
122 3
                        );
123 3
                    }
124
125 5
                    $this->knownAddresses = array_merge(
126 5
                        $this->knownAddresses,
127
                        $addresses
128 3
                    );
129
130 5
                    $deferred->resolve($this);
131 6
                },
132 3
                function ($error) use ($deferred) {
133 1
                    $deferred->reject($error);
134 3
                }
135 4
            )
136
        ;
137
138 6
        return $deferred->promise();
139
    }
140
141
    /**
142
     * @return NetworkAddressInterface[]
143
     */
144 2
    public function getKnownAddresses()
145
    {
146 2
        return $this->knownAddresses;
147
    }
148
149
    /**
150
     * Pop an address from the discovered peers
151
     *
152
     * @return NetworkAddressInterface
153
     * @throws \Exception
154
     */
155 6
    public function popAddress()
156
    {
157 6
        if (count($this->knownAddresses) < 1) {
158 3
            throw new \Exception('No peers');
159
        }
160
161 3
        return array_pop($this->knownAddresses);
162
    }
163
}
164