forked from clue/reactphp-mdns
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathMulticastExecutor.php
More file actions
122 lines (96 loc) · 3.41 KB
/
MulticastExecutor.php
File metadata and controls
122 lines (96 loc) · 3.41 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
<?php
namespace Clue\React\Mdns;
use React\Dns\BadServerException;
use React\Dns\Model\Message;
use React\Dns\Protocol\Parser;
use React\Dns\Protocol\BinaryDumper;
use React\EventLoop\Loop;
use React\EventLoop\LoopInterface;
use React\Promise\Deferred;
use Clue\React\Multicast\Factory as DatagramFactory;
use React\Dns\Query\ExecutorInterface;
use React\Dns\Query\Query;
use React\Dns\Query\TimeoutException;
/**
* DNS executor that uses multicast sockets
*
* Based on React\Dns\Query\Executor which should eventually be split into
* multiple smaller components..
*/
class MulticastExecutor implements ExecutorInterface
{
private $loop;
private $parser;
private $dumper;
private $timeout;
private $factory;
public function __construct(LoopInterface $loop = null, Parser $parser = null, BinaryDumper $dumper = null, $timeout = 5, DatagramFactory $factory = null)
{
$this->loop = $loop ?: Loop::get();
if ($parser === null) {
$parser = new Parser();
}
if ($dumper === null) {
$dumper = new BinaryDumper();
}
if ($factory === null) {
$factory = new DatagramFactory($this->loop);
}
$this->parser = $parser;
$this->dumper = $dumper;
$this->timeout = $timeout;
$this->factory = $factory;
}
public function query($nameserver, Query $query)
{
$request = $this->prepareRequest($query);
$queryData = $this->dumper->toBinary($request);
return $this->doQuery($nameserver, $queryData, $query->name);
}
public function prepareRequest(Query $query)
{
$request = new Message();
$request->header->set('id', $this->generateId());
$request->header->set('rd', 1);
$request->questions[] = (array) $query;
$request->prepare();
return $request;
}
public function doQuery($nameserver, $queryData, $name)
{
$that = $this;
$parser = $this->parser;
$loop = $this->loop;
$deferred = new Deferred(function ($_, $reject) use (&$conn, &$timer, $loop, $name) {
$conn->close();
$loop->cancelTimer($timer);
$reject(new \RuntimeException(sprintf("DNS query for %s cancelled", $name)));
});
$timer = $this->loop->addTimer($this->timeout, function () use (&$conn, $name, $deferred) {
$conn->close();
$deferred->reject(new TimeoutException(sprintf("DNS query for %s timed out", $name)));
});
$conn = $this->factory->createSender();
$conn->on('message', function ($data) use ($conn, $parser, $deferred, $timer, $loop) {
$response = new Message();
$responseReady = $parser->parseChunk($data, $response);
$conn->close();
$loop->cancelTimer($timer);
if (!$responseReady) {
$deferred->reject(new BadServerException('Invalid response received'));
return;
}
if ($response->header->isTruncated()) {
$deferred->reject(new BadServerException('The server set the truncated bit although we issued a TCP request'));
return;
}
$deferred->resolve($response);
});
$conn->send($queryData, $nameserver);
return $deferred->promise();
}
protected function generateId()
{
return mt_rand(0, 0xffff);
}
}