1
0
mirror of https://github.com/danog/dns.git synced 2025-01-22 13:31:12 +01:00
dns/lib/functions.php
Niklas Keller 315e03ff54 Refactor to Record class
Arrays are clearly not a good API and have resulted in bugs before, even within this package.
2017-06-22 19:22:45 +02:00

126 lines
3.4 KiB
PHP

<?php
namespace Amp\Dns;
use Amp\Loop;
use Amp\Promise;
const LOOP_STATE_IDENTIFIER = Resolver::class;
/**
* Retrieve the application-wide dns resolver instance.
*
* @param \Amp\Dns\Resolver $resolver Optionally specify a new default dns resolver instance
*
* @return \Amp\Dns\Resolver Returns the application-wide dns resolver instance
*/
function resolver(Resolver $resolver = null): Resolver {
if ($resolver === null) {
$resolver = Loop::getState(LOOP_STATE_IDENTIFIER);
if ($resolver) {
return $resolver;
}
$resolver = driver();
}
Loop::setState(LOOP_STATE_IDENTIFIER, $resolver);
return $resolver;
}
/**
* Create a new dns resolver best-suited for the current environment.
*
* @return \Amp\Dns\Resolver
*/
function driver(): Resolver {
return new DefaultResolver;
}
/**
* Resolve a hostname name to an IP address [hostname as defined by RFC 3986].
*
* Upon success the returned promise resolves to an array of Record objects.
*
* A null $ttl value indicates the DNS name was resolved from the cache or the local hosts file.
*
* If the custom per-request "server" option is not present the resolver will
* use the first nameserver in /etc/resolv.conf or default to Google's public
* DNS servers on Windows or if /etc/resolv.conf is not found.
*
* @param string $name The hostname to resolve
* @param array $options
*
* @return \Amp\Promise
* @TODO add boolean "clear_cache" option flag
*/
function resolve(string $name, array $options = []): Promise {
return resolver()->resolve($name, $options);
}
/**
* Query specific DNS records.
*
* @param string $name Unlike resolve(), query() allows for requesting _any_ name (as DNS RFC allows for arbitrary
* strings)
* @param int|int[] $type Use constants of Amp\Dns\Record
* @param array $options @see resolve documentation
*
* @return \Amp\Promise
*/
function query(string $name, $type, array $options = []): Promise {
return resolver()->query($name, $type, $options);
}
/**
* Checks whether a string is a valid DNS name.
*
* @param string $name String to check.
*
* @return bool
*/
function isValidDnsName(string $name) {
try {
normalizeDnsName($name);
return true;
} catch (InvalidNameError $e) {
return false;
}
}
/**
* Normalizes a DNS name and automatically checks it for validity.
*
* @param string $name DNS name.
*
* @return string Normalized DNS name.
*
* @throws InvalidNameError If an invalid name or an IDN name without ext/intl being installed has been passed.
*/
function normalizeDnsName(string $name): string {
static $pattern = '/^(?<name>[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)(\.(?&name))*$/i';
if (\function_exists('idn_to_ascii')) {
if (false === $result = \idn_to_ascii($name, 0, INTL_IDNA_VARIANT_UTS46)) {
throw new InvalidNameError("Name '{$name}' could not be processed for IDN.");
}
$name = $result;
} else {
if (\preg_match('/[\x80-\xff]/', $name)) {
throw new InvalidNameError(
"Name '{$name}' contains non-ASCII characters and IDN support is not available. " .
"Verify that ext/intl is installed for IDN support."
);
}
}
if (isset($name[253]) || !\preg_match($pattern, $name)) {
throw new InvalidNameError("Name '{$name}' is not a valid hostname.");
}
return $name;
}