]>
Commit | Line | Data |
---|---|---|
1 | #!/usr/bin/perl -w | |
2 | ||
3 | use strict; | |
4 | use Getopt::Long; | |
5 | use IPC::Open3; | |
6 | use IO::File; | |
7 | use IO::Dir; | |
8 | use IO::Select; | |
9 | use Cwd 'abs_path'; | |
10 | use Gtk2 '-init'; | |
11 | use Gtk2::Html2; | |
12 | use Gtk2::Gdk::Keysyms; | |
13 | use Encode; | |
14 | ||
15 | my $release = '2.0'; | |
16 | ||
17 | my $kapi = `uname -r`; | |
18 | chomp $kapi; | |
19 | ||
20 | my $opt_testmode; | |
21 | ||
22 | if (!GetOptions ('testmode=s' => \$opt_testmode)) { | |
23 | die "usage error\n"; | |
24 | exit (-1); | |
25 | } | |
26 | ||
27 | my $logfd = IO::File->new (">/tmp/install.log"); | |
28 | ||
29 | my $proxmox_dir = $opt_testmode ? "." : "/var/lib/pve-installer"; | |
30 | ||
31 | $ENV{DEBIAN_FRONTEND} = 'noninteractive'; | |
32 | $ENV{LC_ALL} = 'C'; | |
33 | ||
34 | my ($window, $cmdbox, $inbox, $document, $htmlview); | |
35 | my ($next, $next_fctn, $target_hd, $master_hd); | |
36 | my ($progress, $progress_status); | |
37 | my ($ipaddress, $ip_1, $ip_2, $ip_3, $ip_4); | |
38 | my ($netmask, $mask_1, $mask_2, $mask_3, $mask_4); | |
39 | my ($gateway, $gw_1, $gw_2, $gw_3, $gw_4); | |
40 | my ($dnsserver, $dns_1, $dns_2, $dns_3, $dns_4); | |
41 | my $hostname = 'proxmox'; | |
42 | my $domain = 'domain.tld'; | |
43 | my $cmdline = `cat /proc/cmdline` || ''; | |
44 | my $ipconf; | |
45 | my $country; | |
46 | my $timezone = 'Europe/Vienna'; | |
47 | my $password; | |
48 | my $mailto; | |
49 | my $keymap = 'en-us'; | |
50 | my $cmap; | |
51 | ||
52 | my $filesys = ($cmdline =~ m/\sext4(\s.*)$/) ? 'ext4' : 'ext3'; | |
53 | ||
54 | my $postfix_main_cf = <<_EOD; | |
55 | # See /usr/share/postfix/main.cf.dist for a commented, more complete version | |
56 | ||
57 | myhostname=__FQDN__ | |
58 | ||
59 | smtpd_banner = \$myhostname ESMTP \$mail_name (Debian/GNU) | |
60 | biff = no | |
61 | ||
62 | # appending .domain is the MUA's job. | |
63 | append_dot_mydomain = no | |
64 | ||
65 | # Uncomment the next line to generate "delayed mail" warnings | |
66 | #delay_warning_time = 4h | |
67 | ||
68 | alias_maps = hash:/etc/aliases | |
69 | alias_database = hash:/etc/aliases | |
70 | mydestination = \$myhostname, localhost.\$mydomain, localhost | |
71 | relayhost = | |
72 | mynetworks = 127.0.0.0/8 | |
73 | inet_interfaces = loopback-only | |
74 | recipient_delimiter = + | |
75 | ||
76 | _EOD | |
77 | ||
78 | sub syscmd { | |
79 | my ($cmd) = @_; | |
80 | ||
81 | return run_command ($cmd, undef, undef, 1); | |
82 | } | |
83 | ||
84 | sub run_command { | |
85 | my ($cmd, $func, $input, $noout) = @_; | |
86 | ||
87 | my $cmdtxt; | |
88 | if ($input && ($cmd !~ m/chpasswd/)) { | |
89 | $cmdtxt = "# $cmd <<EOD\n$input"; | |
90 | chomp $cmdtxt; | |
91 | $cmdtxt .= "\nEOD\n"; | |
92 | } else { | |
93 | $cmdtxt = "# $cmd\n"; | |
94 | } | |
95 | print $cmdtxt; | |
96 | STDOUT->flush(); | |
97 | print $logfd $cmdtxt; | |
98 | ||
99 | my $reader = IO::File->new(); | |
100 | my $writer = IO::File->new(); | |
101 | my $error = IO::File->new(); | |
102 | ||
103 | my $orig_pid = $$; | |
104 | ||
105 | my $pid; | |
106 | eval { | |
107 | $pid = open3 ($writer, $reader, $error, $cmd) || die $!; | |
108 | }; | |
109 | ||
110 | my $err = $@; | |
111 | ||
112 | # catch exec errors | |
113 | if ($orig_pid != $$) { | |
114 | POSIX::_exit (1); | |
115 | kill ('KILL', $$); | |
116 | } | |
117 | ||
118 | die $err if $err; | |
119 | ||
120 | print $writer $input if defined $input; | |
121 | close $writer; | |
122 | ||
123 | my $select = new IO::Select; | |
124 | $select->add ($reader); | |
125 | $select->add ($error); | |
126 | ||
127 | my ($ostream, $logout) = ('', '', ''); | |
128 | ||
129 | while ($select->count) { | |
130 | my @handles = $select->can_read (0.2); | |
131 | ||
132 | Gtk2->main_iteration while Gtk2->events_pending; | |
133 | ||
134 | next if !scalar (@handles); # timeout | |
135 | ||
136 | foreach my $h (@handles) { | |
137 | my $buf = ''; | |
138 | my $count = sysread ($h, $buf, 4096); | |
139 | if (!defined ($count)) { | |
140 | my $err = $!; | |
141 | kill (9, $pid); | |
142 | waitpid ($pid, 0); | |
143 | die "command '$cmd' failed: $err"; | |
144 | } | |
145 | $select->remove ($h) if !$count; | |
146 | if ($h eq $reader) { | |
147 | $ostream .= $buf if !($noout || $func); | |
148 | $logout .= $buf; | |
149 | while ($logout =~ s/^([^\010\r\n]*)(\r|\n|(\010)+|\r\n)//s) { | |
150 | my $line = $1; | |
151 | &$func($line) if $func; | |
152 | } | |
153 | ||
154 | } elsif ($h eq $error) { | |
155 | $ostream .= $buf if !($noout || $func); | |
156 | } | |
157 | print $buf; | |
158 | STDOUT->flush(); | |
159 | print $logfd $buf; | |
160 | } | |
161 | } | |
162 | ||
163 | &$func($logout) if $func; | |
164 | ||
165 | my $rv = waitpid ($pid, 0); | |
166 | ||
167 | return $? if $noout; # behave like standard system(); | |
168 | ||
169 | my $ec = ($? >> 8); | |
170 | ||
171 | if ($ec) { | |
172 | die "command '$cmd' failed with exit code $ec"; | |
173 | } | |
174 | ||
175 | return $ostream; | |
176 | } | |
177 | ||
178 | sub detect_country { | |
179 | ||
180 | print "trying to detect country...\n"; | |
181 | open (TMP, "traceroute -N 1 -q 1 -n www.debian.org|"); | |
182 | ||
183 | my $country; | |
184 | ||
185 | my $previous_alarm = alarm (10); | |
186 | eval { | |
187 | local $SIG{ALRM} = sub { die "timed out!\n" }; | |
188 | my $line; | |
189 | while (defined ($line = <TMP>)) { | |
190 | print $logfd "DC TRACEROUTE: $line"; | |
191 | if ($line =~ m/\s*\d+\s+(\d+\.\d+\.\d+\.\d+)\s/) { | |
192 | my $geoip = `geoiplookup $1`; | |
193 | print $logfd "DC GEOIP: $geoip"; | |
194 | if ($geoip =~ m/GeoIP Country Edition:\s*([A-Z]+),/) { | |
195 | $country = lc ($1); | |
196 | last; | |
197 | } | |
198 | } | |
199 | } | |
200 | }; | |
201 | ||
202 | my $err = $@; | |
203 | ||
204 | alarm ($previous_alarm); | |
205 | ||
206 | close (TMP); | |
207 | ||
208 | if ($err) { | |
209 | print "unable to detect country - $err\n"; | |
210 | } elsif ($country) { | |
211 | print "detected country: " . uc($country) . "\n"; | |
212 | } else { | |
213 | print "unable to detect country\n"; | |
214 | } | |
215 | ||
216 | return $country; | |
217 | } | |
218 | ||
219 | sub get_memtotal { | |
220 | ||
221 | open (MEMINFO, "/proc/meminfo"); | |
222 | ||
223 | my $res = 512; # default to 512 if something goes wrong | |
224 | while (my $line = <MEMINFO>) { | |
225 | if ($line =~ m/^MemTotal:\s+(\d+)\s*kB/i) { | |
226 | $res = int ($1 / 1024); | |
227 | } | |
228 | } | |
229 | ||
230 | close (MEMINFO); | |
231 | ||
232 | return $res; | |
233 | } | |
234 | ||
235 | my $total_memory = get_memtotal(); | |
236 | ||
237 | sub link_points_to { | |
238 | my ($src, $dest) = @_; | |
239 | ||
240 | my ($dev1,$ino1) = stat ($src); | |
241 | my ($dev2,$ino2) = stat ($dest); | |
242 | ||
243 | return 0 if !($dev1 && $dev2 && $ino1 && $ino2); | |
244 | ||
245 | return $ino1 == $ino2 && $dev1 == $dev2; | |
246 | } | |
247 | ||
248 | sub find_stable_path { | |
249 | my ($stabledir, $bdev) = @_; | |
250 | ||
251 | my $dh = IO::Dir->new ($stabledir); | |
252 | if ($dh) { | |
253 | while (defined(my $tmp = $dh->read)) { | |
254 | my $path = "$stabledir/$tmp"; | |
255 | if (link_points_to ($path, $bdev)) { | |
256 | return wantarray ? ($path, $tmp) : $path; | |
257 | } | |
258 | } | |
259 | ||
260 | $dh->close; | |
261 | } | |
262 | ||
263 | return wantarray ? () : undef; | |
264 | } | |
265 | ||
266 | sub find_dev_by_uuid { | |
267 | my $bdev = shift; | |
268 | ||
269 | my ($full_path, $name) = find_stable_path ("/dev/disk/by-uuid", $bdev); | |
270 | ||
271 | return $name; | |
272 | } | |
273 | ||
274 | sub hd_list { | |
275 | ||
276 | my $res = (); | |
277 | ||
278 | if ($opt_testmode) { | |
279 | push @$res, [-1, $opt_testmode, int((-s $opt_testmode)/512), "TESTDISK"]; | |
280 | } | |
281 | ||
282 | my $count = 0; | |
283 | ||
284 | foreach my $bd (</sys/block/*>) { | |
285 | next if $bd =~ m|^/sys/block/ram\d+$|; | |
286 | next if $bd =~ m|^/sys/block/loop\d+$|; | |
287 | next if $bd =~ m|^/sys/block/md\d+$|; | |
288 | next if $bd =~ m|^/sys/block/dm-.*$|; | |
289 | next if $bd =~ m|^/sys/block/fd\d+$|; | |
290 | next if $bd =~ m|^/sys/block/sr\d+$|; | |
291 | ||
292 | my $dev = `cat '$bd/dev'`; | |
293 | chomp $dev; | |
294 | ||
295 | next if !$dev; | |
296 | ||
297 | my $info = `udevadm info --path $bd --query all`; | |
298 | next if !$info; | |
299 | ||
300 | next if $info !~ m/^E: DEVTYPE=disk$/m; | |
301 | ||
302 | next if $info =~ m/^E: ID_CDROM/m; | |
303 | ||
304 | my ($name) = $info =~ m/^N: (\S+)$/m; | |
305 | ||
306 | if ($name) { | |
307 | my $real_name = "/dev/$name"; | |
308 | ||
309 | my $size = `cat '$bd/size'`; | |
310 | chomp $size; | |
311 | $size = undef if !($size && $size =~ m/^\d+$/); | |
312 | ||
313 | my $model = `cat '$bd/device/model'`; | |
314 | $model =~ s/^\s+//; | |
315 | $model =~ s/\s+$//; | |
316 | if (length ($model) > 30) { | |
317 | $model = substr ($model, 0, 30); | |
318 | } | |
319 | push @$res, [$count++, $real_name, $size, $model] if $size; | |
320 | } else { | |
321 | print STDERR "ERROR: unable to map device $dev ($bd)\n"; | |
322 | } | |
323 | } | |
324 | ||
325 | return $res; | |
326 | } | |
327 | ||
328 | sub read_cmap { | |
329 | my $countryfn = $opt_testmode ? "/usr/share/pve-manager/country.dat" : | |
330 | "/proxmox/country.dat"; | |
331 | open (TMP, "<$countryfn") || die "unable to open '$countryfn' - $!\n"; | |
332 | my $line; | |
333 | my $country = {}; | |
334 | my $countryhash = {}; | |
335 | my $kmap = {}; | |
336 | my $kmaphash = {}; | |
337 | while (defined ($line = <TMP>)) { | |
338 | if ($line =~ m|^map:([^\s:]+):([^:]+):([^:]+):([^:]+):([^:]+):([^:]*):$|) { | |
339 | $kmap->{$1} = { | |
340 | name => $2, | |
341 | kvm => $3, | |
342 | console => $4, | |
343 | x11 => $5, | |
344 | x11var => $6, | |
345 | }; | |
346 | $kmaphash->{$2} = $1; | |
347 | } elsif ($line =~ m|^([a-z]{2}):([^:]+):([^:]*):([^:]*):$|) { | |
348 | $country->{$1} = { | |
349 | name => $2, | |
350 | kmap => $3, | |
351 | mirror => $4, | |
352 | }; | |
353 | $countryhash->{lc($2)} = $1; | |
354 | } else { | |
355 | warn "unable to parse 'country.dat' line: $line"; | |
356 | } | |
357 | } | |
358 | close (TMP); | |
359 | ||
360 | my $zones = {}; | |
361 | my $cczones = {}; | |
362 | my $zonefn = "/usr/share/zoneinfo/zone.tab"; | |
363 | open (TMP, "<$zonefn") || die "unable to open '$zonefn' - $!\n"; | |
364 | while (defined ($line = <TMP>)) { | |
365 | next if $line =~ m/^\#/; | |
366 | next if $line =~ m/^\s*$/; | |
367 | if ($line =~ m|^([A-Z][A-Z])\s+\S+\s+(([^/]+)/\S+)\s|) { | |
368 | my $cc = lc($1); | |
369 | $cczones->{$cc}->{$2} = 1; | |
370 | $country->{$cc}->{zone} = $2 if !defined ($country->{$cc}->{zone}); | |
371 | $zones->{$2} = 1; | |
372 | ||
373 | } | |
374 | } | |
375 | close (TMP); | |
376 | ||
377 | return { | |
378 | zones => $zones, | |
379 | cczones => $cczones, | |
380 | country => $country, | |
381 | countryhash => $countryhash, | |
382 | kmap => $kmap, | |
383 | kmaphash => $kmaphash, | |
384 | } | |
385 | } | |
386 | ||
387 | # search for Harddisks | |
388 | my $hds = hd_list (); | |
389 | ||
390 | sub hd_size { | |
391 | my ($dev) = @_; | |
392 | ||
393 | foreach my $hd (@$hds) { | |
394 | my ($disk, $devname, $size, $model) = @$hd; | |
395 | return int($size/2) if $devname eq $dev; | |
396 | } | |
397 | ||
398 | die "no such device '$dev'"; | |
399 | } | |
400 | ||
401 | # find the master boot disk - return the first found scsi/ide disk | |
402 | sub find_master { | |
403 | my ($target_hd) = @_; | |
404 | ||
405 | foreach my $hd (sort { ${$a}[1] cmp ${$b}[1] } @$hds) { | |
406 | my ($disk, $devname) = @$hd; | |
407 | next if $disk < 0; | |
408 | ||
409 | if ($target_hd =~ m|^/dev/sd|) { | |
410 | return $devname if $devname =~ m|^/dev/sd|; | |
411 | } elsif ($target_hd =~ m|^/dev/hd|) { | |
412 | return $devname if $devname =~ m|^/dev/hd|; | |
413 | } elsif ($target_hd =~ m|^/dev/i2o/|) { | |
414 | return $devname if $devname =~ m|^/dev/i2o/|; | |
415 | } elsif ($target_hd =~ m|^/dev/ataraid/|) { | |
416 | return $devname if $devname =~ m|^/dev/ataraid/|; | |
417 | } elsif ($target_hd =~ m|^/dev/ida/|) { | |
418 | return $devname if $devname =~ m|^/dev/ida/|; | |
419 | } elsif ($target_hd =~ m|^/dev/cciss/|) { | |
420 | return $devname if $devname =~ m|^/dev/cciss/|; | |
421 | } elsif ($target_hd =~ m|^/dev/rd/|) { | |
422 | return $devname if $devname =~ m|^/dev/rd/|; | |
423 | } | |
424 | } | |
425 | ||
426 | return $target_hd; | |
427 | } | |
428 | ||
429 | sub get_partition_dev { | |
430 | my ($target_hd, $partnum) = @_; | |
431 | ||
432 | if ($target_hd =~ m|^/dev/[hxsev]d[a-z]$|) { | |
433 | return "${target_hd}$partnum"; | |
434 | } elsif ($target_hd =~ m|^/dev/[^/]+/c\d+d\d+$|) { | |
435 | return "${target_hd}p$partnum"; | |
436 | } elsif ($target_hd =~ m|^/dev/[^/]+/d\d+$|) { | |
437 | return "${target_hd}p$partnum"; | |
438 | } elsif ($target_hd =~ m|^/dev/[^/]+/hd[a-z]$|) { | |
439 | return "${target_hd}$partnum"; | |
440 | } else { | |
441 | die "unable to get device for partition $partnum on device $target_hd\n"; | |
442 | } | |
443 | ||
444 | } | |
445 | ||
446 | sub get_boot_part { | |
447 | my ($target_hd, $gpt) = @_; | |
448 | ||
449 | return get_partition_dev ($target_hd, $gpt ? 2 : 1); | |
450 | } | |
451 | ||
452 | sub get_lvm_part { | |
453 | my ($target_hd, $gpt) = @_; | |
454 | ||
455 | return get_partition_dev ($target_hd, $gpt ? 3 : 2); | |
456 | } | |
457 | ||
458 | sub write_config { | |
459 | my ($text, $filename) = @_; | |
460 | ||
461 | my $fd = IO::File->new (">$filename") || | |
462 | die "unable to open file '$filename' - $!"; | |
463 | print $fd $text; | |
464 | $fd->close(); | |
465 | } | |
466 | ||
467 | sub update_progress { | |
468 | my ($frac, $start, $end, $text) = @_; | |
469 | ||
470 | my $part = $end - $start; | |
471 | my $res = $start + $frac*$part; | |
472 | ||
473 | $progress->set_fraction ($res); | |
474 | $progress->set_text (sprintf ("%d%%", int ($res*100))); | |
475 | $progress_status->set_text ($text) if defined ($text); | |
476 | ||
477 | Gtk2->main_iteration while Gtk2->events_pending; | |
478 | } | |
479 | ||
480 | sub create_filesystem { | |
481 | my ($dev, $name, $type, $start, $end, $fs, $fe, $opts) = @_; | |
482 | ||
483 | $opts = '' if !$opts; | |
484 | ||
485 | my $range = $end - $start; | |
486 | my $rs = $start + $range*$fs; | |
487 | my $re = $start + $range*$fe; | |
488 | my $max = 0; | |
489 | ||
490 | update_progress (0, $rs, $re, "creating $name filesystem"); | |
491 | ||
492 | run_command ("mkfs.$type $opts -F $dev", sub { | |
493 | my $line = shift; | |
494 | ||
495 | if ($line =~ m/Writing inode tables:\s+(\d+)\/(\d+)/) { | |
496 | $max = $2; | |
497 | } elsif ($max && $line =~ m/(\d+)\/$max/) { | |
498 | update_progress (($1/$max)*0.9, $rs, $re); | |
499 | } elsif ($line =~ m/Creating journal.*done/) { | |
500 | update_progress (0.95, $rs, $re); | |
501 | } elsif ($line =~ m/Writing superblocks and filesystem.*done/) { | |
502 | update_progress (1, $rs, $re); | |
503 | } | |
504 | }); | |
505 | } | |
506 | ||
507 | sub debconfig_set { | |
508 | my ($targetdir, $dcdata) = @_; | |
509 | ||
510 | my $cfgfile = "/tmp/debconf.txt"; | |
511 | write_config ($dcdata, "$targetdir/$cfgfile"); | |
512 | syscmd ("chroot $targetdir debconf-set-selections $cfgfile"); | |
513 | unlink "$targetdir/$cfgfile"; | |
514 | } | |
515 | ||
516 | sub diversion_add { | |
517 | my ($targetdir, $cmd, $new_cmd) = @_; | |
518 | ||
519 | syscmd ("chroot $targetdir dpkg-divert --package proxmox " . | |
520 | "--add --rename $cmd") == 0 || | |
521 | die "unable to exec dpkg-divert\n"; | |
522 | ||
523 | syscmd ("ln -sf ${new_cmd} $targetdir/$cmd") == 0 || | |
524 | die "unable to link diversion to ${new_cmd}\n"; | |
525 | } | |
526 | ||
527 | sub diversion_remove { | |
528 | my ($targetdir, $cmd) = @_; | |
529 | ||
530 | syscmd ("mv $targetdir/${cmd}.distrib $targetdir/${cmd};") == 0 || | |
531 | die "unable to remove $cmd diversion\n"; | |
532 | ||
533 | syscmd ("chroot $targetdir dpkg-divert --remove $cmd") == 0 || | |
534 | die "unable to remove $cmd diversion\n"; | |
535 | } | |
536 | ||
537 | sub extract_data { | |
538 | my ($tgzfile, $targetdir) = @_; | |
539 | ||
540 | die "target '$targetdir' does not exist\n" if ! -d $targetdir; | |
541 | ||
542 | my $rootdev; | |
543 | my $bootdev; | |
544 | my $datadev; | |
545 | my $swapfile; | |
546 | ||
547 | my $ptype = 'msdos'; | |
548 | ||
549 | eval { | |
550 | ||
551 | my $maxper = 0.25; | |
552 | ||
553 | update_progress (0, 0, $maxper, "create partitions"); | |
554 | ||
555 | if ( -b $target_hd) { | |
556 | syscmd ("dd if=/dev/zero of=${target_hd} bs=512 count=256"); | |
557 | my $hdsize = hd_size ($target_hd); # size in blocks (1024 bytes) | |
558 | ||
559 | if ($hdsize >= 2*1024*1024*1024) { # MBR can only handle 2 TB | |
560 | $ptype = 'gpt'; | |
561 | } | |
562 | ||
563 | my $bootsize_mb = 512; | |
564 | my $bootsize = $bootsize_mb * 1024; | |
565 | my $hdsize_mb = $hdsize/1024; | |
566 | ||
567 | my $pcmd = "parted --align optimal ${target_hd}"; | |
568 | $pcmd .= " unit MiB"; | |
569 | $pcmd .= " mklabel $ptype"; | |
570 | ||
571 | my $pnum = 1; | |
572 | ||
573 | if ($ptype eq 'gpt') { | |
574 | $pcmd .= " mkpart primary 1 2"; | |
575 | $pcmd .= " set $pnum bios_grub on"; | |
576 | $pnum++; | |
577 | }; | |
578 | ||
579 | $pcmd .= " mkpart primary ext2 $pnum ${bootsize_mb}"; | |
580 | $pcmd .= " set $pnum boot on"; | |
581 | $pnum++; | |
582 | ||
583 | ||
584 | $pcmd .= " mkpart primary ext2 ${bootsize_mb} ${hdsize_mb}"; | |
585 | $pcmd .= " set $pnum lvm on"; | |
586 | ||
587 | syscmd($pcmd) == 0 || | |
588 | die "unable to partition harddisk '${target_hd}'\n"; | |
589 | ||
590 | sleep(1); # give kernel time to reread part table | |
591 | ||
592 | my $lvmdev = get_lvm_part($target_hd, $ptype eq 'gpt'); | |
593 | ||
594 | $rootdev = '/dev/pve/root'; | |
595 | $datadev = '/dev/pve/data'; | |
596 | $swapfile = '/dev/pve/swap'; | |
597 | ||
598 | # we use --metadatasize 250k, which reseults in "pe_start = 512" | |
599 | # so pe_start is aligned on a 128k boundary (advantage for SSDs) | |
600 | syscmd ("/sbin/pvcreate --metadatasize 250k -y -ff $lvmdev") == 0 || | |
601 | die "unable to initialize physical volume $lvmdev"; | |
602 | syscmd ("/sbin/vgcreate pve $lvmdev") == 0 || | |
603 | die "unable to create volume group"; | |
604 | ||
605 | my $hdgb = int($hdsize/(1024*1024)); | |
606 | die "hardisk too small (${hdgb}GB)" if $hdgb < 4; | |
607 | ||
608 | my $swapsize; | |
609 | if ($cmdline =~ m/swapsize=(\d+)[\s\n]/i) { | |
610 | $swapsize=$1*1024*1024; | |
611 | } else { | |
612 | my $ss = int ($total_memory / 1024); | |
613 | $ss = 4 if $ss < 4; | |
614 | $ss = ($hdgb/8) if $ss > ($hdgb/8); | |
615 | $swapsize = $ss*1024*1024; | |
616 | } | |
617 | ||
618 | my $space = (($hdgb > 128) ? 16 : ($hdgb/8))*1024*1024; | |
619 | ||
620 | my $maxroot; | |
621 | if ($cmdline =~ m/maxroot=(\d+)[\s\n]/i) { | |
622 | $maxroot = $1; | |
623 | } else { | |
624 | $maxroot = 96; | |
625 | } | |
626 | ||
627 | my $rootsize = (($hdgb > ($maxroot*4)) ? $maxroot : $hdgb/4)*1024*1024; | |
628 | my $rest = int($hdsize) - $bootsize - $swapsize - $space - $rootsize; # in KB | |
629 | syscmd ("/sbin/lvcreate -L${swapsize}K -nswap pve") == 0 || | |
630 | die "unable to create swap volume"; | |
631 | ||
632 | syscmd ("/sbin/lvcreate -L${rootsize}K -nroot pve") == 0 || | |
633 | die "unable to create root volume"; | |
634 | ||
635 | syscmd ("/sbin/lvcreate -L${rest}K -ndata pve") == 0 || | |
636 | die "unable to create data volume"; | |
637 | ||
638 | syscmd ("/sbin/vgchange -a y pve") == 0 || | |
639 | die "unable to activate volume group"; | |
640 | ||
641 | } else { | |
642 | $rootdev = $target_hd; | |
643 | syscmd ("umount $rootdev"); | |
644 | } | |
645 | ||
646 | update_progress (0.03, 0, $maxper, "create swap space"); | |
647 | if ($swapfile) { | |
648 | syscmd ("mkswap $swapfile") == 0 || | |
649 | die "unable to create swap space\n"; | |
650 | } | |
651 | ||
652 | update_progress (0.05, 0, $maxper, "creating filesystems"); | |
653 | ||
654 | if ( -b $target_hd) { | |
655 | $bootdev = get_boot_part ($target_hd, $ptype eq 'gpt'); | |
656 | create_filesystem ($bootdev, 'boot', $filesys, 0.05, $maxper, 0, 0.1); | |
657 | create_filesystem ($rootdev, 'root', $filesys, 0.05, $maxper, 0.1, 0.5); | |
658 | create_filesystem ($datadev, 'data', $filesys, 0.05, $maxper, 0.5, 1, '-m 0'); | |
659 | } else { | |
660 | create_filesystem ($rootdev, 'root', $filesys, 0.05, $maxper, 0, 1); | |
661 | } | |
662 | ||
663 | update_progress (1, 0.05, $maxper, "mounting target $rootdev"); | |
664 | ||
665 | if ( -b $target_hd) { | |
666 | ||
667 | # trigger udev to create /dev/disk/by-uuid | |
668 | syscmd ("udevadm trigger --subsystem-match block"); | |
669 | syscmd ("udevadm settle --timeout 10"); | |
670 | ||
671 | syscmd ("mount -n $rootdev -o noatime,barrier=0 $targetdir") == 0 || | |
672 | die "unable to mount $rootdev\n"; | |
673 | ||
674 | mkdir "$targetdir/boot"; | |
675 | syscmd ("mount -n $bootdev -o noatime,barrier=0 $targetdir/boot") == 0 || | |
676 | die "unable to mount $bootdev\n"; | |
677 | ||
678 | mkdir "$targetdir/var"; | |
679 | mkdir "$targetdir/var/lib"; | |
680 | mkdir "$targetdir/var/lib/vz"; | |
681 | syscmd ("mount -n $datadev $targetdir/var/lib/vz") == 0 || | |
682 | die "unable to mount $datadev\n"; | |
683 | ||
684 | } else { | |
685 | syscmd ("mount $rootdev $targetdir -o loop,noatime,barrier=0") == 0 || | |
686 | die "unable to mount $rootdev\n"; | |
687 | } | |
688 | ||
689 | display_html ("extract2-rulesystem.htm"); | |
690 | update_progress (1, 0.05, $maxper, "extracting base system"); | |
691 | ||
692 | my ($dev,$ino,$mode,$nlink,$uid,$gid,$rdev,$size) = stat ($tgzfile); | |
693 | $ino || die "unable to open file '$tgzfile' - $!\n"; | |
694 | ||
695 | my $files; | |
696 | if ($opt_testmode) { | |
697 | $files = `cat /pve/$release/install/pve-base.cnt`; | |
698 | } else { | |
699 | $files = `cat /proxmox/pve-base.cnt`; | |
700 | } | |
701 | ||
702 | my $per = 0; | |
703 | my $count = 0; | |
704 | ||
705 | run_command ("tar xvf $tgzfile -C $targetdir", sub { | |
706 | my $line = shift; | |
707 | $count++; | |
708 | my $nper = int (($count *100)/$files); | |
709 | if ($nper != $per) { | |
710 | $per = $nper; | |
711 | my $frac = $per > 100 ? 100 : $per/100; | |
712 | update_progress ($frac, $maxper, 0.5); | |
713 | } | |
714 | }); | |
715 | ||
716 | syscmd ("mount -n -t tmpfs tmpfs $targetdir/tmp") == 0 || | |
717 | die "unable to mount tmpfs on $targetdir/tmp\n"; | |
718 | syscmd ("mount -n -t proc proc $targetdir/proc") == 0 || | |
719 | die "unable to mount proc on $targetdir/proc\n"; | |
720 | syscmd ("mount -n -t sysfs sysfs $targetdir/sys") == 0 || | |
721 | die "unable to mount sysfs on $targetdir/sys\n"; | |
722 | ||
723 | display_html ("extract3-spam.htm"); | |
724 | update_progress (1, $maxper, 0.5, "configuring base system"); | |
725 | ||
726 | # configure hosts | |
727 | ||
728 | my $hosts = | |
729 | "127.0.0.1 localhost.localdomain localhost\n" . | |
730 | "$ipaddress $hostname.$domain $hostname pvelocalhost\n\n" . | |
731 | "# The following lines are desirable for IPv6 capable hosts\n\n" . | |
732 | "::1 ip6-localhost ip6-loopback\n" . | |
733 | "fe00::0 ip6-localnet\n" . | |
734 | "ff00::0 ip6-mcastprefix\n" . | |
735 | "ff02::1 ip6-allnodes\n" . | |
736 | "ff02::2 ip6-allrouters\n" . | |
737 | "ff02::3 ip6-allhosts\n"; | |
738 | ||
739 | write_config ($hosts, "$targetdir/etc/hosts"); | |
740 | ||
741 | write_config ("$hostname\n", "$targetdir/etc/hostname"); | |
742 | ||
743 | syscmd ("/bin/hostname $hostname") if !$opt_testmode; | |
744 | ||
745 | # configure interfaces | |
746 | ||
747 | my $ifaces = | |
748 | "auto lo\niface lo inet loopback\n\n" . | |
749 | "auto vmbr0\niface vmbr0 inet static\n" . | |
750 | "\taddress $ipaddress\n" . | |
751 | "\tnetmask $netmask\n" . | |
752 | "\tgateway $gateway\n" . | |
753 | "\tbridge_ports eth0\n" . | |
754 | "\tbridge_stp off\n" . | |
755 | "\tbridge_fd 0\n"; | |
756 | ||
757 | write_config ($ifaces, "$targetdir/etc/network/interfaces"); | |
758 | ||
759 | # configure dns | |
760 | ||
761 | my $resolfconf = "search $domain\nnameserver $dnsserver\n"; | |
762 | write_config ($resolfconf, "$targetdir/etc/resolv.conf"); | |
763 | ||
764 | # try to use UUID=XXX for boot device | |
765 | my $boot_uuid = $bootdev; | |
766 | if (my $uuid = find_dev_by_uuid ($bootdev)) { | |
767 | $boot_uuid = "UUID=$uuid"; | |
768 | } | |
769 | ||
770 | # configure fstab | |
771 | ||
772 | my $fstab = | |
773 | "# <file system> <mount point> <type> <options> <dump> <pass>\n" . | |
774 | "$rootdev / $filesys errors=remount-ro 0 1\n"; | |
775 | ||
776 | $fstab .= "$datadev /var/lib/vz $filesys defaults 0 1\n" if $datadev; | |
777 | ||
778 | $fstab .= "${boot_uuid} /boot $filesys defaults 0 1\n" if $boot_uuid; | |
779 | ||
780 | $fstab .= "$swapfile none swap sw 0 0\n" if $swapfile; | |
781 | ||
782 | $fstab .= "proc /proc proc defaults 0 0\n"; | |
783 | ||
784 | write_config ($fstab, "$targetdir/etc/fstab"); | |
785 | write_config ("", "$targetdir/etc/mtab"); | |
786 | ||
787 | syscmd ("cp ${proxmox_dir}/policy-disable-rc.d " . | |
788 | "$targetdir/usr/sbin/policy-rc.d") == 0 || | |
789 | die "unable to copy policy-rc.d\n"; | |
790 | syscmd ("cp ${proxmox_dir}/fake-start-stop-daemon " . | |
791 | "$targetdir/sbin/") == 0 || | |
792 | die "unable to copy start-stop-daemon\n"; | |
793 | ||
794 | diversion_add ($targetdir, "/sbin/start-stop-daemon", "/sbin/fake-start-stop-daemon"); | |
795 | diversion_add ($targetdir, "/usr/sbin/update-grub", "/bin/true"); | |
796 | diversion_add ($targetdir, "/usr/sbin/update-initramfs", "/bin/true"); | |
797 | ||
798 | syscmd ("touch $targetdir/proxmox_install_mode"); | |
799 | ||
800 | debconfig_set ($targetdir, <<_EOD); | |
801 | locales locales/default_environment_locale select en_US.UTF-8 | |
802 | locales locales/locales_to_be_generated select en_US.UTF-8 UTF-8 | |
803 | samba-common samba-common/dhcp boolean false | |
804 | samba-common samba-common/workgroup string WORKGROUP | |
805 | postfix postfix/main_mailer_type select Local only | |
806 | console-data console-data/keymap/policy select Don\'t touch keymap | |
807 | _EOD | |
808 | ||
809 | my $pkgdir = $opt_testmode ? "packages" : "/proxmox/packages"; | |
810 | my $pkg_count = 0; | |
811 | while (<$pkgdir/*.deb>) { $pkg_count++ }; | |
812 | ||
813 | $count = 0; | |
814 | while (<$pkgdir/*.deb>) { | |
815 | chomp; | |
816 | my $path = $_; | |
817 | my ($deb) = $path =~ m/$pkgdir\/(.*\.deb)/; | |
818 | update_progress ($count/$pkg_count, 0.5, 0.75, "extracting $deb"); | |
819 | print "extracting: $deb\n"; | |
820 | syscmd ("cp $path $targetdir/tmp/$deb") == 0 || | |
821 | die "installation of package $deb failed\n"; | |
822 | syscmd ("chroot $targetdir dpkg --force-depends --no-triggers --unpack /tmp/$deb") == 0 || | |
823 | die "installation of package $deb failed\n"; | |
824 | update_progress ((++$count)/$pkg_count, 0.5, 0.75); | |
825 | } | |
826 | ||
827 | display_html ("extract4-virus.htm"); | |
828 | ||
829 | my $cmd = "chroot $targetdir dpkg --force-confold --configure -a"; | |
830 | $count = 0; | |
831 | run_command ($cmd, sub { | |
832 | my $line = shift; | |
833 | if ($line =~ m/Setting up\s+(\S+)/) { | |
834 | update_progress ((++$count)/$pkg_count, 0.75, 0.95, | |
835 | "configuring $1"); | |
836 | } | |
837 | }); | |
838 | ||
839 | debconfig_set ($targetdir, <<_EOD); | |
840 | postfix postfix/main_mailer_type select No configuration | |
841 | _EOD | |
842 | ||
843 | unlink "$targetdir/etc/mailname"; | |
844 | $postfix_main_cf =~ s/__FQDN__/${hostname}.${domain}/; | |
845 | write_config ($postfix_main_cf, "$targetdir/etc/postfix/main.cf"); | |
846 | ||
847 | # make sure we have all postfix directories | |
848 | syscmd ("chroot $targetdir /usr/sbin/postfix check"); | |
849 | # cleanup mail queue | |
850 | syscmd ("chroot $targetdir /usr/sbin/postsuper -d ALL"); | |
851 | ||
852 | unlink "$targetdir/proxmox_install_mode"; | |
853 | ||
854 | # disable bacula-fd | |
855 | syscmd ("touch '$targetdir/etc/bacula/do_not_run'"); | |
856 | ||
857 | # set timezone | |
858 | unlink ("$targetdir/etc/localtime"); | |
859 | symlink ("/usr/share/zoneinfo/$timezone", "$targetdir/etc/localtime"); | |
860 | write_config ("$timezone\n", "$targetdir/etc/timezone"); | |
861 | ||
862 | # set console keymap | |
863 | if (my $kmapfile = $cmap->{kmap}->{$keymap}->{console}) { | |
864 | syscmd ("chroot $targetdir /usr/sbin/install-keymap '/usr/share/keymaps/i386/$kmapfile'"); | |
865 | } | |
866 | ||
867 | # enable apache port redirect | |
868 | syscmd ("chroot $targetdir a2ensite pve-redirect.conf"); | |
869 | ||
870 | # set apt mirror | |
871 | if (my $mirror = $cmap->{country}->{$country}->{mirror}) { | |
872 | my $fn = "$targetdir/etc/apt/sources.list"; | |
873 | syscmd ("sed -i 's/ftp\\.debian\\.org/$mirror/' '$fn'"); | |
874 | } | |
875 | ||
876 | # create extended_states for apt (avoid cron job warning if that | |
877 | # file does not exist) | |
878 | write_config ('', "$targetdir/var/lib/apt/extended_states"); | |
879 | ||
880 | # save installer settings | |
881 | my $ucc = uc ($country); | |
882 | debconfig_set ($targetdir, <<_EOD); | |
883 | pve-manager pve-manager/country string $ucc | |
884 | _EOD | |
885 | ||
886 | update_progress (0.8, 0.95, 1, "make system bootable"); | |
887 | ||
888 | # update default grub settings | |
889 | syscmd ("sed -i -e 's/^GRUB_DISTRIBUTOR.*/GRUB_DISTRIBUTOR=\"Proxmox Virtual Environment\"/' -e 's/#GRUB_DISABLE_LINUX_RECOVERY=.*/GRUB_DISABLE_LINUX_RECOVERY=\"true\"/' $targetdir/etc/default/grub"); | |
890 | ||
891 | diversion_remove ($targetdir, "/usr/sbin/update-grub"); | |
892 | diversion_remove ($targetdir, "/usr/sbin/update-initramfs"); | |
893 | ||
894 | if (!$opt_testmode && -b $target_hd) { | |
895 | ||
896 | unlink ("$targetdir/etc/mtab"); | |
897 | symlink ("/proc/mounts", "$targetdir/etc/mtab"); | |
898 | syscmd ("mount -n --bind /dev $targetdir/dev"); | |
899 | ||
900 | syscmd ("chroot $targetdir /usr/sbin/update-initramfs -c -k $kapi") == 0 || | |
901 | die "unable to install initramfs\n"; | |
902 | ||
903 | syscmd ("chroot $targetdir /usr/sbin/grub-install --no-floppy '(hd0)'") == 0 || | |
904 | die "unable to install the boot loader\n"; | |
905 | ||
906 | syscmd ("chroot $targetdir /usr/sbin/update-grub") == 0 || | |
907 | die "unable to install the boot loader\n"; | |
908 | ||
909 | syscmd ("umount $targetdir/dev"); | |
910 | } | |
911 | ||
912 | ||
913 | # cleanup | |
914 | ||
915 | # hack: remove dead.letter from sshd installation | |
916 | syscmd ("rm -rf $targetdir/dead.letter"); | |
917 | ||
918 | unlink ("$targetdir/etc/mtab"); | |
919 | syscmd ("touch $targetdir/etc/mtab"); | |
920 | ||
921 | unlink "$targetdir/usr/sbin/policy-rc.d"; | |
922 | ||
923 | diversion_remove ($targetdir, "/sbin/start-stop-daemon"); | |
924 | ||
925 | # set root password | |
926 | my $octets = encode("utf-8", $password); | |
927 | run_command ("chroot $targetdir /usr/sbin/chpasswd", undef, | |
928 | "root:$octets\n"); | |
929 | ||
930 | # create pmxcfs DB | |
931 | ||
932 | my $tmpdir = "$targetdir/tmp/pve"; | |
933 | mkdir $tmpdir; | |
934 | ||
935 | # write vnc keymap to datacenter.cfg | |
936 | my $vnckmap = $cmap->{kmap}->{$keymap}->{kvm} || 'en-us'; | |
937 | write_config ("keyboard: $vnckmap\n", | |
938 | "$tmpdir/datacenter.cfg"); | |
939 | ||
940 | # save admin email | |
941 | write_config ("user:root\@pam:1:0:::${mailto}::\n", | |
942 | "$tmpdir/user.cfg"); | |
943 | ||
944 | run_command("chroot $targetdir /usr/bin/create_pmxcfs_db /tmp/pve /var/lib/pve-cluster/config.db"); | |
945 | ||
946 | syscmd ("rm -rf $tmpdir"); | |
947 | }; | |
948 | ||
949 | my $err = $@; | |
950 | ||
951 | update_progress (1, 0, 1, ""); | |
952 | ||
953 | print $err if $err; | |
954 | ||
955 | if ($opt_testmode) { | |
956 | syscmd ("chroot $targetdir /usr/bin/dpkg-query -W --showformat='\${package}\n'> pve-final.pkglist"); | |
957 | } | |
958 | ||
959 | syscmd ("umount $targetdir/boot"); | |
960 | syscmd ("umount $targetdir/var/lib/vz"); | |
961 | syscmd ("umount $targetdir/tmp"); | |
962 | syscmd ("umount $targetdir/proc"); | |
963 | syscmd ("umount $targetdir/sys"); | |
964 | syscmd ("umount -d $targetdir"); | |
965 | ||
966 | die $err if $err; | |
967 | } | |
968 | ||
969 | sub display_html { | |
970 | my ($filename) = @_; | |
971 | ||
972 | $htmlview->set_document(undef); | |
973 | $document->clear; | |
974 | $htmlview->set_document($document); | |
975 | ||
976 | $document->open_stream ("text/html"); | |
977 | ||
978 | my $fn = "${proxmox_dir}/html/$filename"; | |
979 | open (HTML, $fn) || | |
980 | die "unable to open file '$fn' - $!\n"; | |
981 | while (<HTML>) { $document->write_stream ($_); } | |
982 | close (HTML); | |
983 | ||
984 | $document->close_stream; | |
985 | } | |
986 | ||
987 | sub set_next { | |
988 | my ($text, $fctn) = @_; | |
989 | ||
990 | $next_fctn = $fctn; | |
991 | $text = "_Next" if !$text; | |
992 | $next->set_label ($text); | |
993 | ||
994 | $next->grab_focus (); | |
995 | } | |
996 | ||
997 | sub url_requested { | |
998 | my ($doc, $url, $stream) = @_; | |
999 | ||
1000 | $stream->set_cancel_func (sub {}); # hack: avoid warning | |
1001 | ||
1002 | my $path = "${proxmox_dir}/html/$url"; | |
1003 | ||
1004 | if (-f $path) { | |
1005 | open (HTMLTMP, $path) || | |
1006 | die "unable to open file '$path' - $! "; | |
1007 | my $buf; | |
1008 | while (my $i = read (HTMLTMP, $buf, 4096)) { | |
1009 | $stream->write ($buf); | |
1010 | Gtk2->main_iteration while Gtk2->events_pending; | |
1011 | } | |
1012 | close (HTMLTMP); | |
1013 | } | |
1014 | ||
1015 | #$stream->close(); # hack: dont close - avoid crash | |
1016 | } | |
1017 | ||
1018 | sub create_main_window { | |
1019 | ||
1020 | $window = Gtk2::Window->new (); | |
1021 | $window->set_default_size (1024, 768); | |
1022 | $window->set_decorated (0) if !$opt_testmode; | |
1023 | ||
1024 | my $vbox = Gtk2::VBox->new (0, 0); | |
1025 | ||
1026 | my $image = Gtk2::Image->new_from_file ("${proxmox_dir}/proxlogo.xpm"); | |
1027 | $vbox->pack_start ($image, 0, 0, 0); | |
1028 | ||
1029 | my $hbox = Gtk2::HBox->new (0, 0); | |
1030 | $vbox->pack_start ($hbox, 1, 1, 0); | |
1031 | ||
1032 | my $f1 = Gtk2::Frame->new (); | |
1033 | $f1->set_shadow_type ('in'); | |
1034 | $hbox->pack_start ($f1, 1, 1, 0); | |
1035 | ||
1036 | my $sep1 = Gtk2::HSeparator->new; | |
1037 | $vbox->pack_start ($sep1, 0, 0, 0); | |
1038 | ||
1039 | $cmdbox = Gtk2::HBox->new (); | |
1040 | $vbox->pack_start ($cmdbox, 0, 0, 10); | |
1041 | ||
1042 | $next = Gtk2::Button->new ('_Next'); | |
1043 | $next->signal_connect (clicked => sub { &$next_fctn (); }); | |
1044 | $cmdbox->pack_end ($next, 0, 0, 10); | |
1045 | my $abort = Gtk2::Button->new ('_Abort'); | |
1046 | $abort->can_focus (0); | |
1047 | $cmdbox->pack_start ($abort, 0, 0, 10); | |
1048 | $abort->signal_connect (clicked => sub { exit (-1); }); | |
1049 | ||
1050 | my $vbox2 = Gtk2::VBox->new (0, 0); | |
1051 | $f1->add ($vbox2); | |
1052 | ||
1053 | $htmlview = new Gtk2::Html2::View; | |
1054 | # hack: create a separate style - else Gtk2::Html2 modifies | |
1055 | # main window style | |
1056 | my $rcs1 = Gtk2::RcStyle->new; | |
1057 | $htmlview->modify_style ($rcs1); | |
1058 | ||
1059 | $document = new Gtk2::Html2::Document; | |
1060 | $document->signal_connect (request_url => \&url_requested); | |
1061 | ||
1062 | $document->clear; | |
1063 | $htmlview->set_document ($document); | |
1064 | ||
1065 | my $hbox2 = Gtk2::HBox->new (0, 0); | |
1066 | $hbox2->pack_start ($htmlview, 1, 1, 0); | |
1067 | ||
1068 | $vbox2->pack_start ($hbox2, 1, 1, 0); | |
1069 | ||
1070 | my $vbox3 = Gtk2::VBox->new (0, 0); | |
1071 | $vbox2->pack_start ($vbox3, 0, 0, 0); | |
1072 | ||
1073 | my $sep2 = Gtk2::HSeparator->new; | |
1074 | $vbox3->pack_start ($sep2, 0, 0, 0); | |
1075 | ||
1076 | $inbox = Gtk2::HBox->new (0, 0); | |
1077 | $vbox3->pack_start ($inbox, 0, 0, 0); | |
1078 | ||
1079 | $window->add ($vbox); | |
1080 | ||
1081 | $window->show_all; | |
1082 | $window->realize (); | |
1083 | } | |
1084 | ||
1085 | sub cleanup_view { | |
1086 | my $list = $inbox->get_children; | |
1087 | foreach my $c ($list) { | |
1088 | next if !defined ($c); | |
1089 | $inbox->remove ($c); | |
1090 | } | |
1091 | } | |
1092 | ||
1093 | sub check_num { | |
1094 | my ($entry, $event) = @_; | |
1095 | ||
1096 | my $val = $event->keyval; | |
1097 | ||
1098 | if ($val == ord '.') { | |
1099 | $entry->parent->child_focus ('right'); | |
1100 | return 1; | |
1101 | } | |
1102 | ||
1103 | if ($val == $Gtk2::Gdk::Keysyms{ISO_Left_Tab} || | |
1104 | $val == $Gtk2::Gdk::Keysyms{Shift_L} || | |
1105 | $val == $Gtk2::Gdk::Keysyms{Tab} || | |
1106 | $val == $Gtk2::Gdk::Keysyms{BackSpace} || | |
1107 | $val == $Gtk2::Gdk::Keysyms{Delete} || | |
1108 | ($val >= ord '0' && $val <= ord '9') || | |
1109 | ($val >= $Gtk2::Gdk::Keysyms{KP_0} && | |
1110 | $val <= $Gtk2::Gdk::Keysyms{KP_9})) { | |
1111 | return undef; | |
1112 | } | |
1113 | ||
1114 | return 1; | |
1115 | } | |
1116 | ||
1117 | sub check_range { | |
1118 | my ($entry, $event) = @_; | |
1119 | ||
1120 | my $text = $entry->get_text; | |
1121 | if (!defined($text) || ($text !~ m/^(\d+)$/) || ($1 > 255)) { | |
1122 | $entry->set_text ($entry->{default}); | |
1123 | } | |
1124 | ||
1125 | return undef; | |
1126 | } | |
1127 | ||
1128 | ||
1129 | sub creat_text_input { | |
1130 | my ($default, $text) = @_; | |
1131 | ||
1132 | my $hbox = Gtk2::HBox->new (0, 0); | |
1133 | ||
1134 | my $label = Gtk2::Label->new ($text); | |
1135 | $label->set_size_request (150, -1); | |
1136 | $label->set_alignment (1, 0.5); | |
1137 | $hbox->pack_start ($label, 0, 0, 10); | |
1138 | my $e1 = Gtk2::Entry->new (); | |
1139 | $e1->set_width_chars (30); | |
1140 | $hbox->pack_start ($e1, 0, 0, 0); | |
1141 | $e1->set_text ($default); | |
1142 | ||
1143 | return ($hbox, $e1); | |
1144 | } | |
1145 | ||
1146 | sub creat_ip_input { | |
1147 | my ($init, $default, $text) = @_; | |
1148 | ||
1149 | my (@ips) = split /\./, $init; | |
1150 | my (@defs) = split /\./, $default; | |
1151 | ||
1152 | my $hbox = Gtk2::HBox->new (0, 0); | |
1153 | ||
1154 | my $label = Gtk2::Label->new ($text); | |
1155 | $label->set_size_request (150, -1); | |
1156 | $label->set_alignment (1, 0.5); | |
1157 | $hbox->pack_start ($label, 0, 0, 10); | |
1158 | ||
1159 | my $e1 = Gtk2::Entry->new_with_max_length (3); | |
1160 | $e1->{default} = $defs[0]; | |
1161 | $hbox->pack_start ($e1, 0, 0, 0); | |
1162 | $e1->set_width_chars (3); | |
1163 | $e1->set_text ($ips[0]); | |
1164 | $e1->signal_connect (key_press_event => \&check_num); | |
1165 | $e1->signal_connect (focus_out_event => \&check_range); | |
1166 | ||
1167 | my $l1 = Gtk2::Label->new ("."); | |
1168 | $hbox->pack_start ($l1, 0, 0, 2); | |
1169 | ||
1170 | my $e2 = Gtk2::Entry->new_with_max_length (3); | |
1171 | $e2->{default} = $defs[1]; | |
1172 | $hbox->pack_start ($e2, 0, 0, 0); | |
1173 | $e2->set_width_chars (3); | |
1174 | $e2->set_text ($ips[1]); | |
1175 | $e2->signal_connect (key_press_event => \&check_num); | |
1176 | $e2->signal_connect (focus_out_event => \&check_range); | |
1177 | ||
1178 | my $l2 = Gtk2::Label->new ("."); | |
1179 | $hbox->pack_start ($l2, 0, 0, 2); | |
1180 | ||
1181 | my $e3 = Gtk2::Entry->new_with_max_length (3); | |
1182 | $e3->{default} = $defs[2]; | |
1183 | $hbox->pack_start ($e3, 0, 0, 0); | |
1184 | $e3->set_width_chars (3); | |
1185 | $e3->set_text ($ips[2]); | |
1186 | $e3->signal_connect (key_press_event => \&check_num); | |
1187 | $e3->signal_connect (focus_out_event => \&check_range); | |
1188 | ||
1189 | my $l3 = Gtk2::Label->new ("."); | |
1190 | $hbox->pack_start ($l3, 0, 0, 2); | |
1191 | ||
1192 | my $e4 = Gtk2::Entry->new_with_max_length (3); | |
1193 | $e4->{default} = $defs[3]; | |
1194 | $hbox->pack_start ($e4, 0, 0, 0); | |
1195 | $e4->set_width_chars (3); | |
1196 | $e4->set_text ($ips[3]); | |
1197 | $e4->signal_connect (key_press_event => \&check_num); | |
1198 | $e4->signal_connect (focus_out_event => \&check_range); | |
1199 | ||
1200 | return ($hbox, $e1, $e2, $e3, $e4); | |
1201 | } | |
1202 | ||
1203 | sub get_ip_config { | |
1204 | ||
1205 | my $ifconfig = `ifconfig eth0`; | |
1206 | ||
1207 | my ($addr) = $ifconfig =~ m/inet addr:(\S*)/m; | |
1208 | my ($mask) = $ifconfig =~ m/Mask:(\S*)/m; | |
1209 | ||
1210 | my $route = `route -n`; | |
1211 | my ($gateway) = $route =~ m/^0\.0\.0\.0\s+(\d+\.\d+\.\d+\.\d+)\s+/m; | |
1212 | ||
1213 | my $resolvconf = `cat /etc/resolv.conf`; | |
1214 | my ($dnsserver) = $resolvconf =~ m/^nameserver\s+(\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})$/m; | |
1215 | ||
1216 | return { | |
1217 | addr => $addr, | |
1218 | mask => $mask, | |
1219 | gateway => $gateway, | |
1220 | dnsserver => $dnsserver, | |
1221 | } | |
1222 | } | |
1223 | ||
1224 | sub display_message { | |
1225 | my ($msg) = @_; | |
1226 | ||
1227 | my $dialog = Gtk2::MessageDialog->new ($window, 'modal', | |
1228 | 'info', 'ok', $msg); | |
1229 | $dialog->run(); | |
1230 | $dialog->destroy(); | |
1231 | } | |
1232 | ||
1233 | sub display_error { | |
1234 | my ($msg) = @_; | |
1235 | ||
1236 | my $dialog = Gtk2::MessageDialog->new ($window, 'modal', | |
1237 | 'error', 'ok', $msg); | |
1238 | $dialog->run(); | |
1239 | $dialog->destroy(); | |
1240 | } | |
1241 | ||
1242 | sub create_ipconf_view { | |
1243 | ||
1244 | cleanup_view (); | |
1245 | display_html ("ipconf.htm"); | |
1246 | ||
1247 | my $vbox = Gtk2::VBox->new (0, 0); | |
1248 | $inbox->pack_start ($vbox, 1, 0, 0); | |
1249 | my $hbox = Gtk2::HBox->new (0, 0); | |
1250 | $vbox->pack_start ($hbox, 0, 0, 30); | |
1251 | my $vbox2 = Gtk2::VBox->new (0, 0); | |
1252 | $hbox->add ($vbox2); | |
1253 | ||
1254 | my $addr = $ipconf->{addr} || '192.168.100.2'; | |
1255 | my $mask = $ipconf->{mask} || '255.255.255.0'; | |
1256 | ||
1257 | my ($hostbox, $hostentry) = | |
1258 | creat_text_input ('proxmox.domain.tld', 'Hostname (FQDN):'); | |
1259 | $vbox2->pack_start ($hostbox, 0, 0, 2); | |
1260 | ||
1261 | my $ipbox; | |
1262 | ($ipbox, $ip_1, $ip_2, $ip_3, $ip_4) = | |
1263 | creat_ip_input ($addr, '0.0.0.0', 'IP Address:'); | |
1264 | $vbox2->pack_start ($ipbox, 0, 0, 2); | |
1265 | ||
1266 | my $maskbox; | |
1267 | ($maskbox, $mask_1, $mask_2, $mask_3, $mask_4) = | |
1268 | creat_ip_input ($mask, '255.255.255.0', 'Netmask:'); | |
1269 | $vbox2->pack_start ($maskbox, 0, 0, 2); | |
1270 | ||
1271 | $gateway = $ipconf->{gateway} || '192.168.100.1'; | |
1272 | ||
1273 | my $gwbox; | |
1274 | ($gwbox, $gw_1, $gw_2, $gw_3, $gw_4) = | |
1275 | creat_ip_input ($gateway, '0.0.0.0', 'Gateway:'); | |
1276 | ||
1277 | $vbox2->pack_start ($gwbox, 0, 0, 15); | |
1278 | ||
1279 | $dnsserver = $ipconf->{dnsserver} || $gateway; | |
1280 | ||
1281 | my $dnsbox; | |
1282 | ($dnsbox, $dns_1, $dns_2, $dns_3, $dns_4) = | |
1283 | creat_ip_input ($dnsserver, '0.0.0.0', 'DNS Server:'); | |
1284 | ||
1285 | $vbox2->pack_start ($dnsbox, 0, 0, 0); | |
1286 | ||
1287 | $inbox->show_all; | |
1288 | set_next (undef, sub { | |
1289 | my $text = $hostentry->get_text(); | |
1290 | ||
1291 | $text =~ s/^\s+//; | |
1292 | $text =~ s/\s+$//; | |
1293 | ||
1294 | if ($text && $text =~ m/^[\w\-]+(\.[\w\-]+)+$/ && $text !~ m/.domain.tld$/ && | |
1295 | $text =~ m/^([^\.]+)\.(\S+)$/) { | |
1296 | $hostname = $1; | |
1297 | $domain = $2; | |
1298 | create_extract_view (); | |
1299 | return; | |
1300 | } | |
1301 | display_message ("Hostname does not look like a fully qualified domain name."); | |
1302 | $hostentry->grab_focus(); | |
1303 | }); | |
1304 | ||
1305 | $hostentry->grab_focus(); | |
1306 | } | |
1307 | ||
1308 | sub get_device_desc { | |
1309 | my ($devname, $size, $model) = @_; | |
1310 | ||
1311 | if ($size && ($size > 0) && $model) { | |
1312 | $size = int($size/2048); # size in MB | |
1313 | ||
1314 | if ($size >= 1024) { | |
1315 | $size = int($size/1024); # size in GB | |
1316 | return "$devname (${size}GB, $model)"; | |
1317 | } else { | |
1318 | return "$devname (${size}MB, $model)"; | |
1319 | } | |
1320 | ||
1321 | } else { | |
1322 | return $devname; | |
1323 | } | |
1324 | } | |
1325 | ||
1326 | sub update_layout { | |
1327 | my ($cb, $kmap) = @_; | |
1328 | ||
1329 | my $ind; | |
1330 | my $def; | |
1331 | my $i = 0; | |
1332 | my $kmaphash = $cmap->{kmaphash}; | |
1333 | foreach my $layout (sort keys %$kmaphash) { | |
1334 | $def = $i if $kmaphash->{$layout} eq 'en-us'; | |
1335 | $ind = $i if $kmap && $kmaphash->{$layout} eq $kmap; | |
1336 | $i++; | |
1337 | } | |
1338 | ||
1339 | $cb->set_active ($ind || $def || 0); | |
1340 | } | |
1341 | ||
1342 | my $lastzonecb; | |
1343 | sub update_zonelist { | |
1344 | my ($box, $cc) = @_; | |
1345 | ||
1346 | my $cczones = $cmap->{cczones}; | |
1347 | my $zones = $cmap->{zones}; | |
1348 | ||
1349 | my $sel; | |
1350 | if ($lastzonecb) { | |
1351 | $sel = $lastzonecb->get_active_text(); | |
1352 | $box->remove ($lastzonecb); | |
1353 | } else { | |
1354 | $sel = $timezone; # used once to select default | |
1355 | } | |
1356 | ||
1357 | my $cb = $lastzonecb = Gtk2::ComboBox->new_text (); | |
1358 | $cb->set_size_request (200, -1); | |
1359 | ||
1360 | $cb->signal_connect ('changed' => sub { | |
1361 | $timezone = $cb->get_active_text(); | |
1362 | }); | |
1363 | ||
1364 | my @za; | |
1365 | if ($cc && defined ($cczones->{$cc})) { | |
1366 | @za = keys %{$cczones->{$cc}}; | |
1367 | } else { | |
1368 | @za = keys %$zones; | |
1369 | } | |
1370 | my $ind; | |
1371 | my $i = 0; | |
1372 | foreach my $zone (sort @za) { | |
1373 | $ind = $i if $sel && $zone eq $sel; | |
1374 | $cb->append_text ($zone); | |
1375 | $i++; | |
1376 | } | |
1377 | ||
1378 | $cb->set_active ($ind || 0); | |
1379 | ||
1380 | $cb->show; | |
1381 | $box->pack_start ($cb, 0, 0, 0); | |
1382 | } | |
1383 | ||
1384 | sub create_password_view { | |
1385 | ||
1386 | cleanup_view (); | |
1387 | ||
1388 | my $vbox2 = Gtk2::VBox->new (0, 0); | |
1389 | $inbox->pack_start ($vbox2, 1, 0, 0); | |
1390 | my $vbox = Gtk2::VBox->new (0, 0); | |
1391 | $vbox2->pack_start ($vbox, 0, 0, 30); | |
1392 | ||
1393 | my $hbox1 = Gtk2::HBox->new (0, 0); | |
1394 | my $label = Gtk2::Label->new ("Password"); | |
1395 | $label->set_size_request (150, -1); | |
1396 | $label->set_alignment (1, 0.5); | |
1397 | $hbox1->pack_start ($label, 0, 0, 10); | |
1398 | my $pwe1 = Gtk2::Entry->new (); | |
1399 | $pwe1->set_visibility (0); | |
1400 | $pwe1->set_size_request (200, -1); | |
1401 | $hbox1->pack_start ($pwe1, 0, 0, 0); | |
1402 | ||
1403 | my $hbox2 = Gtk2::HBox->new (0, 0); | |
1404 | $label = Gtk2::Label->new ("Confirm"); | |
1405 | $label->set_size_request (150, -1); | |
1406 | $label->set_alignment (1, 0.5); | |
1407 | $hbox2->pack_start ($label, 0, 0, 10); | |
1408 | my $pwe2 = Gtk2::Entry->new (); | |
1409 | $pwe2->set_visibility (0); | |
1410 | $pwe2->set_size_request (200, -1); | |
1411 | $hbox2->pack_start ($pwe2, 0, 0, 0); | |
1412 | ||
1413 | my $hbox3 = Gtk2::HBox->new (0, 0); | |
1414 | $label = Gtk2::Label->new ("E-Mail"); | |
1415 | $label->set_size_request (150, -1); | |
1416 | $label->set_alignment (1, 0.5); | |
1417 | $hbox3->pack_start ($label, 0, 0, 10); | |
1418 | my $eme = Gtk2::Entry->new (); | |
1419 | $eme->set_size_request (200, -1); | |
1420 | $hbox3->pack_start ($eme, 0, 0, 0); | |
1421 | ||
1422 | ||
1423 | $vbox->pack_start ($hbox1, 0, 0, 5); | |
1424 | $vbox->pack_start ($hbox2, 0, 0, 5); | |
1425 | $vbox->pack_start ($hbox3, 0, 0, 15); | |
1426 | ||
1427 | $inbox->show_all; | |
1428 | ||
1429 | display_html ("passwd.htm"); | |
1430 | ||
1431 | set_next (undef, sub { | |
1432 | ||
1433 | my $t1 = $pwe1->get_text; | |
1434 | my $t2 = $pwe2->get_text; | |
1435 | ||
1436 | if (length ($t1) < 5) { | |
1437 | display_message ("Password is too short."); | |
1438 | $pwe1->grab_focus(); | |
1439 | return; | |
1440 | } | |
1441 | ||
1442 | if ($t1 ne $t2) { | |
1443 | display_message ("Password does not match."); | |
1444 | $pwe1->grab_focus(); | |
1445 | return; | |
1446 | } | |
1447 | ||
1448 | my $t3 = $eme->get_text; | |
1449 | if ($t3 !~ m/^\S+\@\S+\.\S+$/) { | |
1450 | display_message ("E-Mail does not look like a vaild address" . | |
1451 | " (user\@domain.tld)"); | |
1452 | $eme->grab_focus(); | |
1453 | return; | |
1454 | ||
1455 | } | |
1456 | ||
1457 | $password = $t1; | |
1458 | $mailto = $t3; | |
1459 | ||
1460 | create_ipconf_view(); | |
1461 | }); | |
1462 | ||
1463 | $pwe1->grab_focus(); | |
1464 | ||
1465 | } | |
1466 | ||
1467 | sub create_country_view { | |
1468 | ||
1469 | cleanup_view (); | |
1470 | ||
1471 | my $countryhash = $cmap->{countryhash}; | |
1472 | my $ctr = $cmap->{country}; | |
1473 | ||
1474 | my $vbox2 = Gtk2::VBox->new (0, 0); | |
1475 | $inbox->pack_start ($vbox2, 1, 0, 0); | |
1476 | my $vbox = Gtk2::VBox->new (0, 0); | |
1477 | $vbox2->pack_start ($vbox, 0, 0, 30); | |
1478 | ||
1479 | my $w = Gtk2::Entry->new (); | |
1480 | $w->set_size_request (200, -1); | |
1481 | ||
1482 | my $c = Gtk2::EntryCompletion->new (); | |
1483 | $c->set_text_column (0); | |
1484 | $c->set_minimum_key_length(0); | |
1485 | $c->set_popup_set_width (1); | |
1486 | ||
1487 | my $hbox2 = Gtk2::HBox->new (0, 0); | |
1488 | my $label = Gtk2::Label->new ("Time zone"); | |
1489 | $label->set_size_request (150, -1); | |
1490 | $label->set_alignment (1, 0.5); | |
1491 | $hbox2->pack_start ($label, 0, 0, 10); | |
1492 | update_zonelist ($hbox2); | |
1493 | ||
1494 | my $hbox3 = Gtk2::HBox->new (0, 0); | |
1495 | $label = Gtk2::Label->new ("Keyboard Layout"); | |
1496 | $label->set_size_request (150, -1); | |
1497 | $label->set_alignment (1, 0.5); | |
1498 | $hbox3->pack_start ($label, 0, 0, 10); | |
1499 | ||
1500 | my $kmapcb = Gtk2::ComboBox->new_text (); | |
1501 | $kmapcb->set_size_request (200, -1); | |
1502 | foreach my $layout (sort keys %{$cmap->{kmaphash}}) { | |
1503 | $kmapcb->append_text ($layout); | |
1504 | } | |
1505 | ||
1506 | update_layout ($kmapcb); | |
1507 | $hbox3->pack_start ($kmapcb, 0, 0, 0); | |
1508 | ||
1509 | $kmapcb->signal_connect ('changed' => sub { | |
1510 | my $sel = $kmapcb->get_active_text(); | |
1511 | if (my $kmap = $cmap->{kmaphash}->{$sel}) { | |
1512 | my $xkmap = $cmap->{kmap}->{$kmap}->{x11}; | |
1513 | my $xvar = $cmap->{kmap}->{$kmap}->{x11var}; | |
1514 | syscmd ("setxkbmap $xkmap $xvar") if !$opt_testmode; | |
1515 | $keymap = $kmap; | |
1516 | } | |
1517 | }); | |
1518 | ||
1519 | $w->signal_connect ('changed' => sub { | |
1520 | my ($entry, $event) = @_; | |
1521 | my $text = $entry->get_text; | |
1522 | ||
1523 | if (my $cc = $countryhash->{lc($text)}) { | |
1524 | update_zonelist ($hbox2, $cc); | |
1525 | my $kmap = $ctr->{$cc}->{kmap} || 'en-us'; | |
1526 | update_layout ($kmapcb, $kmap); | |
1527 | } | |
1528 | }); | |
1529 | ||
1530 | $w->signal_connect (key_press_event => sub { | |
1531 | my ($entry, $event) = @_; | |
1532 | my $text = $entry->get_text; | |
1533 | ||
1534 | my $val = $event->keyval; | |
1535 | if ($val == $Gtk2::Gdk::Keysyms{Tab}) { | |
1536 | my $cc = $countryhash->{lc($text)}; | |
1537 | return undef if $cc; | |
1538 | my $found = 0; | |
1539 | my $compl; | |
1540 | foreach my $cc (keys %$ctr) { | |
1541 | my $ct = $ctr->{$cc}->{name}; | |
1542 | if ($ct =~ m/^\Q$text\E.*$/i) { | |
1543 | $found++; | |
1544 | $compl = $ct; | |
1545 | } | |
1546 | last if $found > 1; | |
1547 | } | |
1548 | if ($found == 1) { | |
1549 | $entry->set_text ($compl); | |
1550 | return undef; | |
1551 | } else { | |
1552 | Gtk2::Gdk->beep(); | |
1553 | } | |
1554 | ||
1555 | $w->insert_text('', -1); # popup selection | |
1556 | return 1; | |
1557 | } | |
1558 | ||
1559 | return undef; | |
1560 | }); | |
1561 | ||
1562 | my $ls = Gtk2::ListStore->new('Glib::String'); | |
1563 | foreach my $cc (sort {$ctr->{$a}->{name} cmp $ctr->{$b}->{name} } keys %$ctr) { | |
1564 | my $iter = $ls->append(); | |
1565 | $ls->set ($iter, 0, $ctr->{$cc}->{name}); | |
1566 | } | |
1567 | $c->set_model ($ls); | |
1568 | ||
1569 | $w->set_completion ($c); | |
1570 | ||
1571 | my $hbox = Gtk2::HBox->new (0, 0); | |
1572 | ||
1573 | $label = Gtk2::Label->new ("Country"); | |
1574 | $label->set_alignment (1, 0.5); | |
1575 | $label->set_size_request (150, -1); | |
1576 | $hbox->pack_start ($label, 0, 0, 10); | |
1577 | $hbox->pack_start ($w, 0, 0, 0); | |
1578 | ||
1579 | $vbox->pack_start ($hbox, 0, 0, 5); | |
1580 | $vbox->pack_start ($hbox2, 0, 0, 5); | |
1581 | $vbox->pack_start ($hbox3, 0, 0, 5); | |
1582 | ||
1583 | if ($country) { | |
1584 | $w->set_text ($ctr->{$country}->{name}); | |
1585 | } | |
1586 | ||
1587 | $inbox->show_all; | |
1588 | ||
1589 | display_html ("country.htm"); | |
1590 | set_next (undef, sub { | |
1591 | ||
1592 | my $text = $w->get_text; | |
1593 | ||
1594 | if (my $cc = $countryhash->{lc($text)}) { | |
1595 | $country = $cc; | |
1596 | create_password_view(); | |
1597 | return; | |
1598 | } else { | |
1599 | display_message ("Please select a country first."); | |
1600 | $w->grab_focus(); | |
1601 | } | |
1602 | }); | |
1603 | ||
1604 | $w->grab_focus(); | |
1605 | } | |
1606 | ||
1607 | sub create_hdsel_view { | |
1608 | ||
1609 | cleanup_view (); | |
1610 | ||
1611 | my $vbox = Gtk2::VBox->new (0, 0); | |
1612 | $inbox->pack_start ($vbox, 1, 0, 0); | |
1613 | my $hbox = Gtk2::HBox->new (0, 0); | |
1614 | $vbox->pack_start ($hbox, 0, 0, 30); | |
1615 | ||
1616 | my ($disk, $devname, $size, $model) = @{@$hds[0]}; | |
1617 | $target_hd = $devname; | |
1618 | $master_hd = find_master ($target_hd); | |
1619 | my $label; | |
1620 | ||
1621 | if (scalar (@$hds) == 1) { | |
1622 | my $devdesc = get_device_desc ($devname, $size, $model); | |
1623 | $label = Gtk2::Label->new ("Target Harddisk: $devdesc"); | |
1624 | $hbox->pack_start ($label, 0, 0, 0); | |
1625 | } else { | |
1626 | $label = Gtk2::Label->new ("Target Harddisks: "); | |
1627 | $hbox->pack_start ($label, 0, 0, 0); | |
1628 | ||
1629 | my $combo = Gtk2::ComboBox->new_text (); | |
1630 | ||
1631 | foreach my $hd (@$hds) { | |
1632 | ($disk, $devname, $size, $model) = @$hd; | |
1633 | $combo->append_text (get_device_desc ($devname, $size, $model)); | |
1634 | } | |
1635 | ||
1636 | $combo->set_active (0); | |
1637 | $combo->signal_connect (changed => sub { | |
1638 | $a = shift->get_active; | |
1639 | my ($disk, $devname) = @{@$hds[$a]}; | |
1640 | $target_hd = $devname; | |
1641 | $master_hd = find_master ($target_hd); | |
1642 | }); | |
1643 | ||
1644 | $hbox->pack_start ($combo, 0, 0, 0); | |
1645 | } | |
1646 | ||
1647 | $inbox->show_all; | |
1648 | ||
1649 | display_html ("page1.htm"); | |
1650 | set_next (undef, \&create_country_view); | |
1651 | } | |
1652 | ||
1653 | sub create_extract_view { | |
1654 | ||
1655 | $ipaddress = $ip_1->get_text . "." . $ip_2->get_text . "." . | |
1656 | $ip_3->get_text . "." . $ip_4->get_text; | |
1657 | ||
1658 | $netmask = $mask_1->get_text . "." . $mask_2->get_text . "." . | |
1659 | $mask_3->get_text . "." . $mask_4->get_text; | |
1660 | ||
1661 | $gateway = $gw_1->get_text . "." . $gw_2->get_text . "." . | |
1662 | $gw_3->get_text . "." . $gw_4->get_text; | |
1663 | ||
1664 | $dnsserver = $dns_1->get_text . "." . $dns_2->get_text . "." . | |
1665 | $dns_3->get_text . "." . $dns_4->get_text; | |
1666 | ||
1667 | # print "TEST $ipaddress $netmask $gateway $dnsserver\n"; | |
1668 | cleanup_view (); | |
1669 | ||
1670 | display_html ("extract1-license.htm"); | |
1671 | $next->set_sensitive (0); | |
1672 | ||
1673 | my $vbox = Gtk2::VBox->new (0, 0); | |
1674 | $inbox->pack_start ($vbox, 1, 0, 0); | |
1675 | my $hbox = Gtk2::HBox->new (0, 0); | |
1676 | $vbox->pack_start ($hbox, 0, 0, 30); | |
1677 | ||
1678 | my $vbox2 = Gtk2::VBox->new (0, 0); | |
1679 | $hbox->pack_start ($vbox2, 0, 0, 0); | |
1680 | ||
1681 | $progress_status = Gtk2::Label->new (); | |
1682 | $vbox2->pack_start ($progress_status, 1, 1, 0); | |
1683 | ||
1684 | $progress = Gtk2::ProgressBar->new; | |
1685 | $progress->set_size_request (400, -1); | |
1686 | ||
1687 | $vbox2->pack_start ($progress, 0, 0, 0); | |
1688 | ||
1689 | $inbox->show_all; | |
1690 | ||
1691 | my $tdir = $opt_testmode ? "target" : "/target"; | |
1692 | mkdir $tdir; | |
1693 | my $base = $opt_testmode ? "/pve/$release/install/pve-base.tar" : "/proxmox/pve-base.tar"; | |
1694 | ||
1695 | eval { extract_data ($base, $tdir); }; | |
1696 | my $err = $@; | |
1697 | ||
1698 | $next->set_sensitive (1); | |
1699 | ||
1700 | set_next ("_Reboot", sub { exit (0); } ); | |
1701 | ||
1702 | display_html ($err ? "fail.htm" : "success.htm"); | |
1703 | ||
1704 | display_error ($err) if $err; | |
1705 | } | |
1706 | ||
1707 | sub mupdate_progress { | |
1708 | my $per = shift; | |
1709 | print "GOT1: $per\n"; | |
1710 | ||
1711 | } | |
1712 | ||
1713 | sub create_intro_view { | |
1714 | ||
1715 | cleanup_view (); | |
1716 | ||
1717 | display_html ("license.htm"); | |
1718 | ||
1719 | set_next ("I a_gree", \&create_hdsel_view); | |
1720 | } | |
1721 | ||
1722 | $ipconf = get_ip_config (); | |
1723 | ||
1724 | $country = detect_country () if $ipconf->{addr}; | |
1725 | ||
1726 | # read country, kmap and timezone infos | |
1727 | $cmap = read_cmap (); | |
1728 | ||
1729 | create_main_window (); | |
1730 | ||
1731 | if (!defined ($hds) || (scalar (@$hds) <= 0)) { | |
1732 | print "no hardisks found\n"; | |
1733 | display_html ("nohds.htm"); | |
1734 | set_next ("Reboot", sub { exit (0); } ); | |
1735 | } else { | |
1736 | ||
1737 | foreach my $hd (@$hds) { | |
1738 | my ($disk, $devname) = @$hd; | |
1739 | next if $devname =~ m|^/dev/md\d+$|; | |
1740 | print "found Disk$disk N:$devname\n"; | |
1741 | } | |
1742 | ||
1743 | create_intro_view (); | |
1744 | } | |
1745 | ||
1746 | Gtk2->main; | |
1747 | ||
1748 | exit 0; |