use strict;
use warnings;
-use PVE::pvecfg;
-use PVE::Tools;
-
-my $report;
-
-my @general = ('hostname', 'pveversion --verbose', 'cat /etc/hosts', 'top -b -n 1 | head -n 15',
- 'pvesubscription get', 'lscpu');
-
-my @storage = ('cat /etc/pve/storage.cfg', 'pvesm status', 'cat /etc/fstab', 'mount', 'df --human');
-
-my @volumes = ('lvdisplay', 'vgdisplay', 'zpool status', 'zfs list');
-my @machines = ('qm list', sub { dir2text('/etc/pve/qemu-server/', '\d.*conf') });
-
-my @net = ('ifconfig', 'cat /etc/network/interfaces', sub { dir2text('/etc/pve/firewall/', '.*fw') },
- 'iptables-save');
-
-my @cluster = ('pvecm nodes', 'pvecm status');
+use PVE::Tools;
-my @bios = ('dmidecode -t bios');
+# output the content of all the files of a directory
+my sub dir2text {
+ my ($target_dir, $regexp) = @_;
-if (PVE::pvecfg::version() >= 4.0) {
- push @cluster, 'cat /etc/pve/corosync.conf 2> /dev/null' ;
- push @machines, sub { dir2text('/etc/pve/lxc/', '\d.*conf') };
-} else {
- push @general, 'grep --max-count=1 "model name" /proc/cpuinfo';
- push @machines, sub { dir2text('/etc/pve/openvz/', '\d.*conf') };
- push @cluster, 'clustat', 'cat /etc/cluster.conf 2> /dev/null';
+ print STDERR "dir2text '${target_dir}${regexp}'...";
+ my $text = "# output '${target_dir}${regexp}' file(s)\n";
+ PVE::Tools::dir_glob_foreach($target_dir, $regexp, sub {
+ my ($file) = @_;
+ return if $file eq '.' || $file eq '..';
+ $text .= "\n# cat $target_dir$file\n";
+ $text .= PVE::Tools::file_get_contents($target_dir.$file)."\n";
+ });
+ return $text;
}
-my $general_report = {
- title => 'general system info',
- commands => \@general,
-};
-
-my $storage_report = {
- title => 'info about storage (lvm and zfs)',
- commands => \@storage,
-};
+# command -v is the posix equivalent of 'which'
+my sub cmd_exists { system("command -v '$_[0]' > /dev/null 2>&1") == 0 }
+
+my $init_report_cmds = sub {
+ my $report_def = {
+ general => {
+ title => 'general system info',
+ order => 10,
+ cmds => [
+ 'hostname',
+ 'date -R',
+ 'pveversion --verbose',
+ 'cat /etc/hosts',
+ 'pvesubscription get',
+ 'cat /etc/apt/sources.list',
+ sub { dir2text('/etc/apt/sources.list.d/', '.*list') },
+ sub { dir2text('/etc/apt/sources.list.d/', '.*sources') },
+ 'lscpu',
+ 'pvesh get /cluster/resources --type node --output-format=yaml',
+ ],
+ },
+ 'system-load' => {
+ title => 'overall system load info',
+ order => 20,
+ cmds => [
+ 'top -b -c -w512 -n 1 -o TIME | head -n 30',
+ 'head /proc/pressure/*',
+ ],
+ },
+ storage => {
+ order => 30,
+ cmds => [
+ 'cat /etc/pve/storage.cfg',
+ 'pvesm status',
+ 'cat /etc/fstab',
+ 'findmnt --ascii',
+ 'df --human -T',
+ 'proxmox-boot-tool status',
+ ],
+ },
+ 'virtual guests' => {
+ order => 40,
+ cmds => [
+ 'qm list',
+ sub { dir2text('/etc/pve/qemu-server/', '\d.*conf') },
+ 'pct list',
+ sub { dir2text('/etc/pve/lxc/', '\d.*conf') },
+ ],
+ },
+ network => {
+ order => 45,
+ cmds => [
+ 'ip -details -statistics address',
+ 'ip -details -4 route show',
+ 'ip -details -6 route show',
+ 'cat /etc/network/interfaces',
+ sub { dir2text('/etc/network/interfaces.d/', '.*') },
+ sub { dir2text('/etc/pve/sdn/', '.*') },
+ ],
+ },
+ firewall => {
+ order => 50,
+ cmds => [
+ sub { dir2text('/etc/pve/firewall/', '.*fw') },
+ 'cat /etc/pve/local/host.fw',
+ 'iptables-save -c | column -t -l4 -o" "',
+ ],
+ },
+ cluster => {
+ order => 60,
+ cmds => [
+ 'pvecm nodes',
+ 'pvecm status',
+ 'cat /etc/pve/corosync.conf 2>/dev/null',
+ 'ha-manager status',
+ 'cat /etc/pve/datacenter.cfg',
+ ],
+ },
+ hardware => {
+ order => 70,
+ cmds => [
+ 'dmidecode -t bios',
+ 'lspci -nnk',
+ ],
+ },
+ 'block devices' => {
+ order => 80,
+ cmds => [
+ 'lsblk --ascii -M -o +HOTPLUG,ROTA,PHY-SEC,FSTYPE,MODEL,TRAN',
+ 'ls -l /dev/disk/by-*/',
+ 'iscsiadm -m node',
+ 'iscsiadm -m session',
+ ],
+ },
+ volumes => {
+ order => 90,
+ cmds => [
+ 'pvs',
+ 'lvs',
+ 'vgs',
+ ],
+ },
+ };
+
+ if (cmd_exists('zfs')) {
+ push @{$report_def->{volumes}->{cmds}},
+ 'zpool status',
+ 'zpool list -v',
+ 'zfs list',
+ 'arcstat',
+ ;
+ }
-my $volume_report = {
- title => 'info about virtual machines',
- commands => \@machines,
-};
+ if (-e '/etc/ceph/ceph.conf') {
+ push @{$report_def->{volumes}->{cmds}},
+ 'pveceph status',
+ 'ceph osd status',
+ 'ceph df',
+ 'ceph osd df tree',
+ 'ceph device ls',
+ 'cat /etc/ceph/ceph.conf',
+ 'ceph config dump',
+ 'pveceph pool ls',
+ 'ceph versions',
+ 'ceph health detail',
+ ;
+ }
-my $net_report = {
- title => 'info about network and firewall',
- commands => \@net,
-};
+ if (cmd_exists('multipath')) {
+ push @{$report_def->{disks}->{cmds}},
+ 'cat /etc/multipath.conf',
+ 'cat /etc/multipath/wwids',
+ 'multipath -ll',
+ ;
+ }
-my $cluster_report = {
- title => 'info about clustering',
- commands => \@cluster,
+ return $report_def;
};
-my $bios_report = {
- title => 'info about bios',
- commands => \@bios,
-};
+sub generate {
+ my $def = $init_report_cmds->();
-my @global_report = ($general_report, $storage_report, $volume_report, $net_report, $cluster_report, $bios_report);
+ my $report = '';
+ my $record_output = sub {
+ $report .= shift . "\n";
+ };
-# output the content of all the files of a directory
-sub dir2text {
- my ($target_dir, $regexp) = @_;
+ local $ENV{'PATH'} = '/sbin:/bin:/usr/sbin:/usr/bin';
+ my $cmd_timeout = 10; # generous timeout
- PVE::Tools::dir_glob_foreach($target_dir, $regexp, sub {
- my ($file) = @_;
- $report .= "# cat $target_dir$file\n";
- $report .= PVE::Tools::file_get_contents($target_dir.$file)."\n";
- });
-}
+ my $run_cmd_params = {
+ outfunc => $record_output,
+ errfunc => $record_output,
+ timeout => $cmd_timeout,
+ noerr => 1, # avoid checking programs exit code
+ };
-# execute commands and display their output as if they've been done on a interactive shell
-# so the local sysadmin can reproduce what we're doing
-sub do_execute {
- my ($command) = @_;
- $report .= "# $command \n";
- open (COMMAND, "$command 2>&1 |");
- while (<COMMAND>) {
- $report .= $_;
- }
-}
+ my $sorter = sub { ($def->{$_[0]}->{order} // 1<<30) <=> ($def->{$_[1]}->{order} // 1<<30) };
-sub generate {
- foreach my $subreport (@global_report) {
- my $title = $subreport->{'title'};
- my @commands = @{$subreport->{'commands'}};
+ for my $section ( sort { $sorter->($a, $b) } keys %$def) {
+ my $s = $def->{$section};
+ my $title = $s->{title} // "info about $section";
$report .= "\n==== $title ====\n";
- foreach my $command (@commands) {
- if (ref $command eq 'CODE') {
- &$command;
- } else {
- do_execute($command);
- next;
- }
+ for my $command (@{$s->{cmds}}) {
+ eval {
+ if (ref $command eq 'CODE') {
+ $report .= PVE::Tools::run_with_timeout($cmd_timeout, $command);
+ } else {
+ print STDERR "Process ".$command."...";
+ $report .= "\n# $command\n";
+ PVE::Tools::run_command($command, %$run_cmd_params);
+ }
+ print STDERR "OK";
+ };
+ print STDERR "\n";
+ $report .= "\nERROR: $@\n" if $@;
}
}
-return $report;
+
+ return $report;
}
+
+1;