home *** CD-ROM | disk | FTP | other *** search
- <?php
- /*
- * License Information:
- *
- * Net_DNS: A resolver library for PHP
- * Copyright (C) 2002 Eric Kilfoil eric@ypass.net
- *
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License as published by the Free Software Foundation; either
- * version 2.1 of the License, or (at your option) any later version.
- *
- * This library is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public
- * License along with this library; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
- */
-
-
- /* Net_DNS_Resolver object definition {{{ */
- /**
- * A DNS Resolver library
- *
- * Resolver library. Builds a DNS query packet, sends the packet to the
- * server and parses the reponse.
- *
- * @package Net_DNS
- */
- class Net_DNS_Resolver
- {
- /* class variable definitions {{{ */
- /**
- * An array of all nameservers to query
- *
- * An array of all nameservers to query
- *
- * @var array $nameservers
- * @access public
- */
- var $nameservers;
- /**
- * The UDP port to use for the query (default = 53)
- *
- * The UDP port to use for the query (default = 53)
- *
- * @var integer $port
- * @access public
- */
- var $port;
- /**
- * The domain in which the resolver client host resides.
- *
- * The domain in which the resolver client host resides.
- *
- * @var string $domain
- * @access public
- */
- var $domain;
- /**
- * The searchlist to apply to unqualified hosts
- *
- * An array of strings containg domains to apply to unqualified hosts
- * passed to the resolver.
- *
- * @var array $searchlist
- * @access public
- */
- var $searchlist;
- /**
- * The number of seconds between retransmission of unaswered queries
- *
- * The number of seconds between retransmission of unaswered queries
- *
- * @var integer $retrans
- * @access public
- */
- var $retrans;
- /**
- * The number of times unanswered requests should be retried
- *
- * The number of times unanswered requests should be retried
- *
- * @var integer $retry
- * @access public
- */
- var $retry;
- /**
- * Whether or not to use TCP (Virtual Circuits) instead of UDP
- *
- * If set to 0, UDP will be used unless TCP is required. TCP is
- * required for questions or responses greater than 512 bytes.
- *
- * @var boolean $usevc
- * @access public
- */
- var $usevc;
- /**
- * Unknown
- */
- var $stayopen;
- /**
- * Ignore TC (truncated) bit
- *
- * If the server responds with the TC bit set on a response, and $igntc
- * is set to 0, the resolver will automatically retransmit the request
- * using virtual circuits (TCP).
- *
- * @access public
- * @var boolean $igntc
- */
- var $igntc;
- /**
- * Recursion Desired
- *
- * Sets the value of the RD (recursion desired) bit in the header. If
- * the RD bit is set to 0, the server will not perform recursion on the
- * request.
- *
- * @var boolean $recurse
- * @access public
- */
- var $recurse;
- /**
- * Unknown
- */
- var $defnames;
- /**
- * Unknown
- */
- var $dnsrch;
- /**
- * Contains the value of the last error returned by the resolver.
- *
- * Contains the value of the last error returned by the resolver.
- *
- * @var string $errorstring
- * @access public
- */
- var $errorstring;
- /**
- * The origin of the packet.
- *
- * This contains a string containing the IP address of the name server
- * from which the answer was given.
- *
- * @var string $answerfrom
- * @access public
- */
- var $answerfrom;
- /**
- * The size of the answer packet.
- *
- * This contains a integer containing the size of the DNS packet the
- * server responded with.
- *
- * @var string $answersize
- * @access public
- */
- var $answersize;
- /**
- * The number of seconds after which a TCP connetion should timeout
- *
- * @var integer $tcp_timeout
- * @access public
- */
- var $tcp_timeout;
- /**
- * The location of the system resolv.conf file.
- *
- * The location of the system resolv.conf file.
- *
- * @var string $resolv_conf
- */
- var $resolv_conf = "/etc/resolv.conf";
- /**
- * The name of the user defined resolv.conf
- *
- * The resolver will attempt to look in both the current directory as
- * well as the user's home directory for a user defined resolver
- * configuration file
- *
- * @var string $dotfile
- * @see Net_DNS_Resolver::$confpath
- */
- var $dotfile = ".resolv.conf";
- /**
- * A array of directories to search for the user's resolver config
- *
- * A array of directories to search for the user's resolver config
- *
- * @var string $confpath
- * @see Net_DNS_Resolver::$dotfile
- */
- var $confpath;
- /**
- * debugging flag
- *
- * If set to TRUE (non-zero), debugging code will be displayed as the
- * resolver makes the request.
- *
- * @var boolean $debug;
- * @access public
- */
- var $debug;
- /**
- * use the (currently) experimental PHP socket library
- *
- * If set to TRUE (non-zero), the Resolver will attempt to use the
- * much more effecient PHP sockets extension (if available).
- *
- * @var boolean $useEnhancedSockets;
- * @access public
- */
- var $useEnhancedSockets = 1;
-
- /* }}} */
- /* class constructor - Net_DNS_Resolver() {{{ */
- /**
- * Initializes the Resolver Object
- */
- function Net_DNS_Resolver()
- {
- $default = array(
- "nameservers" => array(),
- "port" => "53",
- "domain" => "",
- "searchlist" => array(),
- "retrans" => 5,
- "retry" => 4,
- "usevc" => 0,
- "stayopen" => 0,
- "igntc" => 0,
- "recurse" => 1,
- "defnames" => 1,
- "dnsrch" => 1,
- "debug" => 0,
- "errorstring" => "unknown error or no error",
- "answerfrom" => "",
- "answersize" => 0,
- "tcp_timeout" => 120
- );
- foreach ($default as $k => $v) {
- $this->{$k} = $v;
- }
- $this->confpath[0] = getenv("HOME");
- $this->confpath[1] = ".";
- $this->res_init();
- }
-
- /* }}} */
- /* Net_DNS_Resolver::res_init() {{{ */
- /**
- * Initalizes the resolver library
- *
- * res_init() searches for resolver library configuration files an
- * initializes the various properties of the resolver object.
- *
- * @see Net_DNS_Resolver::$resolv_conf, Net_DNS_Resolver::$dotfile,
- * Net_DNS_Resolver::$confpath, Net_DNS_Resolver::$searchlist,
- * Net_DNS_Resolver::$domain, Net_DNS_Resolver::$nameservers
- * @access public
- */
- function res_init()
- {
- $err = error_reporting(0);
- if (file_exists($this->resolv_conf) && is_readable($this->resolv_conf)) {
- $this->read_config($this->resolv_conf);
- }
-
- foreach ($this->confpath as $dir) {
- $file = "$dir/" . $this->dotfile;
- if (file_exists($file) && is_readable($file)) {
- $this->read_config($file);
- }
- }
-
- $this->read_env();
-
- if (!strlen($this->domain) && strlen($this->searchlist)) {
- $this->default{"domain"} = $this->default{"searchlist"}[0];
- } else if (! strlen($this->searchlist) && strlen($this->domain)) {
- $this->searchlist = array($this->domain);
- }
- error_reporting($err);
- }
-
- /* }}} */
- /* Net_DNS_Resolver::read_config {{{ */
- /**
- * Reads and parses a resolver configuration file
- *
- * @param string $file The name of the file to open and parse
- */
- function read_config($file)
- {
- if (! ($f = fopen($file, "r"))) {
- $this->error = "can't open $file";
- }
-
- while (! feof($f)) {
- $line = chop(fgets($f, 10240));
- $line = ereg_replace("(.*)[;#].*", "\\1", $line);
- if (ereg("^[ \t]*$", $line, $regs)) {
- continue;
- }
- ereg("^[ \t]*([^ \t]+)[ \t]+([^ \t]+)", $line, $regs);
- $option = $regs[1];
- $value = $regs[2];
-
- switch ($option) {
- case 'domain':
- $this->domain = $regs[2];
- break;
- case 'search':
- $this->searchlist[count($this->searchlist)] = $regs[2];
- break;
- case 'nameserver':
- foreach (split(" ", $regs[2]) as $ns)
- $this->nameservers[count($this->nameservers)] = $ns;
- break;
- }
- }
- fclose($f);
- }
-
- /* }}} */
- /* Net_DNS_Resolver::read_env() {{{ */
- /**
- * Examines the environment for resolver config information
- */
- function read_env()
- {
- if (getenv("RES_NAMESERVERS")) {
- $this->nameservers = split(" ", getenv("RES_NAMESERVERS"));
- }
-
- if (getenv("RES_SEARCHLIST")) {
- $this->searchlist = split(" ", getenv("RES_SEARCHLIST"));
- }
-
- if (getenv("LOCALDOMAIN")) {
- $this->domain = getenv("LOCALDOMAIN");
- }
-
- if (getenv("RES_OPTIONS")) {
- $env = split(" ", getenv("RES_OPTIONS"));
- foreach ($env as $opt) {
- list($name, $val) = split(":", $opt);
- if ($val == "") {
- $val = 1;
- }
- $this->{$name} = $val;
- }
- }
- }
-
- /* }}} */
- /* Net_DNS_Resolver::string() {{{ */
- /**
- * Builds a string containing the current state of the resolver
- *
- * Builds formatted string containing the state of the resolver library suited
- * for display.
- *
- * @access public
- */
- function string()
- {
- $state = ";; Net_DNS_Resolver state:\n";
- $state .= ";; domain = " . $this->domain . "\n";
- $state .= ";; searchlist = " . implode(" ", $this->searchlist) . "\n";
- $state .= ";; nameservers = " . implode(" ", $this->nameservers) . "\n";
- $state .= ";; port = " . $this->port . "\n";
- $state .= ";; tcp_timeout = ";
- $state .= ($this->tcp_timeout ? $this->tcp_timeout : "indefinite") . "\n";
- $state .= ";; retrans = " . $this->retrans . " ";
- $state .= "retry = " . $this->retry . "\n";
- $state .= ";; usevc = " . $this->usevc . " ";
- $state .= "stayopen = " . $this->stayopen . " ";
- $state .= "igntc = " . $this->igntc . "\n";
- $state .= ";; defnames = " . $this->defnames . " ";
- $state .= "dnsrch = " . $this->dnsrch . "\n";
- $state .= ";; recurse = " . $this->recurse . " ";
- $state .= "debug = " . $this->debug . "\n";
- return($state);
- }
-
- /* }}} */
- /* Net_DNS_Resolver::nextid() {{{ */
- /**
- * Returns the next request Id to be used for the DNS packet header
- */
- function nextid()
- {
- global $_Net_DNS_packet_id;
-
- return($_Net_DNS_packet_id++);
- }
- /* }}} */
- /* not completed - Net_DNS_Resolver::nameservers() {{{ */
- /**
- * Unknown - not ported yet
- */
- function nameservers($nsa)
- {
- $defres = new Net_DNS_Resolver();
-
- if (is_array($ns)) {
- foreach ($nsa as $ns) {
- if (ereg("^[0-9]+(\.[0-9]+){0,3}$", $ns, $regs)) {
- $newns[count($newns)] = $ns;
- } else {
- /*
- * This still needs to be ported
- *
- if ($ns !~ /\./) {
- if (defined $defres->searchlist) {
- @names = map { $ns . "." . $_ }
- $defres->searchlist;
- }
- elsif (defined $defres->domain) {
- @names = ($ns . "." . $defres->domain);
- }
- }
- else {
- @names = ($ns);
- }
-
- my $packet = $defres->search($ns);
- $this->errorstring($defres->errorstring);
- if (defined($packet)) {
- push @a, cname_addr([@names], $packet);
- }
- */
- }
- }
- $this->nameservers = $nsa;
- }
- return($this->nameservers);
- }
-
- /* }}} */
- /* not completed - Net_DNS_Resolver::cname_addr() {{{ */
- /**
- * Unknown - not ported yet
- */
- function cname_addr()
- {
- }
- /* }}} */
- /* Net_DNS_Resolver::search() {{{ */
- /**
- * Searches nameservers for an answer
- *
- * Goes through the search list and attempts to resolve name based on
- * the information in the search list.
- *
- * @param string $name The name (LHS) of a resource record to query.
- * @param string $type The type of record to query.
- * @param string $class The class of record to query.
- * @return mixed an object of type Net_DNS_Packet on success,
- * or FALSE on failure.
- * @see Net_DNS::typesbyname(), Net_DNS::classesbyname()
- * @access public
- */
- function search($name, $type = "A", $class = "IN")
- {
- /*
- * If the name looks like an IP address then do an appropriate
- * PTR query.
- */
- if (preg_match("/^(\d+)\.(\d+)\.(\d+)\.(\d+)$/", $name, $regs)) {
- $name = "$regs[4].$regs[3].$regs[2].$regs[1].in-addr.arpa";
- $type = "PTR";
- }
-
- /*
- * If the name contains at least one dot then try it as is first.
- */
- if (strchr($name, ".")) {
- if ($this->debug) {
- echo ";; search($name, $type, $class)\n";
- }
- $ans = $this->query($name, $type, $class);
- if ((is_object($ans)) && $ans->header->ancount > 0) {
- return($ans);
- }
- }
-
- /*
- * If the name doesn't end in a dot then apply the search list.
- */
- $domain = "";
- if ((! preg_match("/\.$/", $name)) && $this->dnsrch) {
- foreach ($this->searchlist as $domain) {
- $newname = "$name.$domain";
- if ($this->debug) {
- echo ";; search($newname, $type, $class)\n";
- }
- $ans = $this->query($newname, $type, $class);
- if ((is_object($ans)) && $ans->header->ancount > 0) {
- return($ans);
- }
- }
- }
-
- /*
- * Finally, if the name has no dots then try it as is.
- */
- if (! strlen(strchr($name, "."))) {
- if ($this->debug) {
- echo ";; search($name, $type, $class)\n";
- }
- $ans = $this->query("$name.", $type, $class);
- if (($ans = $this->query($name, $type, $class)) &&
- $ans->header->ancount > 0) {
- return($ans);
- }
- }
-
- /*
- * No answer was found.
- */
- return(0);
- }
-
- /* }}} */
- /* Net_DNS_Resolver::query() {{{ */
- /**
- * Queries nameservers for an answer
- *
- * Queries the nameservers listed in the resolver configuration for an
- * answer to a question packet.
- *
- * @param string $name The name (LHS) of a resource record to query.
- * @param string $type The type of record to query.
- * @param string $class The class of record to query.
- * @return mixed an object of type Net_DNS_Packet on success,
- * or FALSE on failure.
- * @see Net_DNS::typesbyname(), Net_DNS::classesbyname()
- * @access public
- */
- function query($name, $type = "A", $class = "IN")
- {
- /*
- * If the name doesn't contain any dots then append the default domain.
- */
- if ((strchr($name, ".") < 0) && $this->defnames) {
- $name .= "." . $this->domain;
- }
-
- /*
- * If the name looks like an IP address then do an appropriate
- * PTR query.
- */
- if (preg_match("/^(\d+)\.(\d+)\.(\d+)\.(\d+)$/", $name, $regs)) {
- $name = "$regs[4].$regs[3].$regs[2].$regs[1].in-addr.arpa";
- $type = "PTR";
- }
-
- if ($this->debug) {
- echo ";; query($name, $type, $class)\n";
- }
- $packet = new Net_DNS_Packet($this->debug);
- $packet->buildQuestion($name, $type, $class);
- $packet->header->rd = $this->recurse;
- $ans = $this->send($packet);
- if (is_object($ans) && $ans->header->ancount > 0) {
- return($ans);
- }
- return(0);
- }
-
- /* }}} */
- /* Net_DNS_Resolver::send($packetORname, $qtype = "", $qclass = "") {{{ */
- /**
- * Sends a packet to a nameserver
- *
- * Determines the appropriate communication method (UDP or TCP) and
- * send a DNS packet to a nameserver. Use of the this function
- * directly is discouraged. $packetORname should always be a properly
- * formatted binary DNS packet. However, it is possible to send a
- * query here and bypass Net_DNS_Resolver::query()
- *
- * @param string $packetORname A binary DNS packet stream or a
- * hostname to query
- * @param string $qtype This should not be used
- * @param string $qclass This should not be used
- * @return object Net_DNS_Packet An answer packet object
- */
- function send($packetORname, $qtype = "", $qclass = "")
- {
- $packet = $this->make_query_packet($packetORname, $qtype, $qclass);
- $packet_data = $packet->data();
-
- if ($this->usevc != 0 || strlen($packet_data > 512)) {
- $ans = $this->send_tcp($packet, $packet_data);
- } else {
- $ans = $this->send_udp($packet, $packet_data);
-
- if ($ans && $ans->header->tc && $this->igntc != 0) {
- if ($this->debug) {
- echo ";;\n;; packet truncated: retrying using TCP\n";
- }
- $ans = $this->send_tcp($packet, $packet_data);
- }
- }
- return($ans);
- }
-
- /* }}} */
- /* Net_DNS_Resolver::printhex($packet_data) {{{ */
- /**
- * Sends a packet via TCP to the list of name servers.
- */
- function printhex($data)
- {
- $data = " " . $data;
- $start = 0;
- while ($start < strlen($data)) {
- printf(";; %03d: ", $start);
- for ($ctr = $start; $ctr < $start+16; $ctr++) {
- if ($ctr < strlen($data))
- printf("%02x ", ord($data[$ctr]));
- else
- echo " ";
- }
- echo " ";
- for ($ctr = $start; $ctr < $start+16; $ctr++) {
- if (ord($data[$ctr]) < 32 || ord($data[$ctr]) > 127) {
- echo ".";
- } else {
- echo $data[$ctr];
- }
- }
- echo "\n";
- $start += 16;
- }
- }
- /* }}} */
- /* Net_DNS_Resolver::send_tcp($packet, $packet_data) {{{ */
- /**
- * Sends a packet via TCP to the list of name servers.
- *
- * @param string $packet A packet object to send to the NS list
- * @param string $packet_data The data in the packet as returned by
- * the Net_DNS_Packet::data() method
- * @return object Net_DNS_Packet Returns an answer packet object
- * @see Net_DNS_Resolver::send_udp(), Net_DNS_Resolver::send()
- */
- function send_tcp($packet, $packet_data)
- {
- if (! count($this->nameservers)) {
- $this->errorstring = "no nameservers";
- if ($this->debug) {
- echo ";; ERROR: send_tcp: no nameservers\n";
- }
- return(NULL);
- }
- $timeout = $this->tcp_timeout;
-
- foreach ($this->nameservers as $ns) {
- $srcport = $this->srcport;
- $srcaddr = $this->srcaddr;
- $dstport = $this->port;
- if ($this->debug) {
- echo ";; send_tcp($ns:$dstport) (src port = $srcport)\n";
- }
- $sock_key = "$ns:$dstport";
- if ($this->persistent_tcp &&
- is_resource($this->sockets[$sock_key])) {
- $sock = &$this->sockets[$sock_key];
- } else {
- if (! ($sock = fsockopen($ns, $dstport, $errno,
- $errstr, $timeout))) {
- $this->errorstring = "connection failed";
- if ($this->debug) {
- echo ";; ERROR: send_tcp: connection failed: $errstr\n";
- }
- continue;
- }
- $this->sockets[$sock_key] = $sock;
- unset($sock);
- $sock = &$this->sockets[$sock_key];
- }
- $lenmsg = pack("n", strlen($packet_data));
- if ($this->debug) {
- echo ";; sending " . strlen($packet_data) . " bytes\n";
- }
-
- if (($sent = fwrite($sock, $lenmsg)) == -1) {
- $this->errorstring = "length send failed";
- if ($this->debug) {
- echo ";; ERROR: send_tcp: length send failed\n";
- }
- continue;
- }
-
- if (($sent = fwrite($sock, $packet_data)) == -1) {
- $this->errorstring = "packet send failed";
- if ($this->debug) {
- echo ";; ERROR: send_tcp: packet data send failed\n";
- }
- }
-
- socket_set_timeout($sock, $timeout);
- $buf = fread($sock, 2);
- $e = socket_get_status($sock);
- $len = unpack("nint", $buf);
- $len = $len["int"];
- if (!$len) {
- continue;
- }
- $buf = fread($sock, $len);
- $actual = strlen($buf);
- $this->answerfrom = $ns;
- $this->answersize = $len;
- if ($this->debug) {
- echo ";; received $actual bytes\n";
- }
- if ($actual != $len) {
- $this->errorstring = "expected $len bytes, received $buf";
- if ($this->debug) {
- echo ";; send_tcp: " . $this->errorstring;
- }
- continue;
- }
-
- $ans = new Net_DNS_Packet($this->debug);
- if (is_null($ans->parse($buf))) {
- continue;
- }
- $this->errorstring = $ans->header->rcode;
- $ans->answerfrom = $this->answerfrom;
- $ans->answersize = $this->answersize;
- return($ans);
- }
- }
-
- /* }}} */
- /* Net_DNS_Resolver::send_udp_no_sock_lib($packet, $packet_data) {{{ */
- /**
- * Sends a packet via UDP to the list of name servers.
- *
- * This function sends a packet to a nameserver. It is called by
- * send_udp if the sockets PHP extension is not compiled into PHP.
- *
- * @param string $packet A packet object to send to the NS list
- * @param string $packet_data The data in the packet as returned by
- * the Net_DNS_Packet::data() method
- * @return object Net_DNS_Packet Returns an answer packet object
- * @see Net_DNS_Resolver::send_tcp(), Net_DNS_Resolver::send(),
- * Net_DNS_Resolver::send_udp(), Net_DNS_Resolver::send_udp_with_sock_lib()
- */
- function send_udp_no_sock_lib($packet, $packet_data)
- {
- $retrans = $this->retrans;
- $timeout = $retrans;
-
- /*
- * PHP doesn't have excellent socket support as of this writing.
- * This needs to be rewritten when PHP POSIX socket support is
- * complete.
- * Obviously, this code is MUCH different than the PERL implementation
- */
-
- $w = error_reporting(0);
- $ctr = 0;
- // Create a socket handle for each nameserver
- foreach ($this->nameservers as $nameserver) {
- if ($sock[$ctr++] = fsockopen("udp://$nameserver", $this->port)) {
- $peerhost[$ctr-1] = $nameserver;
- $peerport[$ctr-1] = $this->port;
- socket_set_blocking($sock, FALSE);
- } else {
- $ctr--;
- }
- }
- error_reporting($w);
-
- if ($ctr == 0) {
- $this->errorstring = "no nameservers";
- return(NULL);
- }
-
- for ($i = 0; $i < $this->retry; $i++, $retrans *= 2,
- $timeout = (int) ($retrans / (count($ns)+1))) {
- if ($timeout < 1) {
- $timeout = 1;
- }
-
- foreach ($sock as $k => $s) {
- if ($this->debug) {
- echo ";; send_udp(" . $peerhost[$k] . ":" . $peerport[$k] . "): sending " . strlen($packet_data) . " bytes\n";
- }
-
- if (! fwrite($s, $packet_data)) {
- if ($this->debug) {
- echo ";; send error\n";
- }
- }
-
- /*
- * Here's where it get's really nasty. We don't have a select()
- * function here, so we have to poll for a response... UGH!
- */
-
- $timetoTO = time() + (double)microtime() + $timeout;
-
- /*
- * let's sleep for a few hundred microseconds to let the
- * data come in from the network...
- */
- usleep(500);
- $buf = "";
- while (! strlen($buf) && $timetoTO > (time() +
- (double)microtime())) {
- socket_set_blocking($s, FALSE);
- if ($buf = fread($s, 512)) {
- $this->answerfrom = $peerhost[$k];
- $this->answersize = strlen($buf);
- if ($this->debug) {
- echo ";; answer from " . $peerhost[$k] . ":" .
- $peerport[$k] . ": " . strlen($buf) . " bytes\n";
- }
- $ans = new Net_DNS_Packet($this->debug);
- if ($ans->parse($buf)) {
- if ($ans->header->qr != "1") {
- continue;
- }
- if ($ans->header->id != $packet->header->id) {
- continue;
- }
- $this->errorstring = $ans->header->rcode;
- $ans->answerfrom = $this->answerfrom;
- $ans->answersize = $this->answersize;
- return($ans);
- }
- }
- // Sleep another 1/100th of a second... this sucks...
- usleep(1000);
- }
-
- $this->errorstring = "query timed out";
- return(NULL);
- }
- }
- }
-
- /* }}} */
- /* Net_DNS_Resolver::send_udp_with_sock_lib($packet, $packet_data) {{{ */
- /**
- * Sends a packet via UDP to the list of name servers.
- *
- * This function sends a packet to a nameserver. It is called by
- * send_udp if the sockets PHP extension is compiled into PHP.
- *
- * @param string $packet A packet object to send to the NS list
- * @param string $packet_data The data in the packet as returned by
- * the Net_DNS_Packet::data() method
- * @return object Net_DNS_Packet Returns an answer packet object
- * @see Net_DNS_Resolver::send_tcp(), Net_DNS_Resolver::send(),
- * Net_DNS_Resolver::send_udp(), Net_DNS_Resolver::send_udp_no_sock_lib()
- */
- function send_udp_with_sock_lib($packet, $packet_data)
- {
- $retrans = $this->retrans;
- $timeout = $retrans;
-
- //$w = error_reporting(0);
- $ctr = 0;
- // Create a socket handle for each nameserver
- foreach ($this->nameservers as $nameserver) {
- if ((($sock[$ctr++] = socket_create(AF_INET, SOCK_DGRAM, SOL_UDP)) >= 0) &&
- socket_connect($sock[$ctr-1], $nameserver, $this->port) >= 0) {
- $peerhost[$ctr-1] = $nameserver;
- $peerport[$ctr-1] = $this->port;
- socket_set_nonblock($sock[$ctr-1]);
- } else {
- $ctr--;
- }
- }
- //error_reporting($w);
-
- if ($ctr == 0) {
- $this->errorstring = "no nameservers";
- return(NULL);
- }
-
- for ($i = 0; $i < $this->retry; $i++, $retrans *= 2,
- $timeout = (int) ($retrans / (count($ns)+1))) {
- if ($timeout < 1) {
- $timeout = 1;
- }
-
- foreach ($sock as $k => $s) {
- if ($this->debug) {
- echo ";; send_udp(" . $peerhost[$k] . ":" . $peerport[$k] . "): sending " . strlen($packet_data) . " bytes\n";
- }
-
- if (! socket_write($s, $packet_data)) {
- if ($this->debug) {
- echo ";; send error\n";
- }
- }
-
- $set = array($s);
- if ($this->debug) {
- echo ";; timeout set to $timeout seconds\n";
- }
- $changed = socket_select($set, $w = null, $e = null, $timeout);
- if ($changed) {
- $buf = socket_read($set[0], 512);
- $this->answerfrom = $peerhost[$k];
- $this->answersize = strlen($buf);
- if ($this->debug) {
- echo ";; answer from " . $peerhost[$k] . ":" .
- $peerport[$k] . ": " . strlen($buf) . " bytes\n";
- }
- $ans = new Net_DNS_Packet($this->debug);
- if ($ans->parse($buf)) {
- if ($ans->header->qr != "1") {
- continue;
- }
- if ($ans->header->id != $packet->header->id) {
- continue;
- }
- $this->errorstring = $ans->header->rcode;
- $ans->answerfrom = $this->answerfrom;
- $ans->answersize = $this->answersize;
- return($ans);
- }
- }
-
- $this->errorstring = "query timed out";
- return(NULL);
- }
- }
- }
-
- /* }}} */
- /* Net_DNS_Resolver::send_udp($packet, $packet_data) {{{ */
- /**
- * Sends a packet via UDP to the list of name servers.
- *
- * This function sends a packet to a nameserver. send_udp calls
- * either Net_DNS_Resolver::send_udp_no_sock_lib() or
- * Net_DNS_Resolver::send_udp_with_sock_lib() depending on whether or
- * not the sockets extension is compiled into PHP. Note that using the
- * sockets extension is MUCH more effecient.
- *
- * @param object Net_DNS_Packet $packet A packet object to send to the NS list
- * @param string $packet_data The data in the packet as returned by
- * the Net_DNS_Packet::data() method
- * @return object Net_DNS_Packet Returns an answer packet object
- * @see Net_DNS_Resolver::send_tcp(), Net_DNS_Resolver::send(),
- * Net_DNS_Resolver::send_udp(), Net_DNS_Resolver::send_udp_no_sock_lib()
- */
- function send_udp($packet, $packet_data)
- {
- if (extension_loaded("sockets") && $this->useEnhancedSockets) {
- if ($this->debug) {
- echo "\n;; using extended PHP sockets\n";
- }
- return($this->send_udp_with_sock_lib($packet, $packet_data));
- } else {
- if ($this->debug) {
- echo "\n;; using simple sockets\n";
- }
- return($this->send_udp_no_sock_lib($packet, $packet_data));
- }
- }
-
- /* }}} */
- /* not completed - Net_DNS_Resolver::bgsend() {{{ */
- /**
- * Unknown - not ported yet
- */
- function bgsend()
- {
- }
-
- /* }}} */
- /* not completed - Net_DNS_Resolver::bgread() {{{ */
- /**
- * Unknown - not ported yet
- */
- function bgread()
- {
- }
- /* }}} */
- /* not completed - Net_DNS_Resolver::bgisready() {{{ */
- /**
- * Unknown - not ported yet
- */
- function bgisready()
- {
- }
- /* }}} */
- /* Net_DNS_Resolver::make_query_packet($packetORname, $type = "", $class = "") {{{ */
- /**
- * Unknown
- */
- function make_query_packet($packetORname, $type = "", $class = "")
- {
- if (is_object($packetORname) && get_class($packetORname) == "net_dns_packet") {
- $packet = $packetORname;
- } else {
- $name = $packetORname;
- if ($type == "") {
- $type = "A";
- }
- if ($class == "") {
- $class = "IN";
- }
-
- /*
- * If the name looks like an IP address then do an appropriate
- * PTR query.
- */
- if (preg_match("/^(\d+)\.(\d+)\.(\d+)\.(\d+)$/", $name, $regs)) {
- $name = "$regs[4].$regs[3].$regs[2].$regs[1].in-addr.arpa";
- $type = "PTR";
- }
-
- if ($this->debug) {
- echo ";; query($name, $type, $class)\n";
- }
- $packet = new Net_DNS_Packet($this->debug);
- $packet->buildQuestion($name, $type, $class);
- }
-
- $packet->header->rd = $this->recurse;
-
- return($packet);
- }
-
- /* }}} */
- /* Net_DNS_Resolver::axfr() {{{ */
- /**
- * Performs an AXFR query (zone transfer)
- *
- * Requests a zone transfer from the nameservers. Note that zone
- * transfers will ALWAYS use TCP regardless of the setting of the
- * Net_DNS_Resolver::$usevc flag
- *
- * @param string $dname The domain (zone) to transfer
- * @param string $class The class in which to look for the zone.
- * @return object Net_DNS_Packet
- * @access public
- */
- function axfr($dname, $class = "IN")
- {
- if ($this->debug) {
- echo ";; axfr_start($dname, $class)\n";
- }
- if (! count($this->nameservers)) {
- $this->errorstring = "no nameservers";
- if ($this->debug) {
- echo ";; ERROR: no nameservers\n";
- }
- return(NULL);
- }
- $packet = $this->make_query_packet($dname, "AXFR", $class);
- $packet_data = $packet->data();
- $ans = $this->send_tcp($packet, $packet_data);
- return($ans);
- }
-
- /* }}} */
- /* not completed - Net_DNS_Resolver::read_tcp() {{{ */
- /**
- * Unknown - not ported yet
- */
- function read_tcp()
- {
- }
-
- /* }}} */
- }
- /* }}} */
- /* VIM settings {{{
- * Local variables:
- * tab-width: 4
- * c-basic-offset: 4
- * soft-stop-width: 4
- * c indent on
- * expandtab on
- * End:
- * vim600: sw=4 ts=4 sts=4 cindent fdm=marker et
- * vim<600: sw=4 ts=4
- * }}} */
- ?>
-