display table header from package keys
[pxelator] / lib / PXElator / client.pm
1 package client;
2
3 use warnings;
4 use strict;
5 use autodie;
6
7 use File::Slurp;
8 use Data::Dump qw/dump/;
9 use File::Path;
10
11 use server;
12 use format;
13 use ip;
14 use ping;
15 use kvm;
16
17 our $debug = $server::debug;
18
19 sub mkbasedir {
20         my $path = shift;
21         $path =~ s{(^.*)/[^/]+$}{$1};
22         mkdir $path unless -d $path;
23         return $path;
24 }
25
26 sub mac_path { $server::conf . '/mac/' . $_[0] }
27 sub  ip_path { $server::conf . '/ip/'  . join('/', @_) }
28 sub conf_value {
29         my $path = shift;
30         my $value;
31         if ( -l $path ) {
32                 $value = readlink $path;
33                 $value =~ s{.*/([^/]+)$}{$1};
34         } elsif ( -f $path ) {
35                 $value = read_file $path;
36         } else {
37                 warn "W: $path not file or symlink\n";
38         }
39         return $value;
40 }
41
42 sub conf {
43         my $ip  = shift;
44         my $name = shift;
45         my ( $default, $value );
46         if ( $#_ == 0 ) {
47                 $value = shift;
48         } elsif ( $#_ == 1 && $_[0] eq 'default' ) {
49                 $default = $_[1]
50         }
51
52         my $path = ip_path $ip;
53         mkdir $path unless -d $path;
54         $path .= '/' . $name;
55
56         if ( defined $value ) {
57                 mkbasedir  $path;
58                 write_file $path, $value;
59                 warn "update $path = $value";
60         } elsif ( ! -e $path && defined $default ) {
61                 mkbasedir  $path;
62                 write_file $path, $default;
63                 warn "default $path = $default";
64                 $value = $default;
65         } elsif ( -f $path ) {
66                 $value = read_file $path;
67                 chomp $value;
68         } else {
69                 warn "# $name missing $path\n" if $debug;
70         }
71         return $value;
72 }
73
74 sub all_conf {
75         my $ip = shift;
76         my $path = ip_path $ip || return;
77         my $conf;
78         foreach my $file ( glob("$path/*"), glob("$path/*/*") ) {
79                 my $name = $file;
80                 $name =~ s{^$path/+}{} || die "can't remove $path from $name";
81                 $conf->{ $name } = read_file $file if -f $file;
82         }
83         return $conf;
84 }
85 sub next_ip($) {
86         my $mac = shift;
87         $mac = format::mac($mac);
88
89         if ( $server::new_clients > 0 ) {
90                 warn "# clients left: ", --$server::new_clients;
91         } else {
92                 warn "W: no new clients accepted";
93                 return '0.0.0.0';
94         }
95
96         my $prefix = $server::ip;
97         $prefix =~ s{\.\d+$}{.};
98         my $addr = $server::ip_from || die;
99         my $ip = $prefix . $addr;
100
101         while ( -e ip_path($ip) || ping::host($ip) ) {
102                 $ip = $prefix . $addr++;
103                 die "all addresses allocated!" if $addr == $server::ip_to;
104                 warn "skip $ip\n";
105         }
106
107         warn "next_ip $ip\n";
108
109         save_ip_mac( $ip, $mac );
110
111         return $ip;
112 }
113
114 sub save_ip_mac {
115         my ($ip,$mac) = @_;
116         $mac = format::mac($mac);
117         return if $mac eq '00:00:00:00:00:00' || $ip eq '0.0.0.0';
118
119         mkdir ip_path($ip) unless -e ip_path($ip);
120
121         my $mac_path = mac_path($mac);
122         unlink $mac_path if -l $mac_path;       # XXX audit?
123         symlink ip_path($ip), $mac_path;
124         write_file( ip_path($ip,'mac'), $mac );
125 }
126
127 sub ip_from_mac($) {
128         my $mac = shift;
129         $mac = format::mac($mac);
130
131         my $mac_path = mac_path $mac;
132         return unless -e $mac_path;
133
134         my $ip;
135
136         if ( -f $mac_path ) {
137                 $ip = read_file $mac_path;
138                 unlink $mac_path;
139                 symlink ip_path($ip), $mac_path;
140                 warn "I: upgrade to mac symlink $mac_path\n";
141         } elsif ( -l $mac_path ) {
142                 $ip = conf_value $mac_path;
143         } else {
144                 die "$mac_path not file or symlink";
145         }
146
147         return $ip;
148 }
149
150 sub mac_from_ip($) {
151         my $ip = shift;
152         conf_value ip_path($ip, 'mac');
153 }
154
155 sub change_ip($$) {
156         my ($old, $new) = @_;
157         return if $old eq $new;
158         my $mac = mac_from_ip($old) || die "no mac for $old";
159         rename ip_path($old), ip_path($new);
160         unlink mac_path($mac);
161         symlink ip_path($new), mac_path($mac);
162         return $new;
163 }
164
165 sub all_ips {
166         sort { ip::to_int($a) cmp ip::to_int($b) }
167         map {
168                 my $ip = $_;
169                 $ip =~ s{^.+/ip/}{};
170                 autocreate_params( $ip );
171                 $ip;
172         } glob("$server::conf/ip/*") 
173 }
174
175 sub remove {
176         my $ip = shift;
177         if ( my $mac = mac_from_ip $ip ) {
178                 unlink "$server::conf/mac/$mac";
179         }
180         rmtree "$server::conf/ip/$ip";
181 }
182
183 sub arp_mac_dev {
184         my $arp = {
185                 map {
186                         my @c = split(/\s+/,$_);
187                         if ( $#c == 5 ) {
188                                 client::save_ip_mac( $c[0], $c[3] );
189                                 ( uc $c[3] => $c[5] )
190                         } else {
191                         }
192                 } read_file('/proc/net/arp')
193         };
194
195         warn "# arp ",dump( $arp );
196         return $arp;
197 }
198
199 sub rebuild_mac_links {
200         warn "# rebuild mac links";
201         foreach my $ip ( all_ips ) {
202                 my $mac = ip_path $ip, 'mac';
203                 if ( -e $mac ) {
204                         $mac = read_file $mac;
205                         chomp $mac;
206                         save_ip_mac( $ip, $mac );
207                         warn "## $ip $mac\n";
208                 }
209         }
210 }
211
212 sub autocreate_params {
213         my $ip = shift;
214         my $mac = mac_from_ip $ip;
215         if ( $mac =~ m{^AC:DE:48:00:00} && ! defined conf( $ip, 'kvm' ) ) {
216                 conf( $ip, 'kvm', default => kvm::nr_from_mac( $mac ) );
217                 warn "# create kvm for $ip";
218         }
219 }
220
221 1;