NAME
POE::Component::Server::DNS - non-blocking, concurrent DNS server component
SYNOPSIS
use strict;
use Net::DNS::RR;
use POE qw(Component::Server::DNS);
my $dns_server = POE::Component::Server::DNS->spawn( alias => 'dns_server' );
POE::Session->create(
package_states => [ 'main' => [ qw(_start handler) ], ],
);
$poe_kernel->run();
exit 0;
sub _start {
my ($kernel,$heap) = @_[KERNEL,HEAP];
$kernel->post( 'dns_server', add_handler,
{
event => 'handler',
label => 'foobar',
match => 'foobar\.com$',
}
);
undef;
}
sub handler {
my ($qname,$qclass,$qtype,$callback) = @_[ARG0..ARG3];
my ($rcode, @ans, @auth, @add);
if ($qtype eq "A") {
my ($ttl, $rdata) = (3600, "10.1.2.3");
push @ans, Net::DNS::RR->new("$qname $ttl $qclass $qtype $rdata");
$rcode = "NOERROR";
} else {
$rcode = "NXDOMAIN";
}
$callback->($rcode, \@ans, \@auth, \@add, { aa => 1 });
undef;
}
DESCRIPTION
POE::Component::Server::DNS is a POE component that implements a DNS server.
It uses POE::Component::Client::DNS to handle resolving when configured as 'forward_only' and Net::DNS::Resolver::Recurse wrapped by POE::Component::Generic to perform recursion.
One may add handlers to massage and manipulate responses to particular queries which is vaguely modelled after Net::DNS::Nameserver.
CONSTRUCTOR
- spawn
-
Starts a POE::Component::Server::DNS component session and returns an object. Takes a number of optional arguments:
"alias", an alias to address the component by; "port", which udp port to listen on. Default is 53, which requires 'root' privilege on UN*X type systems; "resolver_opts", a set of options to pass to the POE::Component::Client::DNS constructor; "forward_only", be a forwarding only DNS server. Default is 0, be recursive.
METHODS
These are methods that may be used with the object returned by spawn().
- session_id
-
Returns the POE::Session ID of the component's session.
- resolver
-
Returns a reference to the POE::Component::Client::DNS object.
- shutdown
-
Terminates the component and associated resolver.
INPUT EVENTS
These are states that the component will accept:
- add_handler
-
Accepts a hashref as an argument with the following keys:
"event", the event the component will post to, mandatory; "label", a unique name for this handler, mandatory; "match", a regex expression ( without // ) to match against the host part of queries, mandatory; "session", the session where this handler event should be sent to, defaults to SENDER;
See OUTPUT EVENTS for details of what happens when a handler is triggered.
- del_handler
-
Accepts a handler label to remove.
- shutdown
-
Terminates the component and associated resolver.
OUTPUT EVENTS
Events are triggered by a DNS query matching a handler. The applicable event is fired in the requested session with the following paramters:
ARG0, query name
ARG1, query class
ARG2, query type
ARG3, a callback coderef
Do your manipulating then use the callback to fire the response back to the component, returning a response code and references to the answer, authority, and additional sections of the response. For advanced usage there is an optional argument containing an hashref with the settings for the aa, ra, and ad header bits. The argument is of the form { ad => 1, aa => 0, ra => 1 }.
$callback->( $rcode, \@ans, \@auth, \@add, { aa => 1 } );
HISTORY
The component's genesis was inspired by Jan-Pieter's 'Fun with POE' talk at YAPC::EU 2006, which lay much of the ground-work code such as the POE::Driver and POE::Filter used internally. BinGOs wrapped it all up in a component, added the tests ( borrowed shamelessly from POE::Component::Client::DNS's testsuite ) and documentation.
Other suggestions as to the API were provided by Ben 'integral' Smith.
Rocco Caputo brought POE::Component::Client::DNS to the party.
AUTHORS
Chris 'BinGOs' Williams
Jan-Pieter Cornet