# # Authentic Theme (https://github.com/authentic-theme/authentic-theme) # Copyright Ilia Rostovtsev # Licensed under MIT (https://github.com/authentic-theme/authentic-theme/blob/master/LICENSE) # use strict; use lib ("$ENV{'THEME_ROOT'}/lib"); use File::Grep qw( fgrep fmap fdo ); use Encode qw( encode decode ); use Time::Local; BEGIN {push(@INC, "..");} use WebminCore; our ( %in, %text, %config, %gconfig, %tconfig, %gaccess, $current_lang, $title, $base_remote_user, $remote_user, $theme_root_directory, $current_theme, $root_directory, $config_directory, $var_directory, %theme_text, %module_text_full, %theme_config, $get_user_level, $theme_webprefix, $http_x_url, $has_virtualmin, $has_cloudmin, $has_usermin, $has_usermin_version, $has_usermin_root_dir, $has_usermin_conf_dir); init_type(); init_config(); do("$ENV{'THEME_ROOT'}/authentic-init.pl"); sub authentic { init(); header([$title]); content(); footer(); } sub get_sysinfo_warning { my ($info_ref) = @_; my $returned_data = ''; # Show notifications first if (ref($info_ref)) { @{$info_ref} = sort {($b->{'type'} eq 'warning') <=> ($a->{'type'} eq 'warning')} @{$info_ref}; $returned_data .= '
'; foreach my $info (@{$info_ref}) { if ($info->{'type'} eq 'warning') { $returned_data .= replace("ui_submit ui_form_end_submit", "btn-tiny ui_submit ui_form_end_submit", &ui_alert_box($info->{'warning'}, $info->{'level'} || 'warn', undef, 1, $info->{'desc'} || undef )); } } } return $returned_data; } sub get_extended_sysinfo { my ($info_ref, $x) = @_; my $returned_sysinfo = ''; if ($info_ref) { $returned_sysinfo .= '
'; foreach my $info (@{$info_ref}) { if ($info->{'id'} ne 'notifications' && $info->{'type'} ne 'link' && $info->{'module'} ne 'mailbox' && $info->{'module'} ne 'system-status' && $info->{'type'} ne 'warning' && $a->{'type'} ne 'warning' && $b->{'type'} ne 'warning') { our $charts_not_supported = 'no'; if ($info->{'type'} eq 'chart') { foreach my $t (@{ $info->{'chart'} }) { if ($t->{'chart'}[0] < 0 || $t->{'chart'}[1] < 0) { $charts_not_supported = 'yes'; } } } if ($theme_config{'settings_sysinfo_hidden_panels_user'} =~ /\'$info->{'id'}\'/) { next; } if ($info->{'id'} && $charts_not_supported eq 'no') { my $open = ($info->{'open'} || $info->{'id'} eq 'domain') ? ' in' : ($theme_config{'settings_sysinfo_expand_all_accordions'} eq 'true' ? ' in' : ''); my $formatted_title = ( $info->{'id'} . '-' . $info->{'module'} eq 'status_services-status' ? $theme_text{'theme_xhred_sysinfo_system_monitors'} : ($info->{'id'} . '-' . $info->{'module'} eq 'sysinfo-virtual-server' ? $theme_text{'theme_xhred_sysinfo_software_versions'} : ($info->{'id'} . '-' . $info->{'module'} eq 'status-virtual-server' ? $theme_text{'theme_xhred_sysinfo_server_status'} : ($info->{'id'} . '-' . $info->{'module'} eq 'quota-virtual-server' ? $theme_text{'theme_xhred_sysinfo_disk_quotas'} : ($info->{'id'} . '-' . $info->{'module'} eq 'bw-virtual-server' ? $theme_text{'theme_xhred_sysinfo_bandwidth_quotas'} : ($info->{'id'} . '-' . $info->{'module'} eq 'updates-virtual-server' ? $theme_text{'theme_xhred_sysinfo_vm_package_updates'} : ($info->{'id'} . '-' . $info->{'module'} eq 'acl_logins-acl' ? $theme_text{'theme_xhred_sysinfo_recent_logins'} : ($info->{'desc'}) ) ) ) ) ) )); $returned_sysinfo .= '
'; if ($info->{'id'} ne 'plugin_virtualmin-notes' && $info->{'id'} ne 'acl_logins') { $returned_sysinfo .= '
'; } if ($info->{'type'} ne 'html') { $returned_sysinfo .= ''; } if ($info->{'type'} eq 'table' && ( $info->{'module'} ne 'system-status' && $info->{'type'} ne 'link' )) { foreach my $t (@{ $info->{'table'} }) { my $__checkmark = ''; my $__stop = ''; my $__down = ''; my $__start = ''; my $__restart = ''; $t->{"value"} =~ s//$__checkmark/g; $t->{"value"} =~ s//$__stop/g; $t->{"value"} =~ s//$__down/g; $t->{"value"} =~ s//$__start/g; $t->{"value"} =~ s//$__restart/g; $returned_sysinfo .= ''; } } elsif ($info->{'type'} eq 'chart') { $returned_sysinfo .= print_charts($info, $x); } elsif ($info->{'type'} eq 'html') { $info->{'html'} =~ s/]*>.*?<\/script>//igs; $returned_sysinfo .= $info->{'html'}; } if ($info->{'id'} ne 'plugin_virtualmin-notes' && $info->{'id'} ne 'acl_logins') { my $cltbody; if ($info->{'type'} ne 'html') { $cltbody = "
' . replace('href=\'', "href='$theme_webprefix", $t->{"desc"}) . ' ' . replace('href=\'', "href='$theme_webprefix", $t->{"value"}) . '
"; } $returned_sysinfo .= "$cltbody
"; } $returned_sysinfo .= '
'; } } } if ($get_user_level eq '0' && $theme_config{'settings_sysinfo_real_time_status'} ne '0' && $theme_config{'settings_sysinfo_real_time_stored'} ne 'false' && (acl_system_status('cpu') || acl_system_status('mem') || acl_system_status('load'))) { my $data = '
' . $theme_text{'theme_xhred_datatable_sloadingrecords'} . '
'; $returned_sysinfo .= print_panel( 1, 'live_stats', "$theme_text{'theme_dashboard_accordion_live_stats'}", $data, 1, 'A', 'live_stats'); } $returned_sysinfo .= '




