icicleio / dns
Asynchronous DNS for Icicle.
Installs: 46 334
Dependents: 3
Suggesters: 0
Security: 0
Stars: 6
Watchers: 4
Forks: 1
Open Issues: 0
Requires
- daverandom/libdns: ^1
- icicleio/icicle: ^0.9.1
- icicleio/socket: ^0.5
- paragonie/random_compat: ^1
Requires (Dev)
- mockery/mockery: ^0.9
- phpunit/phpunit: ^4.6
- symfony/yaml: ^2.7
README
This library is a component for Icicle, providing an asynchronous DNS query executor, resolver, and client connector. An asynchronous DNS server is currently under development and will be added to this component in the future. Like other Icicle components, this library uses Coroutines built from Awaitables and Generators to make writing asynchronous code more like writing synchronous code.
Requirements
- PHP 5.5+ for v0.6.x branch (current stable) and v1.x branch (mirrors current stable)
- PHP 7 for v2.0 branch (under development) supporting generator delegation and return expressions
Installation
The recommended way to install is with the Composer package manager. (See the Composer installation guide for information on installing and using Composer.)
Run the following command to use this library in your project:
composer require icicleio/dns
You can also manually edit composer.json
to add this library as a project requirement.
// composer.json { "require": { "icicleio/dns": "^0.6" } }
Example
The example below uses the Icicle\Dns\resolver()
function to asynchronously find the IP address for the domain icicle.io
.
use Icicle\Coroutine\Coroutine; use Icicle\Dns; use Icicle\Loop; $generator = function ($domain) { try { $ips = (yield Dns\resolve($domain)); foreach ($ips as $ip) { echo "IP: {$ip}\n"; } } catch (Exception $exception) { echo "Error when executing query: {$exception->getMessage()}\n"; } }; $coroutine = new Coroutine($generator('icicle.io')); $coroutine->done(); Loop\run();
Documentation
- Executors - Executes a DNS query.
- Resolver - Resolves the IP address for a domain name.
- Connector - Connects to a host and port.
Methods returning a Generator
can be used to create a Coroutine (e.g., new Coroutine($executor->execute(...))
) or yielded within another Coroutine (use yield from
in PHP 7 for better performance).
This library uses LibDNS to create and parse DNS messages. Unfortunately the documentation for this library is currently limited to DocComments in the source code. If only using resolvers and connectors in this library, there is no need to worry about how LibDNS works. Executors returns coroutines that are resolved with LibDNS\Messages\Message
instances, representing the response from the DNS server. Using these objects is simple and will be described in the executor section below.
Function prototypes
Prototypes for object instance methods are described below using the following syntax:
ClassOrInterfaceName::methodName(ArgumentType $arg): ReturnType
Executors
Executors are the foundation of the DNS component, performing any DNS query and returning the full results of that query. Resolvers and connectors depend on executors to perform the DNS query required for their operation.
Each executor implements Icicle\Dns\Executor\Executor
that defines a single method, execute()
.
Executor::execute( string $domain, string|int $type, array $options = [] ): \Generator
An executor will retry a query a number of times if it doesn't receive a response within timeout
seconds. The number of times a query will be retried before failing is defined by retries
, with timeout
seconds elapsing between each query attempt.
execute() Function
The simplest way to perform a DNS query is to use the Icicle\Dns\execute()
function. This function uses an Icicle\Dns\Executor\Executor
object that can be set or retrieved using the Icicle\Dns\executor()
function.
`Icicle\Dns\execute( string $domain, string|int $type, array $options = [] ): \Generator
Example using the execute()
function to find the A record for a domain:
use Icicle\Coroutine\Coroutine; use Icicle\Dns; use Icicle\Loop; use LibDNS\Messages\Message; $coroutine = new Coroutine(Dns\execute('google.com', 'A')); $coroutine->done( function (Message $message) { foreach ($message->getAnswerRecords() as $resource) { echo "TTL: {$resource->getTTL()} Value: {$resource->getData()}\n"; } }, function (Exception $exception) { echo "Query failed: {$exception->getMessage()}\n"; } ); Loop\run();
Creating an Executor
The simplest executor is Icicle\Dns\Executor\BasicExecutor
, created by providing the constructor with the IP address of a DNS server to use to perform queries. It is recommended to use a DNS server closest to you, such as the local router. If this is not possible, Google operates two DNS server that also can be used at 8.8.8.8
and 8.8.4.4
.
use Icicle\Dns\Executor\BasicExecutor; $executor = new BasicExecutor('8.8.8.8');
The Icicle\Dns\Executor\BasicExecutor
constructor also accepts an instance of Icicle\Socket\Connector\Connector
as the second argument if custom behavior is desired when connecting to the name server. If no instance is given, the default global connector is used (see Icicle\Socket\connector()
).
Using an Executor
Once created, an executor is used by calling the execute()
method with the domain and type of DNS query to be performed. The type may be a case-insensitive string naming a record type (e.g., 'A'
, 'MX'
, 'NS'
, 'PTR'
, 'AAAA'
) or the integer value corresponding to a record type (LibDNS\Records\ResourceQTypes
defines constants corresponding to a the integer value of a type). execute()
returns a coroutine fulfilled with an instance of LibDNS\Messages\Message
that represents the response from the name server. LibDNS\Messages\Message
objects have several methods that will need to be used to fetch the data in the response.
getAnswerRecords()
: Returns an instance ofLibDNS\Records\RecordCollection
, a traversable collection ofLibDNS\Record\Resource
objects containing the response answer records.getAuthorityRecords()
: Returns an instance ofLibDNS\Records\RecordCollection
containing the response authority records.getAdditionalRecords()
: Returns an instance ofLibDNS\Records\RecordCollection
containing the response additional records.getAuthorityRecords()
: Returns an instance ofLibDNS\Records\RecordCollection
containing the response authority records.isAuthoritative()
: Determines if the response is authoritative for the records returned.
DNS records in the traversable LibDNS\Records\RecordCollection
objects are represented as instances of LibDNS\Records\Resource
. These objects have several methods to access the data associated with the record.
getType()
: Returns the record type as aninteger
.getName()
: Gets the domain name associated with the record as astring
.getData()
: Returns anLibDNS\Records\RData
instance representing the records data. This object may be cast to astring
or each field can be accessed with theLibDNS\Records\RData::getField(int $index)
method. The number of fields in a resource depends on the type of resource (e.g.,MX
records contain two fields, a priority and a host name).getTTL()
: Gets the TTL (time-to-live) as aninteger
.
Below is an example of how an executor can be used to find the NS records for a domain.
use Icicle\Coroutine\Coroutine; use Icicle\Dns\Executor\BasicExecutor; use Icicle\Loop; use LibDNS\Messages\Message; $executor = new BasicExecutor('8.8.8.8'); $coroutine = new Coroutine($executor->execute('google.com', 'NS')); $coroutine->done( function (Message $message) { foreach ($message->getAnswerRecords() as $resource) { echo "TTL: {$resource->getTTL()} Value: {$resource->getData()}\n"; } }, function (Exception $exception) { echo "Query failed: {$exception->getMessage()}\n"; } ); Loop\run();
MultiExecutor
The Icicle\Dns\Executor\MultiExecutor
class can be used to combine multiple executors to send queries to several name servers so queries can be resolved even if some name servers stop responding.
use Icicle\Coroutine\Coroutine; use Icicle\Dns\Executor\BasicExecutor; use Icicle\Dns\Executor\MultiExecutor; use Icicle\Loop; use LibDNS\Messages\Message; $executor = new MultiExecutor(); $executor->add(new BasicExecutor('8.8.8.8')); $executor->add(new BasicExecutor('8.8.4.4')); // Executor will send query to 8.8.4.4 if 8.8.8.8 does not respond. $coroutine = new Coroutine($executor->execute('google.com', 'MX')); $coroutine->done( function (Message $message) { foreach ($message->getAnswerRecords() as $resource) { echo "TTL: {$resource->getTTL()} Value: {$resource->getData()}\n"; } }, function (Exception $exception) { echo "Query failed: {$exception->getMessage()}\n"; } ); Loop\run();
Queries using the above executor will automatically send requests to the second name server if the first does not respond. Subsequent queries are initially sent to the last server that successfully responded to a query.
Resolver
A resolver finds the IP addresses for a given domain. Icicle\Dns\Resolver\BasicResolver
implements Icicle\Dns\Resolver\Resolver
, which defines a single method, resolve()
. A resolver is essentially a specialized executor that performs only A
queries, fulfilling the coroutine returned from resolve()
with an array of IP addresses (even if only one or zero IP addresses is found, the coroutine is still resolved with an array).
Resolver::resolve( string $domain, array $options = [] ): \Generator
resolve() Function
The simplest way find the IP address for a domain is to use the Icicle\Dns\resolve()
function. This function uses an Icicle\Dns\Resolver\Resolver
object that can be set or retrieved using the Icicle\Dns\resolver()
function.
`Icicle\Dns\resolve( string $domain, array $options = [] ): \Generator
Example using the resolve()
function to resolve the IP address of a domain:
use Icicle\Coroutine\Coroutine; use Icicle\Dns; use Icicle\Loop; $coroutine = new Coroutine(Dns\resolve('google.com')); $coroutine->done( function (array $ips) { foreach ($ips as $ip) { echo "IP: {$ip}\n"; } }, function (Exception $exception) { echo "Query failed: {$exception->getMessage()}\n"; } ); Loop\run();
Like executors, a resolver will retry a query retries
times if the name server does not respond within timeout
seconds.
The Icicle\Resolver\BasicResolver
class is constructed by passing an Icicle\Executor\Executor
instance that is used to execute queries to resolve domains. If no executor is given, one will be created by default, using 8.8.8.8
and 8.8.4.4
as DNS servers for the executor.
Example
use Icicle\Coroutine\Coroutine; use Icicle\Dns\Resolver\BasicResolver; use Icicle\Loop; $resolver = new BasicResolver(); $coroutine = new Coroutine($resolver->resolve('google.com')); $coroutine->done( function (array $ips) { foreach ($ips as $ip) { echo "IP: {$ip}\n"; } }, function (Exception $exception) { echo "Query failed: {$exception->getMessage()}\n"; } ); Loop\run();
Connector
The connector component connects to a server by first resolving the hostname provided, then making the connection and resolving the returned coroutine with an instance of Icicle\Socket\Socket
.
Icicle\Dns\Connector\Connector
defines a single method, connect()
that should resolve a host name and connect to one of the resolved servers, resolving the coroutine with the connected client.
Connector::connect( string $domain, int $port, array $options = [], ): \Generator
connect() Function
The simplest way to resolve a domain name and connect to a port on the resolved host is with the Icicle\Dns\connect()
function. This function uses an Icicle\Dns\Connector\Connector
object that can be set or retrieved using the Icicle\Dns\connector()
function.
`Icicle\Dns\connect( string $domain, int $port, array $options = [] ): \Generator
Example using the connect()
function to resolve the IP address of a domain and connect to port 443:
use Icicle\Dns; use Icicle\Loop; use Icicle\Socket\Socket; $connector = new DefaultConnector(); $coroutine = new Coroutine(Dns\connect('google.com', 443)); $coroutine->done( function (Socket $client) { echo "IP: {$client->getRemoteAddress()}\n"; echo "Port: {$client->getRemotePort()}\n"; }, function (Exception $exception) { echo "Connecting failed: {$exception->getMessage()}\n"; } ); Loop\run();
Icicle\Dns\Connector\DefaultConnector
will attempt to connect to one of the IP addresses found for a given host name. If the server at that IP is unresponsive, the connector will attempt to establish a connection to the next IP in the list until a server accepts the connection. Only if the connector is unable to connect to all of the IPs will it reject the coroutine returned from connect()
. The constructor also optionally accepts an instance of Icicle\Socket\Connector\Connector
if custom behavior is desired when connecting to the resolved host.
Additionally, all the other options available to Icicle\Socket\Connector\Connector::connect()
may also be used.
Example
use Icicle\Dns\Connector\DefaultConnector; use Icicle\Loop; use Icicle\Socket\Socket; $connector = new DefaultConnector(); $coroutine = new Coroutine($connector->connect('google.com', 80)); $coroutine->done( function (Socket $client) { echo "IP: {$client->getRemoteAddress()}\n"; echo "Port: {$client->getRemotePort()}\n"; }, function (Exception $exception) { echo "Connecting failed: {$exception->getMessage()}\n"; } ); Loop\run();