/[pxelator]/lib/PXElator/httpd.pm
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Diff of /lib/PXElator/httpd.pm

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 327 by dpavlin, Fri Aug 28 19:41:16 2009 UTC revision 419 by dpavlin, Thu Sep 10 13:08:36 2009 UTC
# Line 19  use File::Slurp; Line 19  use File::Slurp;
19  use IO::Socket::INET;  use IO::Socket::INET;
20  use Regexp::Common qw/net/;  use Regexp::Common qw/net/;
21    
22  sub menu {qq{  our $title;
23    
24    sub html_start {
25    qq{
26    <html>
27    <head>
28    <title>$title</title>
29    </head>
30    <body>
31    }}
32    
33    sub html_end {
34    qq{
35    </body>
36    </html>
37    }}
38    
39    sub menu {
40    qq{
41  <div style="font-size: 80%; color: #888">  <div style="font-size: 80%; color: #888">
42  <a href=/>home</a>  <a href=/ target=/>home</a>
43  <a href=/server>server</a>  <a href=/server target=server>server</a>
44  <a href=/brctl>brctl</a>  <a href=/brctl target=brctl>brctl</a>
45  <a href=/ip>ip</a>  <a href=/ip target=ip>ip</a>
46  <a href=/nmap>nmap</a>  <a href=/nmap target=nmap>nmap</a>
47  <a href=/client>client</a>  <a href=/client target=client>client</a>
48    <a href=http://$server::ip:5984/_utils/ target=couchdb>couchdb</a>
49  </div>  </div>
50    
51  }}  }}
# Line 56  use ip; Line 74  use ip;
74  use wireshark;  use wireshark;
75  use syslogd;  use syslogd;
76  use nmap;  use nmap;
77    use ping;
78    use wol;
79    
80  use CouchDB;  use CouchDB;
81    
# Line 66  sub static { Line 86  sub static {
86    
87          return if ! -f $full;          return if ! -f $full;
88    
89            return if $full =~ m{\.ico$};
90    
91          if ( my $pid = fork ) {          if ( my $pid = fork ) {
92                  # parent                  # parent
93                  close($client);                  close($client);
# Line 107  sub static { Line 129  sub static {
129  }  }
130    
131  sub ok {  sub ok {
132          qq|HTTP/1.0 200 OK\r\nContent-Type: text/html\r\nConnection: close\r\n\r\n| . menu()          qq|HTTP/1.0 200 OK\r\nContent-Type: text/html\r\nConnection: close\r\n\r\n| . html_start() . menu()
133  }  }
134    
135  sub redirect {  sub redirect {
# Line 123  sub get_request { Line 145  sub get_request {
145    
146          CouchDB::audit( 'request', { path => $path, param => $param, peerhost => $client->peerhost } );          CouchDB::audit( 'request', { path => $path, param => $param, peerhost => $client->peerhost } );
147    
148            $title = $path;
149    
150          if ( my $found = static( $client,$path ) ) {          if ( my $found = static( $client,$path ) ) {
151                  warn "static $found" if $debug;                  warn "static $found" if $debug;
152          } elsif ( $path eq '/' ) {          } elsif ( $path eq '/' ) {
# Line 152  warn "XXX pids = ", dump( $daemons::pids Line 176  warn "XXX pids = ", dump( $daemons::pids
176                                                  ;                                                  ;
177                                  }                                  }
178    
179                                  if ( $name->can('fork_if_active') ) {                                  my $class = $name;
180                                          $html .= qq| <a href=/start_stop/$name/$_>$_</a>| foreach $name->fork_if_active;                                  $class =~ s{\.\d+$}{};
181    
182                                    if ( $class->can('fork_if_active') ) {
183                                            $html .= qq| <a href=/start_stop/$name/$_>$_</a>| foreach $class->fork_if_active;
184                                  }                                  }
185    
186                                  if ( $name->can('actions') ) {                                  if ( $class->can('actions') ) {
187                                          $html .= qq| <a href=/action/$name/$_>$_</a>| foreach $name->actions;                                          $html .= qq| <a href=/action/$name/$_>$_</a>| foreach $class->actions;
188                                  }                                  }
189                          } else {                          } else {
190                                  if ( $pid =~ m{^\d+$} ) {                                  if ( $pid =~ m{^\d+$} ) {
# Line 191  warn "XXX pids = ", dump( $daemons::pids Line 218  warn "XXX pids = ", dump( $daemons::pids
218                          }                          }
219                  }                  }
220    
221                    my $kvm = kvm::next_nr;
222                    $kvm = qq|<div><a href=/start_stop/kvm?nr=$kvm>create new kvm $kvm</a></div>|;
223    
224                  print $client ok                  print $client ok
225                          , html::table( 2, @rows )                          , html::table( 2, @rows )
226                          , $below_table                          , $below_table
227                            , $kvm
228                          , html::tabs( log::mac_changes )                          , html::tabs( log::mac_changes )
229                          , $debug_proc                          , $debug_proc
230                          ;                          ;
231    
232          } elsif ( $path =~ m{^/server} ) {          } elsif ( $path =~ m{^/server} ) {
233                    foreach my $name ( keys %$param ) {
234                            eval '$server::' . $name . '= $param->{$name}';
235                    }
236                    my @table = (
237                              'debug' => qq|<a href=/our/debug/| . boolean::toggle($debug) . qq|>$debug</a>|,
238                            , 'new_clients' => qq|<input type=text name=new_clients size=3 value="$server::new_clients">|
239                    );
240    
241                    foreach my $editable ( 'ip', 'netmask', 'ip_from', 'ip_to', 'domain' ) {
242                            my $v = eval '$server::' . $editable;
243                            push @table, ( $editable, qq|<input type=text name=$editable value="$v">| );
244                    }
245    
246                    foreach my $readonly ( 'base_dir', 'conf' ) {
247                            my $v = eval '$server::' . $readonly;
248                            push @table, ( $readonly, html::tt $v );
249                    }
250                            
251                  print $client ok                  print $client ok
252                          , html::table( 2,                          , qq|<form method=get>|
253                                  'debug' => qq|<a href=/our/debug/| . boolean::toggle($debug) . qq|>$debug</a>|,                          , html::table( 2, @table )
254                                   map {                          , qq|
255                                          ( $_, html::tt eval '$server::'.$_ )                                  <input type=submit name=action value=change>
256                                   } ( 'ip', 'netmask', 'ip_from', 'ip_to', 'domain_name', 'base_dir', 'conf' )                                  </form>
257                          )                          |
258                          ;                          ;
259    
260          } elsif ( $path =~ m!^/client(?:/$RE{net}{IPv4}{-keep})?! ) {          } elsif ( $path =~ m!^/client(?:/$RE{net}{IPv4}{-keep})?! ) {
261                  my $ip = $1;                  my $ip = $1;
262                    $title = $ip if $ip;
263    
264                  if ( $param->{action} eq 'remove' ) {                  if ( $param->{action} eq 'remove' ) {
265                          client::remove( $param->{change_ip} );                          client::remove( $param->{change_ip} );
# Line 224  warn "XXX pids = ", dump( $daemons::pids Line 275  warn "XXX pids = ", dump( $daemons::pids
275                  if ( ! $ip ) {                  if ( ! $ip ) {
276                          my $peer_ip = $client->peerhost;                          my $peer_ip = $client->peerhost;
277    
278                          my $netmask = ip::to_int $server::netmask;                          my $netmask  = ip::to_int $server::netmask;
279                          my $network = ip::to_int $server::ip & $netmask;                          my $network  = ip::to_int($server::ip) & $netmask;
280                          my ( $from, $to ) = ( $network | $server::ip_from, $network | $server::ip_to );                          my $from_int = $network | $server::ip_from;
281                          my $ip_int  = ip::to_int $peer_ip;                          my $to_int   = $network | $server::ip_to;
282                            my $ip_int   = ip::to_int $peer_ip;
283    
284                          # show edit for clients in our dhcp range                          # show edit for clients in our dhcp range
285                          if ( $ip_int >= ( $network | $server::ip_from ) && $ip_int <= ( $network | $server::ip_to ) ) {                          if ( $ip_int >= $from_int && $ip_int <= $to_int ) {
286                                  $ip = $peer_ip;                                  $ip = $peer_ip;
287                          }                          }
288                  }                  }
289    
290                  if ( $ip && $ip ne $server::ip ) {                  if ( $ip && $ip ne $server::ip ) {
                         my $hostname = client::conf( $ip, 'hostname' => $param->{hostname} );  
291    
292                            my @editable = ( qw/hostname config homepage/ );
293    
294                            client::conf( $ip, $_ => $param->{$_} ) foreach @editable;
295    
296                            my $conf = client::all_conf( $ip );
297                            my $config = delete $conf->{config};
298    
299                            my $nmap = qq|<a href=/nmap?scan=$ip>nmap</a>|;
300                          my @table = (                          my @table = (
301                                    'ping' => ping::host($ip)
302                                            ? qq|<span style="color:green">up</span> $nmap|
303                                            : qq|<span style="color: red">down</span> <a href=/wol/$ip>wol</a> $nmap|
304                                            ,
305                                  'ip' => qq|<input type=text name=change_ip value="$ip" onChange="document.getElementById('old_ip').style.display = '';"><span id=old_ip style="display: none; color: #888;">old: $ip<span>|,                                  'ip' => qq|<input type=text name=change_ip value="$ip" onChange="document.getElementById('old_ip').style.display = '';"><span id=old_ip style="display: none; color: #888;">old: $ip<span>|,
306                                  'hostname' => qq|<input type=text name=hostname value="$hostname">|,                                  'mac' => format::mac( delete $conf->{mac}, 'html' ),
307                                    'hostname' => qq|<input type=text name=hostname value="| . delete($conf->{hostname}) . qq|">|,
308                                    'config' => html::select( 'config', $config, config::available ),
309                                    html::conf( $ip, $conf, 'edit', @editable )
310                          );                          );
311    
                         my $deploy;  
   
                         if ( my $mac = client::mac_from_ip( $ip ) ) {  
                                 $deploy = client::conf( $ip, 'deploy' => $param->{deploy} );  
                                 push @table, (  
                                         'mac' => format::mac( $mac => 'html' ),  
                                         'deploy' => html::select( 'deploy', $deploy, config::available )  
                                 );  
                                 if ( my $pxelinux = config::for_ip( $ip ) ) {  
                                         $deploy = qq|<h2>PXElinux</h2>| . html::pre( $pxelinux );  
                                 }  
                         }  
   
312                          print $client ok                          print $client ok
313                                  , qq|<form method=get>|                                  , qq|<form method=get>|
314                                  , html::table( 2, @table ),                                  , html::table( 2, @table ),
# Line 263  warn "XXX pids = ", dump( $daemons::pids Line 316  warn "XXX pids = ", dump( $daemons::pids
316                                          <input type=submit name=action value=change>                                          <input type=submit name=action value=change>
317                                          <input type=submit name=action value=remove style="color: red">                                          <input type=submit name=action value=remove style="color: red">
318                                          </form>|                                          </form>|
                                 , $deploy  
319                                  ;                                  ;
320    
321                          if ( my $amt = client::conf( $ip, 'amt' ) ) {                          if ( $config ) {
322                                  print $client qq|<h2>AMT</h2>|, amt::info( $amt, $ip );                                  if ( my $for_ip = config::for_ip( $ip ) ) {
323                                            print $client qq|<h2>config::for_ip</h2>| . html::pre( $for_ip );
324                                    }
325                          }                          }
326    
327                            if ( $conf->{amt} ) {
328                                    print $client qq|<h2>amt network</h2>|, html::pre_dump( amt::network( $ip ) );
329                                    print $client qq|<h2>amt log</h2>|, html::pre_dump( amt::log( $ip ) );
330                            }
331    
332                  } else {                  } else {
333    
334                          my $arp = {                          my @ping;
335                                  map {                          if ( my $host = $param->{ping_target} ) {
336                                          my @c = split(/\s+/,$_);                                  @ping = ( $host );
337                                          if ( $#c == 5 ) {                          } elsif ( $param->{ping} ) {
338                                                  client::save_ip_mac( $c[0], $c[3] );                                  @ping = client::all_ips;
339                                                  ( uc $c[3] => $c[5] )                          }
                                         } else {  
                                         }  
                                 } read_file('/proc/net/arp')  
                         };  
340    
341                          warn "# arp ",dump( $arp );                          my $ping = ping::fping( @ping ) if @ping;
342                            my $arp = client::arp_mac_dev;
343    
344                          print $client ok                          print $client ok
345                                  , qq|<h2>Clients on $server::ip</h2>|                                  , qq|<h2>Clients on $server::ip</h2>|
346                                  , html::table( -6,                                  , html::table( -5,
347                                          'ip', 'mac', 'dev', 'hostname', 'deploy', 'conf',                                          'ip', 'mac', 'dev', 'hostname', 'conf',
348                                          map {                                          map {
349                                                  my $ip = $_;                                                  my $ip = $_;
350                                                  my $conf = client::all_conf( $ip );                                                  my $conf = client::all_conf( $ip );
351                                                  my $mac = delete $conf->{mac} || '';                                                  my $mac = delete $conf->{mac} || '';
352                                                    my $style;
353                                                    $style
354                                                            = 'style="color:'
355                                                            . ( $ping->{$ip} ? 'green' : 'red' )
356                                                            . '"'
357                                                            if $ping;
358                                                    $style ||= '';
359                                                    my $ip_text = qq|<tt>$ip</tt>|;
360                                                    $ip_text = qq|<tt><b>$ip</b></tt>| if ip::in_dhcp_range($ip);
361                                                  (                                                  (
362                                                          qq|<a name=$ip href=/client/$ip>$ip</a>|                                                          qq|<a $style name=$ip target=$ip href=/client/$ip>$ip_text</a>|
363                                                          , format::mac( $mac => 'html' )                                                          , format::mac( $mac => 'html' )
364                                                          , $arp->{$mac}                                                          , $arp->{$mac}
365                                                          , delete $conf->{hostname}                                                          , delete $conf->{hostname}
366                                                          , delete $conf->{deploy}                                                          , html::conf( $ip, $conf, 'inline' )
367                                                          , ( %$conf ? html::pre_dump( $conf ) : qq|<a href=/nmap?scan=$ip>nmap</a>| )                                                  )
                                                 );  
368                                          } client::all_ips                                          } client::all_ips
369                                  )                                  )
370                                  ;                                  ;
371                            print $client qq|
372                                    <form method=get>
373                                    <input type=text   name=ping_target   size=15>
374                                    <input type=submit name=ping value=ping>
375                                    </form>
376                            |;
377                  }                  }
378    
379    
380          } elsif ( $path =~ m{^/brctl} ) {          } elsif ( $path =~ m{^/brctl} ) {
381    
382                    system 'brctl addif virtual ' . $param->{addif} if $param->{addif};
383                    system 'brctl delif virtual ' . $param->{delif} if $param->{delif};
384    
385                    my $in_virtual;
386    
387                    my @table =
388                            map {
389                                    my @c = split(/\t+/,$_,4);
390                                    if ( $#c == 1 ) {
391                                            $in_virtual->{ $c[1] }++;
392                                            @c = ( '', '', '', $c[1] );
393                                    } else {
394                                            $in_virtual->{ $c[3] }++;
395                                    }
396                                    if ( $c[3] =~ m{\d$} ) {
397                                            $c[3] = qq|<input type=submit name=delif value=$c[3] style="color:red" title="remove $c[3] from bridge">|;
398                                    }
399                                    @c
400                            } split(/\n/, `brctl show`)
401                    ;
402    
403                    my @add_ifs = grep { ! $in_virtual->{$_} && $_ ne 'virtual' } ip::devices_up;
404    
405                    push @table, ( '', '', '', html::select( 'addif', @add_ifs ) . qq|<input type=submit value=add></form>| );
406    
407                  print $client ok                  print $client ok
408                          ,html::pre( `brctl show` )                          , qq|<form>|
409                            , html::table( -4, @table )
410                            , qq|</form>|
411                          ;                          ;
412    
413    
414          } elsif ( $path =~ m{^/ip/?(\w+)?} ) {          } elsif ( $path =~ m{^/ip/?(\w+)?} ) {
415                  print $client ok                  print $client ok
416                          , join("\n", map { qq|<a href=/ip/$_>$_</a>| } ( qw/link addr route neigh ntable tunnel maddr mroute xfrm/ ))                          , join("\n", map { qq|<a href=/ip/$_>$_</a>| } ( qw/link addr route neigh ntable tunnel maddr mroute xfrm/ ))
# Line 325  warn "XXX pids = ", dump( $daemons::pids Line 428  warn "XXX pids = ", dump( $daemons::pids
428                                  </form>                                  </form>
429                          |;                          |;
430                  }                  }
431            } elsif ( $path =~ m{^/wol/(\S+)} ) {
432                    print $client redirect( "$url/client/$1" ), wol::power_on($1);
433            } elsif ( $path =~ m!^/amt/(\w+)/$RE{net}{IPv4}{-keep}! ) {
434                    my ( $run, $ip ) = ( $1, $2 );
435                    print $client redirect( "$url/client/$ip" ), amt::RemoteControl( $ip, $run );
436          } elsif ( $path =~ m{^/our/(\w+)/(\S+)} ) {          } elsif ( $path =~ m{^/our/(\w+)/(\S+)} ) {
437                  eval 'our $' . $1 . ' = ' . $2;                  eval 'our $' . $1 . ' = ' . $2;
438                  warn $@ if $@;                  warn $@ if $@;
439                  print $client redirect($url), qq|<big>$1 = $2</big><br>Location: <a href="$url">$url</a>|;                  print $client redirect($url), qq|<big>$1 = $2</big><br>Location: <a href="$url">$url</a>|;
440                  server::debug( $debug ) if $1 eq 'debug';                  server::debug( $debug ) if $1 eq 'debug';
441          } elsif ( $path =~ m{^/start_stop/(\S+)} ) {          } elsif ( $path =~ m{^/start_stop/(\S+)} ) {
442                  print $client redirect, daemons::start_stop($1);                  print $client redirect, daemons::start_stop($1,$param);
443          } elsif ( $path =~ m{^/action/([^/]+)/(.+)} ) {          } elsif ( $path =~ m{^/action/([^/]+)/(.+)} ) {
444                  $1->$2();                  my ( $package, $method ) = ( $1, $2 );
445                    $ENV{nr} = $1 if $package =~ s{\.(\d+)$}{};
446                    $package->$method();
447                  print $client redirect;                  print $client redirect;
448          } elsif ( $path =~ m{^/kill/static/(\d+)} ) {          } elsif ( $path =~ m{^/kill/static/(\d+)} ) {
449                  print $client redirect;                  print $client redirect;
# Line 349  sub start { Line 459  sub start {
459    
460          warn 'network ', network::setup();          warn 'network ', network::setup();
461    
462          daemons::start_stop 'browser', $url;          daemons::start_stop 'browser', { url => $url };
463          daemons::start_stop $_ foreach ( qw/dhcpd tftpd dnsd syslogd/ );          daemons::start_stop $_ foreach ( qw/dhcpd tftpd dnsd syslogd/ );
464          daemons::start_stop 'kvm' unless $ENV{DEV}; # skip kvm statup when running on real device  #       daemons::start_stop 'kvm' unless $ENV{DEV}; # skip kvm statup when running on real device
465    
466          my $server = IO::Socket::INET->new(          my $server = IO::Socket::INET->new(
467                          Proto     => 'tcp',                          Proto     => 'tcp',
468                          LocalAddr => $server::ip,  #                       LocalAddr => $server::ip,
469                          LocalPort => $httpd::port,                          LocalPort => $httpd::port,
470                          Listen    => SOMAXCONN,                          Listen    => SOMAXCONN,
471                          Reuse     => 1                          Reuse     => 1
# Line 363  sub start { Line 473  sub start {
473    
474          print "url $url\n";          print "url $url\n";
475    
476            syslogd::install_local;
477            client::rebuild_mac_links;
478    
479          while (1) {          while (1) {
480                  my $client = $server->accept() || next; # ALARM trickle us                  my $client = $server->accept() || next; # ALARM trickle us
481                  my $request = <$client>;                  my $request = <$client>;
# Line 371  sub start { Line 484  sub start {
484    
485                  if ($request =~ m{^GET (/.*) HTTP/1.[01]}) {                  if ($request =~ m{^GET (/.*) HTTP/1.[01]}) {
486                          my $path = $1;                          my $path = $1;
487                            $path =~ s/%([0-9A-Fa-f]{2})/chr(hex($1))/ge;
488                          my $param;                          my $param;
489                          if ( $path =~ s{\?(.+)}{} ) {                          if ( $path =~ s{\?(.+)}{} ) {
490                                  foreach my $p ( split(/[&;]/, $1) ) {                                  foreach my $p ( split(/[&;]/, $1) ) {
# Line 385  sub start { Line 499  sub start {
499                          warn "500 $request";                          warn "500 $request";
500                  }                  }
501    
502                  print $client menu() if $client->connected;                  print $client menu() . html_end() if $client->connected;
503    
504          }          }
505    

Legend:
Removed from v.327  
changed lines
  Added in v.419

  ViewVC Help
Powered by ViewVC 1.1.26