'; return $returned_sysinfo; } } sub print_charts { my ($info, $x) = @_; my $returned_sysinfo = ''; foreach my $t (@{ $info->{'chart'} }) { my $unlimited = 0; my $percent_width_1 = int($t->{'chart'}[1]); my $percent_width_2 = int($t->{'chart'}[2]); my $percent_width_sum = $percent_width_1 + $percent_width_2; my $is_2 = defined($t->{'chart'}[2]); my $percent_1 = ' ' . $percent_width_1 . '%'; my $percent_2 = ' ' . $percent_width_2 . '%'; my $dd = $theme_text{'right_out'}; $dd =~ s/\s| |\$1|\$2//g; if ($t->{"value"} !~ /\Q$dd/) { $percent_1 = ' ' . $theme_text{'right_unlimited'}; $percent_width_1 = '0'; $unlimited = 1; } my $color; if ($percent_width_sum <= 49) { $color = 'green'; } elsif ($percent_width_sum <= 90) { $color = 'yellow'; } else { $color = 'red'; } if ($unlimited || $percent_width_sum == 0) { $color = 'gray'; } my $bar; if ($is_2 && !$unlimited && $percent_width_2) { $bar = '{'id'} . '-' . $info->{'module'} . $x . '-collapse', '(') . ' class="bar ' . $color . '" style="width:' . $percent_width_1 . '%;">' . $percent_1 . ''; $bar .= '{'id'} . '-' . $info->{'module'} . $x . '-collapse', '(') . ' class="bar ' . $color . '" style="width:' . $percent_width_2 . '%;">' . $percent_2 . ''; } else { $bar = '{'id'} . '-' . $info->{'module'} . $x . '-collapse', '(') : undef ) . ' class="bar ' . $color . '" style="width:' . $percent_width_1 . '%;">' . $percent_1 . ''; } $returned_sysinfo .= ' ' . replace('edit_domain', 'summary_domain', replace('href=\'', "href='$theme_webprefix", $t->{"desc"})) . '
' . $bar . '
' . $t->{"value"} . ' '; } return $returned_sysinfo; } sub print_easypie_charts { my ($cpu_percent, $mem_percent, $virt_percent, $disk_percent) = @_; if (defined($cpu_percent) || defined($mem_percent) || defined($virt_percent) || defined($disk_percent)) { print '
' . "\n"; my $columns = '3'; # CPU usage defined($cpu_percent) && print_easypie_chart($columns, ( ($cpu_percent || $cpu_percent eq "0") ? $cpu_percent : 'NaN' ), $theme_text{'body_cp'}, 'sysinfo_cpu_percent'); # Memory allocation defined($mem_percent) && print_easypie_chart($columns, ( ($mem_percent || $mem_percent eq "0") ? $mem_percent : 'NaN' ), ( ($current_lang eq 'ru' || $current_lang eq 'ru.UTF-8') ? $theme_text{'body_real2'} : $theme_text{'body_real'} ), 'sysinfo_mem_percent'); defined($virt_percent) && print_easypie_chart($columns, (($virt_percent || $virt_percent eq "0") ? $virt_percent : 'NaN'), ( ($current_lang eq 'ru' || $current_lang eq 'ru.UTF-8') ? $theme_text{'body_virt2'} : $theme_text{'body_virt'} ), 'sysinfo_virt_percent'); # Disk usage defined($disk_percent) && print_easypie_chart($columns, ( ($disk_percent || $disk_percent eq "0") ? $disk_percent : 'NaN' ), ( ($current_lang eq 'ru' || $current_lang eq 'ru.UTF-8') ? $theme_text{'body_disk2'} : $theme_text{'body_disk'} ), 'sysinfo_disk_percent'); print '
' . "\n"; } } sub print_sysstats_panel_end { print ''; } sub print_sysstats_panel_start { my ($info_ref) = @_; my $recollect; if ($info_ref) { my @recollect = @{$info_ref}; @recollect = grep {$_->{'id'} =~ /recollect/} @recollect; if (@recollect) { $recollect = ''; } } my %virtualmin_config = foreign_config('virtual-server'); my %cloudmin_config = foreign_config('server-manager'); print '
' . "\n"; print '
' . "\n"; print '

' . $recollect . '' . ($get_user_level eq '3' ? $theme_text{'body_header1'} : $theme_text{'body_header0'}) . ( $cloudmin_config{'docs_link'} && foreign_available("server-manager") ? ' ' . $cloudmin_config{'docs_text'} . '' : undef ) . ' ' . ($virtualmin_config{'docs_link'} && foreign_available("virtual-server") ? ' ' . $virtualmin_config{'docs_text'} . '' : undef ) . '

