--- bin/dhcpd.pl 2009/07/28 16:35:49 22 +++ bin/dhcpd.pl 2009/07/29 00:46:29 27 @@ -8,11 +8,12 @@ use autodie; use IO::Socket::INET; -use Net::DHCP::Packet; -use Net::DHCP::Constants; use File::Slurp; use Data::Dump qw/dump/; +use lib 'lib'; +use Net::DHCP::Packet; +use Net::DHCP::Constants 0.67; die "need to run $0 as root like this\nsudo $0\n" unless $< == 0; my $debug = shift @ARGV; @@ -76,6 +77,8 @@ return $ip; } +my $transaction = 0; # FIXME predictible transaction numbers + while (1) { require "config.pl"; # refresh config @@ -88,14 +91,10 @@ if (defined $buf) { - my $dhcp; + my $dhcp = Net::DHCP::Packet->new($buf); + $dhcp->comment( $transaction++ ); - eval { $dhcp = Net::DHCP::Packet->new($buf); }; - die "can't use request", dump( $buf ) if $@; - - if ( $debug ) { - warn "recv: ", $dhcp->toString, "\n\n"; - } + warn "recv: ", $dhcp->toString, "\n\n"; my $mac = substr($dhcp->chaddr(),0,$dhcp->hlen()*2); my $ip = client_ip($mac); @@ -107,7 +106,7 @@ $file = 'undionly.kpxe'; } - my $packet = new Net::DHCP::Packet( + my $packet = { Op => BOOTREPLY(), Hops => $dhcp->hops(), Xid => $dhcp->xid(), @@ -119,11 +118,34 @@ Chaddr => $dhcp->chaddr(), File => $file, # DHO_DHCP_MESSAGE_TYPE() => DHCPACK(), - DHO_SUBNET_MASK() => '255.0.0.0', - ); +# DHO_SUBNET_MASK() => '255.255.255.0', + }; + + my $messagetype = $dhcp->getOptionValue(DHO_DHCP_MESSAGE_TYPE()); + + if ($messagetype eq DHCPDISCOVER()) { + warn "DHCP DISCOVER"; + $packet->{Comment} = $dhcp->comment(); + $packet->{DHO_DHCP_MESSAGE_TYPE()} = DHCPOFFER(); + } elsif ($messagetype eq DHCPREQUEST()) { + my $requested_ip = $dhcp->getOptionValue(DHO_DHCP_REQUESTED_ADDRESS()); + warn "DHCP REQUEST $requested_ip"; + if ( $ip eq $requested_ip ) { + $packet->{DHO_DHCP_MESSAGE_TYPE()} = DHCPACK(); + } else { + $packet->{DHO_DHCP_MESSAGE_TYPE()} = DHCPNAK(); + $packet->{DHO_DHCP_MESSAGE()} = "Bad request, expected $ip"; + } + } elsif ($messagetype eq DHCPINFORM()) { + warn "DHCP INFORM ignored"; + } else { + warn "$messagetype igored (bootp?)"; + } + - warn ">> $mac == $ip server: $server_ip file: $file\n"; + warn ">> $mac == $ip server: $server_ip", $file ? " file: $file\n" : "\n"; + $packet = new Net::DHCP::Packet( %$packet ); warn "## ",$packet->toString(),"\n" if $debug; my $reply = IO::Socket::INET->new(