bind to all IP addresses and nost just $server::ip (insecure, but convinient
[pxelator] / lib / PXElator / httpd.pm
index cf1edc3..c01a1f4 100644 (file)
@@ -19,15 +19,33 @@ use File::Slurp;
 use IO::Socket::INET;
 use Regexp::Common qw/net/;
 
-sub menu {qq{
+our $title;
+
+sub html_start {
+qq{
+<html>
+<head>
+<title>$title</title>
+</head>
+<body>
+}}
+
+sub html_end {
+qq{
+</body>
+</html>
+}}
 
+sub menu {
+qq{
 <div style="font-size: 80%; color: #888">
-<a href=/>home</a>
-<a href=/server>server</a>
-<a href=/brctl>brctl</a>
-<a href=/ip>ip</a>
-<a href=/nmap>nmap</a>
-<a href=/client>client</a>
+<a href=/ target=/>home</a>
+<a href=/server target=server>server</a>
+<a href=/brctl target=brctl>brctl</a>
+<a href=/ip target=ip>ip</a>
+<a href=/nmap target=nmap>nmap</a>
+<a href=/client target=client>client</a>
+<a href=http://$server::ip:5984/_utils/ target=couchdb>couchdb</a>
 </div>
 
 }}
@@ -68,6 +86,8 @@ sub static {
 
        return if ! -f $full;
 
+       return if $full =~ m{\.ico$};
+
        if ( my $pid = fork ) {
                # parent
                close($client);
@@ -109,7 +129,7 @@ sub static {
 }
 
 sub ok {
-       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()
 }
 
 sub redirect {
@@ -125,6 +145,8 @@ sub get_request {
 
        CouchDB::audit( 'request', { path => $path, param => $param, peerhost => $client->peerhost } );
 
+       $title = $path;
+
        if ( my $found = static( $client,$path ) ) {
                warn "static $found" if $debug;
        } elsif ( $path eq '/' ) {
@@ -154,12 +176,15 @@ warn "XXX pids = ", dump( $daemons::pids );
                                                ;
                                }
 
-                               if ( $name->can('fork_if_active') ) {
-                                       $html .= qq| <a href=/start_stop/$name/$_>$_</a>| foreach $name->fork_if_active;
+                               my $class = $name;
+                               $class =~ s{\.\d+$}{};
+
+                               if ( $class->can('fork_if_active') ) {
+                                       $html .= qq| <a href=/start_stop/$name/$_>$_</a>| foreach $class->fork_if_active;
                                }
 
-                               if ( $name->can('actions') ) {
-                                       $html .= qq| <a href=/action/$name/$_>$_</a>| foreach $name->actions;
+                               if ( $class->can('actions') ) {
+                                       $html .= qq| <a href=/action/$name/$_>$_</a>| foreach $class->actions;
                                }
                        } else {
                                if ( $pid =~ m{^\d+$} ) {
@@ -193,24 +218,48 @@ warn "XXX pids = ", dump( $daemons::pids );
                        }
                }
 
+               my $kvm = kvm::next_nr;
+               $kvm = qq|<div><a href=/start_stop/kvm?nr=$kvm>create new kvm $kvm</a></div>|;
+
                print $client ok
                        , html::table( 2, @rows )
                        , $below_table
+                       , $kvm
                        , html::tabs( log::mac_changes )
                        , $debug_proc
                        ;
 
        } elsif ( $path =~ m{^/server} ) {
+               foreach my $name ( keys %$param ) {
+                       eval '$server::' . $name . '= $param->{$name}';
+               }
+               my @table = (
+                         'debug' => qq|<a href=/our/debug/| . boolean::toggle($debug) . qq|>$debug</a>|,
+                       , 'new_clients' => qq|<input type=text name=new_clients size=3 value="$server::new_clients">|
+               );
+
+               foreach my $editable ( 'ip', 'netmask', 'ip_from', 'ip_to', 'domain' ) {
+                       my $v = eval '$server::' . $editable;
+                       push @table, ( $editable, qq|<input type=text name=$editable value="$v">| );
+               }
+
+               foreach my $readonly ( 'base_dir', 'conf' ) {
+                       my $v = eval '$server::' . $readonly;
+                       push @table, ( $readonly, html::tt $v );
+               }
+                       
                print $client ok
-                       , html::table( 2,
-                               'debug' => qq|<a href=/our/debug/| . boolean::toggle($debug) . qq|>$debug</a>|,
-                                map {
-                                       ( $_, html::tt eval '$server::'.$_ )
-                                } ( 'ip', 'netmask', 'ip_from', 'ip_to', 'domain_name', 'base_dir', 'conf' )
-                       )
+                       , qq|<form method=get>|
+                       , html::table( 2, @table )
+                       , qq|
+                               <input type=submit name=action value=change>
+                               </form>
+                       |
                        ;
+
        } elsif ( $path =~ m!^/client(?:/$RE{net}{IPv4}{-keep})?! ) {
                my $ip = $1;
+               $title = $ip;
 
                if ( $param->{action} eq 'remove' ) {
                        client::remove( $param->{change_ip} );
@@ -240,33 +289,26 @@ warn "XXX pids = ", dump( $daemons::pids );
 
                if ( $ip && $ip ne $server::ip ) {
 
-                       my @editable = ( qw/hostname deploy homepage/ );
+                       my @editable = ( qw/hostname config homepage/ );
 
                        client::conf( $ip, $_ => $param->{$_} ) foreach @editable;
 
                        my $conf = client::all_conf( $ip );
-                       my $deploy = delete $conf->{deploy};
+                       my $config = delete $conf->{config};
 
                        my $nmap = qq|<a href=/nmap?scan=$ip>nmap</a>|;
                        my @table = (
+                               'ping' => ping::host($ip)
+                                       ? qq|<span style="color:green">up</span> $nmap|
+                                       : qq|<span style="color: red">down</span> <a href=/wol/$ip>wol</a> $nmap|
+                                       ,
                                '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>|,
                                'mac' => format::mac( delete $conf->{mac}, 'html' ),
                                'hostname' => qq|<input type=text name=hostname value="| . delete($conf->{hostname}) . qq|">|,
-                               'ping' => ping::host($ip) ? qq|<span style="color:green">up</span> $nmap| : qq|<span style="color: red">down</span> <a href=/wol/$ip>wol</a> $nmap|,
-                               'deploy' => html::select( 'deploy', $deploy, config::available ),
+                               'config' => html::select( 'config', $config, config::available ),
                                html::conf( $ip, $conf, 'edit', @editable )
                        );
 
-warn "XX conf ",dump( $conf );
-
-                       my $config = '';
-
-                       if ( $deploy ) {
-                               if ( $config = config::for_ip( $ip ) ) {
-                                       $config = qq|<h2>config</h2>| . html::pre( $config );
-                               }
-                       }
-
                        print $client ok
                                , qq|<form method=get>|
                                , html::table( 2, @table ),
@@ -274,11 +316,17 @@ warn "XX conf ",dump( $conf );
                                        <input type=submit name=action value=change>
                                        <input type=submit name=action value=remove style="color: red">
                                        </form>|
-                               , $config
                                ;
 
-                       if ( my $amt = client::conf( $ip, 'amt' ) ) {
-                               print $client qq|<h2>AMT</h2>|, amt::info( $amt, $ip );
+                       if ( $config ) {
+                               if ( my $for_ip = config::for_ip( $ip ) ) {
+                                       print $client qq|<h2>config::for_ip</h2>| . html::pre( $for_ip );
+                               }
+                       }
+
+                       if ( $conf->{amt} ) {
+                               print $client qq|<h2>amt network</h2>|, html::pre_dump( amt::network( $ip ) );
+                               print $client qq|<h2>amt log</h2>|, html::pre_dump( amt::log( $ip ) );
                        }
 
                } else {
@@ -308,8 +356,10 @@ warn "XX conf ",dump( $conf );
                                                        . '"'
                                                        if $ping;
                                                $style ||= '';
+                                               my $ip_text = qq|<tt>$ip</tt>|;
+                                               $ip_text = qq|<tt><b>$ip</b></tt>| if ip::in_dhcp_range($ip);
                                                (
-                                                       qq|<a $style name=$ip href=/client/$ip>$ip</a>|
+                                                       qq|<a $style name=$ip target=$ip href=/client/$ip>$ip_text</a>|
                                                        , format::mac( $mac => 'html' )
                                                        , $arp->{$mac}
                                                        , delete $conf->{hostname}
@@ -325,17 +375,42 @@ warn "XX conf ",dump( $conf );
                                </form>
                        |;
                }
+
+
        } elsif ( $path =~ m{^/brctl} ) {
-               print $client ok, html::table( -4,
+
+               system 'brctl addif virtual ' . $param->{addif} if $param->{addif};
+               system 'brctl delif virtual ' . $param->{delif} if $param->{delif};
+
+               my $in_virtual;
+
+               my @table =
                        map {
                                my @c = split(/\t+/,$_,4);
                                if ( $#c == 1 ) {
-                                       ( '', '', '', $c[1] )
+                                       $in_virtual->{ $c[1] }++;
+                                       @c = ( '', '', '', $c[1] );
                                } else {
-                                       @c
+                                       $in_virtual->{ $c[3] }++;
+                               }
+                               if ( $c[3] =~ m{\d$} ) {
+                                       $c[3] = qq|<input type=submit name=delif value=$c[3] style="color:red" title="remove $c[3] from bridge">|;
                                }
+                               @c
                        } split(/\n/, `brctl show`)
-               );
+               ;
+
+               my @add_ifs = grep { ! $in_virtual->{$_} && $_ ne 'virtual' } ip::devices_up;
+
+               push @table, ( '', '', '', html::select( 'addif', @add_ifs ) . qq|<input type=submit value=add></form>| );
+
+               print $client ok
+                       , qq|<form>|
+                       , html::table( -4, @table )
+                       , qq|</form>|
+                       ;
+
+
        } elsif ( $path =~ m{^/ip/?(\w+)?} ) {
                print $client ok
                        , join("\n", map { qq|<a href=/ip/$_>$_</a>| } ( qw/link addr route neigh ntable tunnel maddr mroute xfrm/ ))
@@ -355,15 +430,20 @@ warn "XX conf ",dump( $conf );
                }
        } elsif ( $path =~ m{^/wol/(\S+)} ) {
                print $client redirect( "$url/client/$1" ), wol::power_on($1);
+       } elsif ( $path =~ m!^/amt/(\w+)/$RE{net}{IPv4}{-keep}! ) {
+               my ( $run, $ip ) = ( $1, $2 );
+               print $client redirect( "$url/client/$ip" ), amt::RemoteControl( $ip, $run );
        } elsif ( $path =~ m{^/our/(\w+)/(\S+)} ) {
                eval 'our $' . $1 . ' = ' . $2;
                warn $@ if $@;
                print $client redirect($url), qq|<big>$1 = $2</big><br>Location: <a href="$url">$url</a>|;
                server::debug( $debug ) if $1 eq 'debug';
        } elsif ( $path =~ m{^/start_stop/(\S+)} ) {
-               print $client redirect, daemons::start_stop($1);
+               print $client redirect, daemons::start_stop($1,$param);
        } elsif ( $path =~ m{^/action/([^/]+)/(.+)} ) {
-               $1->$2();
+               my ( $package, $method ) = ( $1, $2 );
+               $ENV{nr} = $1 if $package =~ s{\.(\d+)$}{};
+               $package->$method();
                print $client redirect;
        } elsif ( $path =~ m{^/kill/static/(\d+)} ) {
                print $client redirect;
@@ -379,13 +459,13 @@ sub start {
 
        warn 'network ', network::setup();
 
-       daemons::start_stop 'browser', $url;
+       daemons::start_stop 'browser', { url => $url };
        daemons::start_stop $_ foreach ( qw/dhcpd tftpd dnsd syslogd/ );
-       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
 
        my $server = IO::Socket::INET->new(
                        Proto     => 'tcp',
-                       LocalAddr => $server::ip,
+#                      LocalAddr => $server::ip,
                        LocalPort => $httpd::port,
                        Listen    => SOMAXCONN,
                        Reuse     => 1
@@ -393,6 +473,9 @@ sub start {
 
        print "url $url\n";
 
+       syslogd::install_local;
+       client::rebuild_mac_links;
+
        while (1) {
                my $client = $server->accept() || next; # ALARM trickle us
                my $request = <$client>;
@@ -416,7 +499,7 @@ sub start {
                        warn "500 $request";
                }
 
-               print $client menu() if $client->connected;
+               print $client menu() . html_end() if $client->connected;
 
        }