' . "\n"; print '
'; print '
' . "\n"; } sub print_sysstats_table { my ($data, $quota, $prod) = @_; if ((defined($data) && scalar(@{$data})) || (defined($quota) && scalar(@{$quota}))) { print '' . "\n"; if (defined($data) && scalar(@{$data})) { foreach my $t (@{ @{$data}[0]->{'table'} }) { my $insert = ($t->{"desc"} =~ /\Q$prod/i); if ($insert && $get_user_level ne '3') { print_table_row($theme_text{'body_webmin'}, get_webmin_version(1)); } print_table_row($t->{"desc"}, $t->{"value"}); if ($insert) { print_table_row($theme_text{'theme_version'}, get_theme_user_link()); } } } if (defined($quota) && scalar(@{$quota})) { print_table_row(@{$quota}[0]->{'desc'}, @{$quota}[0]->{"chart"}->[0]->{'value'}); } print '
' . "\n"; } } sub print_easypie_chart { my ($columns, $percent, $label, $id) = @_; print '
' . "\n"; print '
' . $label . '
'; print '
' . "\n"; } sub theme_list_combined_system_info { return &list_combined_system_info({ 'qshow' => 1, 'max' => $theme_config{'settings_sysinfo_max_servers'} }); } sub show_sysinfo_section { my ($s) = @_; my %access = &get_module_acl(undef, 'system-status'); $access{'show'} ||= ""; if ($access{'show'} eq '*') { return 1; } else { return &indexof($s, split(/\s+/, $access{'show'})) >= 0; } } sub get_sysinfo_vars { my ($info_ref) = @_; my ($info, $info_arr, @info); # Ask for collected info if (@_) { @info = @{$info_ref}; } else { @info = theme_list_combined_system_info(); } @info = grep {$_->{'id'} eq 'sysinfo'} @info; # Define used vars my ($cpu_percent, $mem_percent, $virt_percent, $disk_percent, $host, $os, $webmin_version, $virtualmin_version, $cloudmin_version, $authentic_theme_version, $local_time, $kernel_arch, $cpu_type, $cpu_temperature, $hdd_temperature, $uptime, $running_proc, $load, $real_memory, $virtual_memory, $disk_space, $package_message, $csf_title, $csf_data, $csf_remote_version, $authentic_remote_version); if (@info) { $info_arr = @info[0]->{'raw'}; $info = @$info_arr[0]; } else { return; } if (!@$info_arr) { return; } # Require memory information my @m; if ($info->{'mem'}) { @m = @{ $info->{'mem'} }; } # Easypie charts numbers if (show_sysinfo_section('cpu') && $info->{'cpu'}) { my @c = @{ $info->{'cpu'} }; $cpu_percent = $c[0] + $c[1] + $c[3]; $cpu_percent = int($cpu_percent); } if (show_sysinfo_section('mem')) { if (@m && $m[0]) { $mem_percent = ($m[0] - $m[1]) / $m[0] * 100; $mem_percent = int($mem_percent); } if (@m && $m[2]) { $virt_percent = ($m[2] - $m[3]) / $m[2] * 100; $virt_percent = int($virt_percent); } } if (show_sysinfo_section('disk')) { if ($info->{'disk_total'}) { my ($total, $free) = ($info->{'disk_total'}, $info->{'disk_free'}); $disk_percent = ($total - $free) / $total * 100; $disk_percent = int($disk_percent); } } if (show_sysinfo_section('host')) { # Operation system my $ip = $info->{'ips'} ? $info->{'ips'}->[0]->[0] : &to_ipaddress(get_system_hostname()); $ip = " ($ip)" if ($ip); $host = &get_display_hostname() . $ip; if (&foreign_available("net")) { $host = '' . $host . ''; } # Operating System Info if ($gconfig{'os_version'} eq '*') { $os = $gconfig{'real_os_type'}; } else { $os = $gconfig{'real_os_type'} . ' ' . $gconfig{'real_os_version'}; } my $is_hidden_link = ($get_user_level ne '0' ? ' hidden-force ' : undef); #Webmin version $webmin_version = product_version_update(get_webmin_version(1), 'w') . ' '; # Virtualmin version if ($has_virtualmin) { my ($vs_license, $__virtual_server_version); $vs_license = check_pro_package('vm'); $__virtual_server_version = (defined(@$info_arr[2]) ? @$info_arr[2]->{'vm_version'} : undef); $__virtual_server_version =~ s/.gpl//igs; $__virtual_server_version =~ s/.pro//igs; $virtualmin_version = ( product_version_update($__virtual_server_version, 'v') . " " . ( $vs_license eq '0' ? '' : '' . ' Pro ' : '
' ) . '' )); } # Cloudmin version if ($has_cloudmin) { my ($cm_licensed, $cm_version, $cm_type, $cm_type_g, $cm_type_p, $cm_version_o); $cm_licensed = check_pro_package('cm'); if (defined(&server_manager::get_module_version_and_type)) { ($cm_version, $cm_type) = &server_manager::get_module_version_and_type(); } else { $cm_version_o = (defined(@$info_arr[3]) ? @$info_arr[3]->{'cm_version'} : (defined(@$info_arr[2]) ? @$info_arr[2]->{'cm_version'} : undef)); $cm_version_o =~ s/(\.[a-z]+)$//igs; } $cm_type_g = $cm_version ? $cm_type : ''; $cm_type_p = $cm_version ? $cm_type : 'Pro'; $cloudmin_version = ( product_version_update($cm_version || $cm_version_o, 'c') . " " . ( $cm_licensed eq '0' ? $cm_type_g : ' ' . $cm_type_p . ' ' : '
' ) . '' )); } # Fetch theme version if ($get_user_level eq '0') { # Theme version/update my $authentic_remote_data = theme_remote_version(1); my $authentic_installed_version = theme_version(); my $incompatible = theme_update_incompatible($authentic_remote_data); ($authentic_remote_version) = $authentic_remote_data =~ /^version=(.*)/gm; my $authentic_remote_version_local = $authentic_remote_version; if ($incompatible && $authentic_remote_version_local !~ /alpha|beta|RC/) { $authentic_remote_version = $authentic_installed_version; } if ( $theme_config{'settings_sysinfo_theme_updates'} eq 'true' && ( (!$incompatible || ($incompatible && $authentic_remote_version_local =~ /alpha|beta|RC/)) && ( ( ($authentic_remote_version_local !~ /alpha|beta|RC/ && $authentic_installed_version =~ /alpha|beta|RC/ ) && lc($authentic_remote_version_local) ge substr($authentic_installed_version, 0, 5) ) || lc($authentic_remote_version_local) gt lc($authentic_installed_version)) )) { my $authentic_remote_beta = $authentic_remote_version_local =~ /alpha|beta|RC/; my $authentic_remote_alpha_beta = $authentic_remote_version_local =~ /alpha|beta/; my $authentic_remote_version_tag = $authentic_remote_version_local; my @_remote_version_tag = split /-/, $authentic_remote_version_tag; $authentic_remote_version_tag = $_remote_version_tag[0]; $authentic_theme_version = '' . $theme_text{'theme_name'} . ' ' . $authentic_installed_version . '. ' . ($authentic_remote_beta ? $theme_text{'theme_git_patch_available'} : $theme_text{'theme_update_available'}) . ' ' . $authentic_remote_version_local . '   
' . ' ' . $theme_text{'theme_update'} . '' . ' ' . $theme_text{'theme_changelog'} . '' . ' ' . $theme_text{'theme_download'} . '' . ' ' . $theme_text{'theme_xhred_global_configuration'} . '' . '
'; } else { $authentic_theme_version = get_theme_user_link(); } } else { $authentic_theme_version = get_theme_user_link(); } # Load ConfigServer Security & Firewall lib if available ($csf_title, $csf_data, $csf_remote_version) = lib_csf_control('strings'); #System time my ($_time); $_time = time(); $local_time = localtime($_time); my $no_theme_date_time = $theme_config{'settings_theme_make_date'} eq 'false'; if (foreign_available("time")) { if ($no_theme_date_time) { $local_time = '' . $local_time . ''; } else { $local_time = '' . $local_time . ''; } } else { if ($no_theme_date_time) { $local_time = '' . $local_time . ''; } else { $local_time = '' . $local_time . ''; } } } if (show_sysinfo_section('cpu')) { # Kernel and arch if ($info->{'kernel'}) { $kernel_arch = &theme_text('body_kernelon', $info->{'kernel'}->{'os'}, $info->{'kernel'}->{'version'}, $info->{'kernel'}->{'arch'}); } # CPU Type and cores my @c; if ($info->{'load'}) { @c = @{ $info->{'load'} }; if (@c > 3) { $cpu_type = &theme_text('body_cputype', @c); } } } if (show_sysinfo_section('temp')) { # Temperatures if ($info->{'cputemps'}) { foreach my $t (@{ $info->{'cputemps'} }) { $cpu_temperature .= ' ' . $theme_text{'theme_global_core'} . ' ' . (int($t->{'core'}) + 1) . ': ' . ( get_module_config_data('system-status', 'collect_units') ? (int(($t->{'temp'} * 9.0 / 5) + 32) . "°F") : (int($t->{'temp'}) . '°C ') ) . ''; } } if ($info->{'drivetemps'}) { foreach my $t (@{ $info->{'drivetemps'} }) { my $short = $t->{'device'}; $short =~ s/^\/dev\///; my $emsg; if ($t->{'errors'}) { $emsg .= '  ' . &theme_text('body_driveerr', $t->{'errors'}) . ""; } elsif ($t->{'failed'}) { $emsg .= '  ' . &theme_text('body_drivefailed') . ''; } $hdd_temperature .= '' . $short . ': ' . ( get_module_config_data('system-status', 'collect_units') ? (int(($t->{'temp'} * 9.0 / 5) + 32) . "°F") : (int($t->{'temp'}) . '°C ') ) . $emsg . ''; } } } if (show_sysinfo_section('load')) { # System uptime if (foreign_check("proc") && foreign_available("proc")) { foreign_require("proc"); my @system_uptime = defined(&proc::get_system_uptime) ? proc::get_system_uptime() : (); if (@system_uptime) { my ($day, $hour, $minute) = @system_uptime; my $uptime_text; if ($day) { $uptime_text = &theme_text('body_updays', $day, $hour, $minute); } elsif ($minute && $hour) { $uptime_text = &theme_text('body_uphours', $hour, $minute); } elsif ($minute) { $uptime_text = &theme_text('body_upmins', $minute); } $uptime = '' . $uptime_text . ''; } # Running processes my @procs = proc::list_processes(); $running_proc = scalar(@procs); $running_proc = '' . $running_proc . ''; } # Load averages if ($info->{'load'}) { my @c = @{ $info->{'load'} }; if (@c) { $load = &theme_text('body_load', @c); } } } if (show_sysinfo_section('mem')) { # Memory if (@m) { # Real memory details if ($m[0] && $m[1]) { $real_memory = &theme_text($m[4] ? 'body_used_cached_total' : 'body_used', nice_size(($m[0]) * 1024, -1), nice_size(($m[0] - $m[1]) * 1024, -1), ($m[4] ? nice_size($m[4] * 1024, -1) : undef)); } # Virtual memory details if ($m[2] > 0) { $virtual_memory = &theme_text('body_used', nice_size(($m[2]) * 1024, -1), nice_size(($m[2] - $m[3]) * 1024, -1)); } if (get_text_ltr()) { $real_memory = reverse_string($real_memory, "/"); if ($virtual_memory) { $virtual_memory = reverse_string($virtual_memory, "/"); } } } } if (show_sysinfo_section('disk')) { # Local disk space if ($info->{'disk_total'} && $info->{'disk_total'}) { $disk_space = &theme_text('body_used_and_free', nice_size($info->{'disk_total'}, -1), nice_size($info->{'disk_free'}, -1), nice_size($info->{'disk_total'} - $info->{'disk_free'}, -1)); if ($disk_space && get_text_ltr()) { $disk_space = reverse_string($disk_space, "/"); } } } if (show_sysinfo_section('poss')) { # Package updates if (&foreign_available("package-updates") && $info->{'poss'}) { my $msg; my @poss = @{ $info->{'poss'} }; my @secs = grep {$_->{'security'}} @poss; my $poss = scalar(@poss); my $secs = scalar(@secs); if ($poss && $secs) { $msg = &theme_text( ($poss gt 1 && $secs gt 1 ? 'body_upsec' : $poss gt 1 && $secs eq 1 ? 'body_upsec1' : $poss eq 1 && $secs gt 1 ? 'body_upsec2' : 'body_upsec3' ), $poss, $secs); } elsif ($poss) { $msg = &theme_text(($poss gt 1 ? 'body_upneed' : 'body_upneed1'), $poss); } else { $msg = $theme_text{'body_upok'}; } $msg =~ s/([0-9]+)/" $1 <\/i>"/eg; $package_message = '' . $msg . ''; } } return ($cpu_percent, $mem_percent, $virt_percent, $disk_percent, $host, $os, $webmin_version, $virtualmin_version, $cloudmin_version, $authentic_theme_version, $local_time, $kernel_arch, $cpu_type, $cpu_temperature, $hdd_temperature, $uptime, $running_proc, $load, $real_memory, $virtual_memory, $disk_space, $package_message, $csf_title, $csf_data, $csf_remote_version, $authentic_remote_version); } sub get_current_user_config { our ($___user) = grep {$_->{'name'} eq $base_remote_user} &acl::list_users(); return $___user; } sub get_col_num { my ($info, $max_col) = @_; my $num_col = 0; if ($info->{'cpu'}) {$num_col++;} if ($info->{'mem'}) { my @m = @{ $info->{'mem'} }; if (@m && $m[0]) {$num_col++;} if (@m && $m[2]) {$num_col++;} } if ($info->{'disk_total'}) {$num_col++;} my $col = $max_col / $num_col; return $col; } sub print_table_row { my ($title, $content, $id) = @_; print '' . "\n"; print '' . $title . '' . "\n"; print '' . $content . '' . "\n"; print '' . "\n"; } sub print_table_row_responsive { my ($title, $content, $id, $title2, $content2, $id2) = @_; print '' . "\n"; print '' . $title . '' . "\n"; print '' . $content . '' . "\n"; if ($title2) { print '' . $title2 . '' . "\n"; print '' . $content2 . '' . "\n"; print '' . "\n"; } } sub print_favorites { # Support for previous installs my $ff = $config_directory . "/$current_theme/favorites.json"; if (-r $ff) { my $ffn = $ff; $ffn =~ s/\.json/-$remote_user.json/; rename_file($ff, $ffn); } my $f = &read_file_contents($config_directory . "/$current_theme/favorites-$remote_user.json"); print '
'; } sub print_panels_group_start { my ($id, $get) = @_; my $str = '
'; if ($get) { return $str; } print $str; } sub print_panels_group_end { my ($get) = @_; my $str = '
'; if ($get) { return $str; } print $str; } sub print_panel { my ($opened, $id, $title, $data, $get, $sorter, $ref) = @_; if ($sorter) { $sorter = ' data-sorter="' . $sorter . '" '; } if ($ref) { $ref = ' data-referrer="' . $ref . '" '; } my $str = '
' . $data . '
'; if ($get) { return $str; } print $str; } sub parse_license_date { if ($_[0] =~ /^(\d{4})-(\d+)-(\d+)$/) { return eval {timelocal(0, 0, 0, $3, $2 - 1, $1 - 1900)}; } return undef; } sub embed_logo { my $lgt; my $img; ((get_env('script_name') eq '/session_login.cgi' || get_env('script_name') eq '/pam_login.cgi') ? ($lgt = 'logo_welcome') : ($lgt = 'logo')); my $lnk = $config_directory . "/$current_theme/" . $lgt . ".png"; if (-r $lnk) { $img = (''); } if ($get_user_level eq '1') { my %reseller = get_user_acl(undef, 'virtual-server'); if (length $reseller{'logo'} > 4 && $reseller{'link'}) { $img = (''); } elsif ($reseller{'logo'}) { $img = (''); } } if ($img && $img !~ /="none"/) { print '
'; print $img; print '
' . "\n"; } } sub head { print "Content-type: text/html\n\n"; } sub embed_login_head { my ($inline) = @_; my $ext = (theme_debug_mode() ? 'src' : 'min'); # Define page title my $title = $text{'session_header'}; print '', "\n"; embed_noscript(); print '', "\n"; embed_favicon(); print '', $title, '', "\n"; print '' . "\n"; if ($inline) { my $file_contents = read_file_contents("$root_directory/$current_theme/unauthenticated/css/bundle.min.css"); print ''; if (theme_night_mode_login()) { my $file_contents = read_file_contents("$root_directory/$current_theme/unauthenticated/css/palettes/nightrider.min.css"); print ''; } my $file_contents = read_file_contents("$root_directory/$current_theme/unauthenticated/css/fonts-roboto.min.css"); print ''; } else { print '' . "\n"; print ''; embed_css_night_rider(); embed_css_fonts(); } embed_background(); embed_styles(); embed_overlay_head(); print '', "\n"; } sub error_40x { my %miniserv; get_miniserv_config(\%miniserv); our %theme_config = (settings($config_directory . "/$current_theme/settings-admin", 'settings_'), settings($config_directory . "/$current_theme/settings-root", 'settings_')); # Get block time to refresh the page afterwards my $block_time = $miniserv{'blockhost_time'} < $miniserv{'blockuser_time'} ? $miniserv{'blockuser_time'} : $miniserv{'blockhost_time'}; if ($block_time < 30) { $block_time = 30; } $block_time += 5; my $sec = lc(get_env('https')) eq 'on' ? "; secure" : ""; my $sidname = "sid"; print "Set-Cookie: $sidname=x; path=/$sec\r\n" if ($in{'logout'}); print "Set-Cookie: redirect=1; path=/\r\n"; print "Set-Cookie: testing=1; path=/$sec\r\n"; my $charset = &get_charset(); &PrintHeader($charset); print '', "\n"; print '', "\n"; embed_login_head(!$main::session_id); print '' . "\n"; print ''; embed_overlay_prebody(); print '
' . "\n"; if (defined($in{'code'})) { print '
' . "\n"; print ' ' . html_escape($in{'code'}) . '
' . html_escape($in{'message'}) . "\n"; print '
' . "\n"; } &footer(); } sub theme_update_incompatible { my ($authentic_remote_data, $force_stable) = @_; my $webmin_compatible_version; my $usermin_compatible_version; my @notice; $force_stable ||= 0; my $force_button = '' . $theme_text{'theme_xhred_global_click_here'} . ''; my $usermin_enabled_updates = ($theme_config{'settings_sysinfo_theme_updates_for_usermin'} ne 'false' ? 1 : 0); my ($authentic_remote_version) = $authentic_remote_data =~ /^version=(.*)/gm; $authentic_remote_data =~ /^depends=(\d.\d\d\d)\s+(\d.\d\d\d)|(\d.\d\d\d)/gm; $webmin_compatible_version = $3 ? $3 : $1; $usermin_compatible_version = $2; my $webmin_version_file = "$root_directory/version"; my $usermin_version_file = "$has_usermin_root_dir/version"; my $get_latest_dev = sub { my ($file, $var) = @_; if (-r $file) { my $version_dev = read_file_lines($file, 1)->[0]; if ($version_dev =~ /\d\.\d{4,}/) { my @file_stat = stat($file); if ($file_stat[9] > time() - (60 * 60)) { ${$var} = 1; } } } }; # Do we have latest dev version of Webmin installed &$get_latest_dev($webmin_version_file, \$webmin_compatible_version); # Do we have latest dev version of Usermin installed &$get_latest_dev($usermin_version_file, \$usermin_compatible_version); if ( ($authentic_remote_version && $webmin_compatible_version && $usermin_compatible_version && (get_webmin_version() < $webmin_compatible_version) && ($has_usermin && $usermin_enabled_updates && $has_usermin_version < $usermin_compatible_version)) ) { @notice = { "incompatible" => ( theme_text('theme_git_patch_incompatible_message', $theme_text{'theme_name'}, $authentic_remote_version, $theme_text{'theme_xhred_titles_wm'}, $webmin_compatible_version, $theme_text{'theme_xhred_titles_um'}, $usermin_compatible_version ) . " " . theme_text('theme_git_patch_incompatible_message_desc', $force_button, ($theme_text{'theme_xhred_titles_wm'} . "/" . $theme_text{'theme_xhred_titles_um'}) ) ) }; } elsif ( ($authentic_remote_version && $webmin_compatible_version && (get_webmin_version() < $webmin_compatible_version)) ) { @notice = { "incompatible" => ( theme_text('theme_git_patch_incompatible_message_s', $theme_text{'theme_name'}, $authentic_remote_version, $theme_text{'theme_xhred_titles_wm'}, $webmin_compatible_version ) . " " . theme_text('theme_git_patch_incompatible_message_desc', $force_button, $theme_text{'theme_xhred_titles_wm'} ) ) }; } elsif ( ($authentic_remote_version && $usermin_compatible_version && ($has_usermin && $usermin_enabled_updates && $has_usermin_version < $usermin_compatible_version)) ) { @notice = { "incompatible" => ( theme_text('theme_git_patch_incompatible_message_s', $theme_text{'theme_name'}, $authentic_remote_version, $theme_text{'theme_xhred_titles_um'}, $usermin_compatible_version ) . " " . theme_text('theme_git_patch_incompatible_message_desc', $force_button, $theme_text{'theme_xhred_titles_um'} ) ) }; } return @notice; } sub theme_remote_version { my ($data, $force_stable_check, $force_beta_check, $nocache) = @_; my $remote_version = 0; my $remote_release; my $error; if (($theme_config{'settings_sysinfo_theme_updates'} eq 'true' || $data) && $get_user_level eq '0' && $in =~ /xhr-/) { if (($tconfig{'beta_updates'} eq '1' || $force_beta_check) && !$force_stable_check) { if (!$nocache) { $remote_version = theme_cached('version-theme-development'); } if (!$remote_version) { http_download('api.github.com', '443', '/repos/authentic-theme/authentic-theme/contents/theme.info', \$remote_version, \$error, undef, 1, undef, undef, 30, undef, undef, { 'accept', 'application/vnd.github.v3.raw' }); theme_cached('version-theme-development', $remote_version, $error); } } else { if (!$nocache) { $remote_version = theme_cached('version-theme-stable'); } if (!$remote_version) { http_download('api.github.com', '443', '/repos/authentic-theme/authentic-theme/releases/latest', \$remote_release, \$error, undef, 1, undef, undef, 30); $remote_release =~ /tag_name":"(.*?)"/; http_download('api.github.com', '443', '/repos/authentic-theme/authentic-theme/contents/theme.info?ref=' . $1 . '', \$remote_version, \$error, undef, 1, undef, undef, 30, undef, undef, { 'accept', 'application/vnd.github.v3.raw' }); theme_cached('version-theme-stable', $remote_version, $error); } } } if ($data) { return $remote_version; } else { ($remote_version) = $remote_version =~ /^version=(.*)/m; return $remote_version; } } sub theme_cached { my ($id, $cvalue, $error) = @_; $id || die "Can't use undefined as cache filename"; my ($theme_var_dir) = theme_var_dir(); my $fcached = "$theme_var_dir/$id"; my @cached = stat($fcached); my $ctime = $theme_config{'settings_cache_interval'} || 24 * 60 * 60; my $cache = read_file_contents($fcached); my $cdata = $cache ? unserialise_variable($cache) : undef; my @data; if (@cached && $cached[9] > time() - $ctime) { # Use cache for now @data = @$cdata; } else { # Error when catching remote data if ($error) { if ($cdata) { # Error: Use current cache for another period @data = @$cdata; } else { # Error: No cache available return undef; } } elsif ($cvalue) { # Use supplied data push(@data, $cvalue); } if (@data) { # Write cache my $fh = "cache"; open_tempfile($fh, ">$fcached"); print_tempfile($fh, serialise_variable(\@data)); close_tempfile($fh); } } return wantarray ? @data : $data[0]; } sub theme_var_dir { my $product_var = $var_directory || get_env('webmin_var'); if (!$product_var) { open(VARPATH, "$config_directory/var-path"); chop($product_var = ); close(VARPATH); } my $var_dir = $product_var . "/modules"; my $theme_var_dir = "$var_dir/$current_theme"; if (!-d $var_dir) { mkdir($var_dir, 0700); } if (!-d $theme_var_dir) { mkdir($theme_var_dir, 0700); } else { chmod(0700, $theme_var_dir); } return ($theme_var_dir, $product_var); } sub clear_theme_cache { my ($root) = @_; my $salt = substr(encode_base64($main::session_id), 0, 16); my $tmp_dir = tempname_dir(); my ($theme_var_dir, $product_var) = theme_var_dir(); # Clear cached files if ($root) { unlink_file("$theme_var_dir/version-theme-stable"); unlink_file("$theme_var_dir/version-theme-development"); unlink_file("$theme_var_dir/version-csf-stable"); # Clear stats history unlink_file("$theme_var_dir/stats-$remote_user.json"); kill_byname("$current_theme/stats.cgi", 9); } # Remove cached downloads unlink_file("$product_var/cache"); # Remove and regenerate OS cache if (&foreign_available('webmin')) { &foreign_require("webmin"); &webmin::detect_operating_system(); } # Clear potentially stuck BIND cache if (&foreign_available('bind8')) { &foreign_require("bind8"); &bind8::flush_zone_names(); } # Clear potentially stuck Apache cache if (&foreign_available('apache')) { &foreign_require("apache", "postinstall.pl"); &apache::module_install(); } # Clear links cache if (&foreign_available('virtual-server')) { &foreign_require("virtual-server"); &virtual_server::clear_links_cache(); } # Clear potentially stuck menus and other cache flush_webmin_caches(); # Clear session specific temporary files opendir(my $dir, $tmp_dir); grep {unlink_file("$tmp_dir/$_") if (/^\.theme/ && $_ =~ /$salt/)} readdir($dir); closedir $dir; } sub theme_make_config_dir { my $_wm_at_conf_dir = $config_directory . '/' . $current_theme; if (!-d $_wm_at_conf_dir) { mkdir($_wm_at_conf_dir, 0755); } else { chmod(0755, $_wm_at_conf_dir); } if ($has_usermin) { (my $_um_at_conf_dir = $config_directory) =~ s/webmin/usermin/; if (!-d $_um_at_conf_dir) { mkdir($_um_at_conf_dir, 0755); } else { chmod(0755, $_um_at_conf_dir); } } } sub get_theme_user_link { my $is_hidden = (!foreign_available("webmin") && $theme_config{'settings_theme_config_admins_only_privileged'} eq 'true' ? ' hidden-force ' : undef); my $is_hidden_link = ($get_user_level ne '0' ? ' hidden-force ' : undef); my $link = '/tconfig.cgi'; my $mversion = theme_version(1, 1); return '' . theme_version() . $mversion . '
'; } sub get_xhr_request { if ($in =~ /xhr-/) { head(); if ($in{'xhr-settings'} eq '1') { if ($in{'restore'} eq '1') { theme_config_restore(); } } elsif ($in{'xhr-manage-config'} eq '1') { if ($in{'save'} eq '1') { theme_config_save(); } elsif ($in{'load'} eq '1') { print theme_config_get(); } } elsif ($in{'xhr-get_available_modules'} eq '1') { print get_available_modules('json'); } elsif ($in{'xhr-get_theme_locale_languages'} eq '1') { my %__config = settings(get_tuconfig_file(), 'config_'); print ui_select( "config_portable_theme_locale_languages", ($__config{'config_portable_theme_locale_languages'} ? $__config{'config_portable_theme_locale_languages'} : get_before_delimiter($current_lang, '.') ), [ map { !string_contains(lc($_->{'lang'}), 'utf') ? [get_before_delimiter(lc(replace('_', '-', $_->{'lang'})), '.'), $_->{'desc'}] : () } list_languages() ]); } elsif ($in{'xhr-get_size'} eq '1') { set_user_level(); my $module = $in{'xhr-get_size_cmodule'}; my $jailed_user = get_fm_jailed_user($module); my $path = ($jailed_user || get_access_data('root')) . $in{'xhr-get_size_path'}; my $nodir = $in{'xhr-get_size_nodir'}; my $home = ($jailed_user || get_user_home()); if (($jailed_user || $get_user_level eq '3') && !string_starts_with($path, $home)) { $path = $home . $path; $path =~ s/\/\//\//g; } if ($nodir && -d $path) { print "$theme_text{'theme_xhred_global_error'}|-2"; } elsif (!-r $path) { print "$theme_text{'theme_xhred_global_error'}|-1"; } else { my $size = recursive_disk_usage($path); print nice_size($size, -1) . '|' . nice_number($size); } } elsif ($in{'xhr-get_list'} eq '1') { my $path = "$in{'xhr-get_list_path'}"; my $module = $in{'xhr-get_list_cmodule'}; my @dirs; my $jailed_user = get_fm_jailed_user($module); if ($jailed_user || $get_user_level eq '2' || $get_user_level eq '4') { $path = ($jailed_user || get_user_home()) . $path; } opendir(my $dirs, $path); while (my $dir = readdir $dirs) { next unless -d $path . '/' . $dir; next if $dir eq '.' or $dir eq '..'; push @dirs, $dir; } closedir $dirs; @dirs = sort {"\L$a" cmp "\L$b"} @dirs; print convert_to_json(\@dirs); } elsif ($in{'xhr-encoding_convert'} eq '1') { my $module = $in{'xhr-encoding_convert_cmodule'}; my $jailed_user = get_fm_jailed_user($module, 1); my $jailed_user_home = get_fm_jailed_user($module); my $cfile = $in{'xhr-encoding_convert_file'}; if ($jailed_user) { switch_to_unix_user_local($jailed_user); $cfile = $jailed_user_home . $cfile; } else { set_user_level(); } my $data = &ui_read_file_contents_limit( { 'file', $cfile, 'limit', $in{'xhr-encoding_convert_limit'}, 'reverse', $in{'xhr-encoding_convert_reverse'}, 'head', $in{'xhr-encoding_convert_head'}, 'tail', $in{'xhr-encoding_convert_tail'} }); if (-s $cfile < 128 || -T $cfile) { eval {$data = Encode::encode('utf-8', Encode::decode($in{'xhr-encoding_convert_name'}, $data));}; } print $data; } elsif ($in{'xhr-file_stats'} eq '1') { ### This old and ugly designed `get_xhr_request` sub needs a refactoring my $module = $in{'xhr-file_stats_cmodule'}; my $jailed_user = get_fm_jailed_user($module, 1); my $jailed_user_home = get_fm_jailed_user($module); my $cfile = $in{'xhr-file_stats_file'}; if ($jailed_user) { switch_to_unix_user_local($jailed_user); $cfile = $jailed_user_home . $cfile; } else { set_user_level(); } my $fz = recursive_disk_usage($cfile); $fz = nice_size($fz, -1); my $f = &backquote_command("file " . quotemeta($cfile) . " 2>/dev/null"); my $s = &backquote_command("stat " . quotemeta($cfile) . " 2>/dev/null"); $f = trim($f); $s =~ /(Size:)(\s+)(\d+)(\s+)/; my $sz = length($3) + length($4); my $nz = length($fz); $sz -= $nz; $sz = " " x ($sz + 2); $s =~ s/(Size:)(\s+)(\d+)(\s+)/$1$2$fz$sz/; $s =~ s/(File:)(\s+)(.*)/$1$2$f/; my @data = [$fz, $s]; print convert_to_json(\@data); } elsif ($in{'xhr-get_gpg_keys'} eq '1') { my $module = $in{'xhr-get_gpg_keys_cmodule'}; my $jailed_user = get_fm_jailed_user($module, 1); switch_to_unix_user_local($jailed_user || undef); my ($public, $secret, $gpgpath) = get_gpg_keys($in{'xhr-get_gpg_keys_all'}); my %keys; $keys{'public'} = $public; $keys{'secret'} = $secret; $keys{'gpgpath'} = $gpgpath; print convert_to_json(\%keys); } elsif ($in{'xhr-get_user_level'} eq '1') { print $get_user_level; } elsif ($in{'xhr-get_update_notice'} eq '1') { print update_notice(); } elsif ($in{'xhr-get_nice_size'} eq '1') { print nice_size($in{'xhr-get_nice_size_sum'}, -1); } elsif ($in{'xhr-get_command_exists'} eq '1') { print has_command($in{'xhr-get_command_exists_name'}); } elsif ($in{'xhr-get_symlink'} eq '1') { print(resolve_links(get_access_data('root') . ($in{'xhr-get_symlink_path'}))); } elsif ($in{'xhr-theme_temp_data'} eq '1') { if ($in{'xhr-theme_temp_data_action'} eq 'set') { set_theme_temp_data($in{'xhr-theme_temp_data_name'}, $in{'xhr-theme_temp_data_value'}); } elsif ($in{'xhr-theme_temp_data_action'} eq 'get') { print get_theme_temp_data($in{'xhr-theme_temp_data_name'}, $in{'xhr-theme_temp_data_keep'}); } } elsif ($in{'xhr-shell-pop'}) { my $file = get_history_shell_file(); my $index = (int($in{'xhr-shell-pop'}) - 1); my $history = read_file_lines($file); if (@$history[$index]) { splice(@$history, $index, 1); flush_file_lines($file); print 1; } } elsif ($in{'xhr-shell-insert'}) { my $file = get_history_shell_file(); my $history = read_file_lines($file); push(@$history, $in{'xhr-shell-inserted'}) if ($in{'xhr-shell-inserted'}); flush_file_lines($file); print convert_to_json($history); } elsif ($in{'xhr-get_autocompletes'} eq '1') { my @data = get_autocomplete_shell($in{'xhr-get_autocomplete_type'}, $in{'xhr-get_autocomplete_string'}); print convert_to_json(\@data); } elsif ($in{'xhr-theme_latest_version'} eq '1') { my @current_versions; push(@current_versions, (theme_remote_version(1, 1) =~ /^version=(.*)/m), (theme_remote_version(1, 0, 1) =~ /^version=(.*)/m)); print convert_to_json(\@current_versions); } elsif ($in{'xhr-theme_clear_cache'} eq '1') { my $is_root = $get_user_level eq '0'; clear_theme_cache($is_root); } elsif ($in{'xhr-update'} eq '1' && foreign_available('webmin')) { my @update_rs; my $version_type = ($in{'xhr-update-type'} eq '-beta' ? '-beta' : '-release'); my $update_force = $in{'xhr-update-force'}; my $usermin_enabled_updates = ($theme_config{'settings_sysinfo_theme_updates_for_usermin'} ne 'false' ? 1 : 0); if (!has_command('git') || !has_command('curl') || !has_command('bash')) { @update_rs = { "no_git" => replace((!has_command('curl') || !has_command('bash') ? '>git<' : '~'), (!has_command('curl') ? '>curl<' : '>bash<'), $theme_text{'theme_git_patch_no_git_message'} ), }; print convert_to_json(\@update_rs); } else { if ($update_force ne "1") { my $authentic_remote_data; if ($version_type eq '-release') { $authentic_remote_data = theme_remote_version(1, 1, undef, 1); } else { $authentic_remote_data = theme_remote_version(1, 0, 1, 1); } if ($authentic_remote_data eq '0') { @update_rs = { "no_connection" => $theme_text{'theme_git_update_locked'} }; print convert_to_json(\@update_rs); exit; } @update_rs = theme_update_incompatible($authentic_remote_data, ($version_type eq '-release' ? 1 : 0)); if (@update_rs) { print convert_to_json(\@update_rs); exit; } } my $usermin = ($has_usermin && $usermin_enabled_updates); my $usermin_root; backquote_logged("yes | $root_directory/$current_theme/theme-update.sh $version_type -no-restart"); if ($usermin) { $usermin_root = $root_directory; $usermin_root =~ s/webmin/usermin/; backquote_logged("yes | $usermin_root/$current_theme/theme-update.sh $version_type -no-restart"); } my $tversion = theme_version(); my $mversion = theme_version(1, 1); $tversion = $tversion . $mversion; @update_rs = { "success" => ($usermin ? theme_text('theme_git_patch_update_success_message2', $tversion) : theme_text('theme_git_patch_update_success_message', $tversion) ) }; print convert_to_json(\@update_rs); } } elsif ($in{'xhr-info'} eq '1') { my @info = theme_list_combined_system_info(); our ($cpu_percent, $mem_percent, $virt_percent, $disk_percent, $host, $os, $webmin_version, $virtualmin_version, $cloudmin_version, $authentic_theme_version, $local_time, $kernel_arch, $cpu_type, $cpu_temperature, $hdd_temperature, $uptime, $running_proc, $load, $real_memory, $virtual_memory, $disk_space, $package_message, $csf_title, $csf_data, $csf_remote_version, $authentic_remote_version ) = get_sysinfo_vars(\@info); # Build update info my @updated_info = { "data" => 1, "cpu_percent" => $cpu_percent, "mem_percent" => $mem_percent, "virt_percent" => $virt_percent, "disk_percent" => $disk_percent, "host" => $host, "os" => $os, "webmin_version" => $webmin_version, "virtualmin_version" => $virtualmin_version, "cloudmin_version" => $cloudmin_version, "authentic_theme_version" => $authentic_theme_version, "local_time" => $local_time, "kernel_arch" => $kernel_arch, "cpu_type" => $cpu_type, "cpu_temperature" => $cpu_temperature, "hdd_temperature" => $hdd_temperature, "uptime" => $uptime, "proc" => $running_proc, "cpu" => $load, "mem" => $real_memory, "virt" => $virtual_memory, "disk" => $disk_space, "package_message" => $package_message, "csf_title" => $csf_title, "csf_data" => $csf_data, "csf_remote_version" => $csf_remote_version, "authentic_remote_version" => $authentic_remote_version, "csf_deny" => (defined(&csf_temporary_list) ? csf_temporary_list() : undef), "collect_interval" => get_module_config_data('system-status', 'collect_interval'), "extended_si" => get_extended_sysinfo(\@info, undef), "warning_si" => get_sysinfo_warning(\@info), }; print convert_to_json(\@updated_info); } elsif ($in{'xhr-search-in-file'} eq '1') { set_user_level(); my @files = split(/,/, $in{'xhr-search-in-file-files'}); my $match = trim($in{'xhr-search-in-file-string'}); my @match; fdo { my ($file, $line, $text) = @_; if ($text =~ /\Q$match\E/i) { push(@match, ([$files[$file] => [html_escape(substr($text, 0, 120)), $line]])); } } @files; print convert_to_json(\@match); } elsif ($in{'xhr-csf-unload'} eq '1') { lib_csf_control('unload'); } elsif ($in{'xhr-gennewpass'} eq 'get') { my $pass; if (&foreign_available('virtual-server')) { &foreign_require("virtual-server"); $pass = &virtual_server::random_password(); } elsif (&foreign_available('useradmin')) { &foreign_require("useradmin", "user-lib.pl"); $pass = &useradmin::generate_random_password(); } print $pass; } exit; } } sub init_type { (($ENV{'CONTENT_TYPE'} =~ /multipart\/form-data/i) ? ReadParseMime() : ($ENV{'SCRIPT_NAME'} =~ /session_login|pam_login/i ? ReadParse(undef, undef, undef, 2) : ReadParse())); } sub init { # Don't log XHR requests my %tmp_miniserv; get_miniserv_config(\%tmp_miniserv); my $nolog = quotemeta('/stats.cgi?xhr-stats=general'); $nolog =~ s/\\ / /g; if ($tmp_miniserv{'nolog'} ne $nolog) { $tmp_miniserv{'nolog'} = $nolog; put_miniserv_config(\%tmp_miniserv); reload_miniserv(); } # Make sure that config directory exists theme_make_config_dir(); # Provide unobstructive access for AJAX calls get_xhr_request(); # Load module lib if available lib_csf_control('load'); } sub content { # Mobile toggle print '
'; print '' . "\n"; print '
' . "\n"; # Navigation do("$ENV{'THEME_ROOT'}/navigation-lib.pl"); print '' . "\n"; # Authenticated logo embed_logo(); # Favorites menu print_favorites(); # Content print '
' . "\n"; print '
' . "\n"; } sub update_notice { my $changelog_data = (read_file_contents($root_directory . '/' . $current_theme . "/CHANGELOG.md") =~ /#### Version(.*?)/s)[0]; if ($changelog_data) { $changelog_data =~ s/###(.*?)\)/<\/ul>@{[get_version_link($1, 2)]}
    /g; } else { $changelog_data = (read_file_contents($root_directory . '/' . $current_theme . "/CHANGELOG.md") =~ /### Version(.*?)/s)[0]; } my @changelog_version = split /\n/, $changelog_data; $changelog_data =~ s/^[^\n]*\n/\n/s; $changelog_data =~ s/`(.*?)`/$1<\/code>/g; $changelog_data =~ s/__(.*?)__/$1<\/strong>/g; $changelog_data =~ s/_(.*?)_/$1<\/em>/g; $changelog_data =~ s/(Fixed bugs)/$1<\/span>/g; $changelog_data =~ s/\[([^\[]+)\]\(([^\)]+)\)/$1<\/a>/g; $changelog_data =~ s/\n\*(.*)/\n
  • $1<\/li>/g; my @version = split(/ /, $changelog_version[0]); my $changelog_content = ' '; return $changelog_content; } sub get_cookies { my @r = split /; /, get_env('http_cookie'); my %c; foreach (@r) { my ($k, $v) = split /=/, $_; $c{$k} = $v; } return %c; } sub get_access_data { my ($key) = @_; if ($key) { if ($key eq 'root' && $gaccess{'root'} eq '/') { return undef; } else { return $gaccess{$key}; } } else { return %gaccess; } } sub get_available_modules { my ($json) = @_; my @mods; foreach my $x (get_all_module_infos()) { if ($x->{'dir'} ne undef && &foreign_available($x->{'dir'})) { push @mods, $x->{'dir'}; } } if ($json eq 'json') { return convert_to_json(\@mods); } else { return @mods; } } sub theme_config_save { theme_make_config_dir(); my %u = settings_filter(%in); my %a = %u; # Never allow saving privileged options sent from user delete @u{ grep(/_privileged$/, keys %u) }; write_file(get_tuconfig_file(), \%u); # Master administrator must also save certain options to # global `settings.js` config file to affect all users if ($get_user_level eq '0') { delete @a{ grep(!/^settings_/, keys %a) }; # Never save user-based options to global config delete @a{ grep(/_user$/, keys %a) }; write_file(get_tgconfig_file(), \%a); # Check for Usermin configuration if ($has_usermin_conf_dir) { my $theme_settings_webmin_file = get_tgconfig_file(); my $theme_settings_usermin_file = $theme_settings_webmin_file; $theme_settings_usermin_file =~ s/$config_directory/$has_usermin_conf_dir/; if (!-l $theme_settings_usermin_file) { symlink_file($theme_settings_webmin_file, $theme_settings_usermin_file); } } } } sub theme_config_get { my %tuconfig; my $tuconfig_file = get_tuconfig_file(); if (-f $tuconfig_file) { my %tuconfig = settings($tuconfig_file); return convert_to_json(\%tuconfig); } else { return convert_to_json(); } } sub theme_config_restore { my $tuconfig_file = get_tuconfig_file(); unlink_file($tuconfig_file); if ($get_user_level eq '0') { my $tgconfig_file = get_tgconfig_file(); unlink_file($tgconfig_file); if ($has_usermin) { my $tugconfig_file = $tgconfig_file; $tugconfig_file =~ s/$config_directory/$has_usermin_conf_dir/; unlink_file($tugconfig_file); } } } sub get_user_acl { my ($key, $module) = @_; if ($module) { $module = '/' . $module; } my $acl = "$config_directory$module/$remote_user.acl"; my %config; read_file($acl, \%config); if (-r $acl) { if ($key) { return $config{$key}; } else { return %config; } } else { return undef; } } sub get_module_config_data { my ($module, $key) = @_; if (-r $config_directory . '/' . $module . '/config') { my %config; read_file(($config_directory . '/' . $module . '/config'), \%config); if ($key) { return $config{$key}; } else { return %config; } } else { return undef; } } sub get_history_shell_file { my $file; if ($in{'xhr-shell-cms'} eq "1") { my $id = $in{'xhr-shell-cmsid'}; $id =~ s/[^\p{L}\p{N}.\-\/]//g; $file = "$config_directory/server-manager/previous/$id"; } else { $file = "$config_directory/shell/previous.$remote_user"; } return $file; } sub get_autocomplete_shell { my ($type, $string) = @_; my ($cd, $cmd, $cmd2, $cd_cmd, $command, @rs, @rs_tmp); if ($type eq 'commands') { $command = '-c'; } elsif ($type eq 'groups') { $command = '-g'; } elsif ($type eq 'service') { (!string_starts_with($string, '::::') && ($command = '-s')); } elsif ($type eq 'systemctl') { $command = undef; } elsif ($type eq 'users') { $command = '-u'; } else { my @strings = split /::::/, $string; ($cd, $string, $cmd, $cmd2) = @strings[0, 1, 2, 3]; $cd_cmd = "cd $cd; "; $command = '-o default'; } if ($command) { @rs = array_unique( backquote_command($cd_cmd . "bash -c 'compgen " . $command . " '" . quotemeta($cmd2 ? $cmd2 : $string) . "")); } else { if ($type eq 'service' && has_command('service')) { my @cmd = split /::::/, $string; my $units_tmp = backquote_command("service " . quotemeta($cmd[1])); my ($unit_tmp) = $units_tmp =~ / \( ( [^\)]+ ) \) /x; if (!$unit_tmp) { ($unit_tmp) = $units_tmp =~ / { ( [^}]+ ) } /x; } if (!$unit_tmp) { ($unit_tmp) = $units_tmp =~ / \[ ( [^]]+ ) \] /x; } $unit_tmp =~ s/\s+//g; $unit_tmp =~ s/\|/,/g; $unit_tmp =~ s/;/,/g; my @units_tmp = split /,/, $unit_tmp; my @units_possible_tmp = ('start', 'stop', 'restart', 'try-restart', 'reload', 'force-reload', 'status'); @rs_tmp = (@units_tmp ? @units_tmp : @units_possible_tmp); my @rs_cmd; if ($cmd[2]) { foreach my $cmd (@rs_tmp) { if (string_starts_with($cmd, $cmd[2])) { push @rs_cmd, $cmd; } } @rs = @rs_cmd; } else { @rs = @rs_tmp; } } if ($type eq 'systemctl' && has_command('systemctl')) { my (@units, @units_tmp); @units_tmp = array_unique(backquote_command("systemctl list-unit-files")); my $i = 0; my $n = $#units_tmp; foreach my $unit (@units_tmp) { my @tmp = split / {1,}/, $unit; my ($unit_tmp, $status_tmp) = @tmp[0, 1]; if ($i && --$n && $unit_tmp && (!$string || string_starts_with($unit_tmp, $string))) { push @units, $unit_tmp; } $i++; } @rs = @units; } } if ($cd || $cmd2) { my @rs_tmp; foreach my $file (@rs) { if (-d $file || -d ($cd . $file)) { push @rs_tmp, ($file . '/'); } else { if ($cmd ne 'cd') { push @rs_tmp, $file; } } } @rs = @rs_tmp; } return @rs; } 1;