Codebase list reactphp-dns / 3d5acd7 src / Query / Executor.php
3d5acd7

Tree @3d5acd7 (Download .tar.gz)

Executor.php @3d5acd7raw · history · blame

<?php

namespace React\Dns\Query;

use React\Dns\Model\Message;
use React\Dns\Protocol\Parser;
use React\Dns\Protocol\BinaryDumper;
use React\EventLoop\LoopInterface;
use React\Promise\Deferred;
use React\Promise;
use React\Stream\DuplexResourceStream;
use React\Stream\Stream;

/**
 * @deprecated unused, exists for BC only
 * @see UdpTransportExecutor
 */
class Executor implements ExecutorInterface
{
    private $loop;
    private $parser;
    private $dumper;
    private $timeout;

    /**
     *
     * Note that albeit supported, the $timeout parameter is deprecated!
     * You should pass a `null` value here instead. If you need timeout handling,
     * use the `TimeoutConnector` instead.
     *
     * @param LoopInterface $loop
     * @param Parser $parser
     * @param BinaryDumper $dumper
     * @param null|float $timeout DEPRECATED: timeout for DNS query or NULL=no timeout
     */
    public function __construct(LoopInterface $loop, Parser $parser, BinaryDumper $dumper, $timeout = 5)
    {
        $this->loop = $loop;
        $this->parser = $parser;
        $this->dumper = $dumper;
        $this->timeout = $timeout;
    }

    public function query($nameserver, Query $query)
    {
        $request = Message::createRequestForQuery($query);

        $queryData = $this->dumper->toBinary($request);
        $transport = strlen($queryData) > 512 ? 'tcp' : 'udp';

        return $this->doQuery($nameserver, $transport, $queryData, $query->name);
    }

    /**
     * @deprecated unused, exists for BC only
     */
    public function prepareRequest(Query $query)
    {
        return Message::createRequestForQuery($query);
    }

    public function doQuery($nameserver, $transport, $queryData, $name)
    {
        // we only support UDP right now
        if ($transport !== 'udp') {
            return Promise\reject(new \RuntimeException(
                'DNS query for ' . $name . ' failed: Requested transport "' . $transport . '" not available, only UDP is supported in this version'
            ));
        }

        $that = $this;
        $parser = $this->parser;
        $loop = $this->loop;

        // UDP connections are instant, so try this without a timer
        try {
            $conn = $this->createConnection($nameserver, $transport);
        } catch (\Exception $e) {
            return Promise\reject(new \RuntimeException('DNS query for ' . $name . ' failed: ' . $e->getMessage(), 0, $e));
        }

        $deferred = new Deferred(function ($resolve, $reject) use (&$timer, $loop, &$conn, $name) {
            $reject(new CancellationException(sprintf('DNS query for %s has been cancelled', $name)));

            if ($timer !== null) {
                $loop->cancelTimer($timer);
            }
            $conn->close();
        });

        $timer = null;
        if ($this->timeout !== null) {
            $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->on('data', function ($data) use ($conn, $parser, $deferred, $timer, $loop, $name) {
            $conn->end();
            if ($timer !== null) {
                $loop->cancelTimer($timer);
            }

            try {
                $response = $parser->parseMessage($data);
            } catch (\Exception $e) {
                $deferred->reject($e);
                return;
            }

            if ($response->header->isTruncated()) {
                $deferred->reject(new \RuntimeException('DNS query for ' . $name . ' failed: The server returned a truncated result for a UDP query, but retrying via TCP is currently not supported'));
                return;
            }

            $deferred->resolve($response);
        });
        $conn->write($queryData);

        return $deferred->promise();
    }

    /**
     * @deprecated unused, exists for BC only
     */
    protected function generateId()
    {
        return mt_rand(0, 0xffff);
    }

    /**
     * @param string $nameserver
     * @param string $transport
     * @return \React\Stream\DuplexStreamInterface
     */
    protected function createConnection($nameserver, $transport)
    {
        $fd = @stream_socket_client("$transport://$nameserver", $errno, $errstr, 0, STREAM_CLIENT_CONNECT | STREAM_CLIENT_ASYNC_CONNECT);
        if ($fd === false) {
            throw new \RuntimeException('Unable to connect to DNS server: ' . $errstr, $errno);
        }

        // Instantiate stream instance around this stream resource.
        // This ought to be replaced with a datagram socket in the future.
        // Temporary work around for Windows 10: buffer whole UDP response
        // @coverageIgnoreStart
        if (!class_exists('React\Stream\Stream')) {
            // prefer DuplexResourceStream as of react/stream v0.7.0
            $conn = new DuplexResourceStream($fd, $this->loop, -1);
        } else {
            // use legacy Stream class for react/stream < v0.7.0
            $conn = new Stream($fd, $this->loop);
            $conn->bufferSize = null;
        }
        // @coverageIgnoreEnd

        return $conn;
    }
}