%PDF- %PDF-
Direktori : /usr/share/webmin/authentic-theme/extensions/file-manager/ |
Current File : //usr/share/webmin/authentic-theme/extensions/file-manager/file-manager-lib.pm |
# # Authentic Theme (https://github.com/authentic-theme/authentic-theme) # Copyright Ilia Rostovtsev <programming@rostovtsev.io> # Licensed under MIT (https://github.com/authentic-theme/authentic-theme/blob/master/LICENSE) # use strict; use File::Basename; use lib (dirname(__FILE__) . '/../../lib'); do(dirname(__FILE__) . "/../../authentic-funcs.pm"); use Cwd 'abs_path'; use Encode qw(decode encode); use File::MimeInfo; use File::Find; use File::Grep qw( fdo ); use POSIX; use JSON::PP; our (%access, %gconfig, %in, %text, %theme_text, @remote_user_info, $base_remote_user, $config_directory, $current_theme, %userconfig, @allowed_paths, $base, $cwd, $path, $remote_user); our $checked_path; our $module_path; our %request_uri = get_request_uri(); set_module($request_uri{'module'}); get_libs($request_uri{'module'}); sub set_module { my ($module) = @_; $module_path = get_env('document_root') . '/' . $module; set_env('foreign_module_name', $module); set_env('foreign_root_directory', $module_path); } sub get_libs { my ($module) = @_; require($module_path . '/' . $module . '-lib.pl'); &ReadParse(); get_paths(); switch_to_user($in{'username'}); $checked_path = $path; if (join(" , ", @allowed_paths) ne '/') { $checked_path =~ s/$in{'cwd'}\//\//ig; } %text = (load_language($current_theme), load_language($module), %text); %theme_text = %text; } sub get_type { my ($dir) = @_; if (-d $dir) { return 1; } else { return 0; } } sub get_errors { my %errors = %{ $_[0] }; if (scalar %errors) { return convert_to_json(\%errors); } else { return undef; } } sub get_request_uri { (my $uri = get_env('request_uri')) =~ s/\?/&/; my @r = split /&/, $uri; my %c; foreach (@r) { my ($k, $v) = split /=/, $_; $c{$k} = $v; } return %c; } sub get_user_config { my ($k) = @_; my %t; read_file("$config_directory/$current_theme/settings-$remote_user", \%t); if ($k) { my $v = $t{$k}; $v =~ s/'|;//g; return $v; } else { my %c = map {(my $v = $t{$_}) =~ s/'|;//g; $_ => $v} keys %t; return %c; } } sub kill_previous { my $pid = tokenize($_[0]); if ($pid) { kill(9, $pid); } tokenize($_[0], $_[1]); } sub tokenize { my ($key, $value) = @_; my $salt = substr(encode_base64($main::session_id), 0, 16); my %var; my $user = $remote_user; my $tmp_file; $key =~ s/(?|([\w-]+$)|([\w-]+)\.)//; $key = $1; $key =~ tr/A-Za-z0-9//cd; $user =~ tr/A-Za-z0-9//cd; $salt =~ tr/A-Za-z0-9//cd; $tmp_file = tempname('.theme_' . $salt . '_' . get_product_name() . '_' . $key . '_' . $user); $var{$key} = $value; if ($value) { write_file($tmp_file, \%var); } else { my %theme_temp_data; read_file($tmp_file, \%theme_temp_data); unlink_file($tmp_file); return $theme_temp_data{$key}; } } sub get_pagination { my ($page, $pages, $query) = @_; our ($path); my $search_follow_symlinks = $in{'follow'}; my $search_case_insensitive = $in{'caseins'}; my $search_grep = $in{'grepstring'}; my $fsid = $in{'fsid'}; my $exclude = $in{'exclude'}; my $regex = $in{'regex'}; my $all_items = $in{'all_items'}; my $left = $page - 2; my $right = $page + 3; my @range; my $last; my $pagination; my $invisible = $pages == 1 ? ' invisible' : undef; my $start = sub { my $start; my $disabled = ($page == 1 ? " disabled" : undef); $start = "<div class=\"dataTables_paginate paging_simple_numbers spaginates$invisible\">"; $start .= '<ul class="pagination">'; $start .= "<li class='paginate_button previous$disabled'>"; $start .= '<a><i class="fa fa-fw fa-angle-left"></i></a>'; $start .= "</li>"; return $start; }; my $current = sub { my ($i) = @_; my $end; my $active = ($page eq $i ? " active" : undef); $end = "<li class='paginate_button$active'>"; $end .= "<a class='spaginated' href='list.cgi?page=$i&path=@{[urlize($path)]}&query=@{[urlize($query)]}&follow=$search_follow_symlinks&caseins=$search_case_insensitive&grepstring=$search_grep&fsid=$fsid&exclude=$exclude®ex=$regex&all_items=$all_items'>@{[nice_number($i, ',')]}</a>"; $end .= "</li>"; return $end; }; my $range = sub { my $range; $range = '<li class="paginate_button disabled">'; $range .= '<a>...</a>'; $range .= "</li>"; }; my $end = sub { my $end; my $disabled = ($page == $pages ? " disabled" : undef); $end = "<li class='paginate_button next$disabled'>"; $end .= '<a><i class="fa fa-fw fa-angle-right"></i></a>'; $end .= "</li>"; $end .= '</ul>'; $end .= '</div>'; return $end; }; for (my $i = 1; $i <= $pages; $i++) { if ($i == 1 || $i == $pages || $i >= $left && $i < $right) { push(@range, $i); } } foreach my $i (@range) { if ($last) { if ($i - $last == 2) { $pagination .= &$current($last + 1); } elsif ($i - $last != 1) { $pagination .= &$range(); } } $pagination .= &$current($i); $last = $i; } $pagination = &$start() . $pagination . &$end(); return $pagination; } sub test_all_items_query { return $in{'all_items'} eq '3' ? 3 : 0; } sub get_entries_list { my @entries_list; if (test_all_items_query()) { if ($in{'query'}) { @entries_list = exec_search('list'); } else { find( { wanted => sub { my $found = $File::Find::name; push(@entries_list, $_); }, }, $cwd); @entries_list = grep {$_ ne '.' && $_ ne '..'} @entries_list; } } else { @entries_list = split(/\0/, $in{'name'}); } return @entries_list; } sub extra_query { my $page = $in{'page'}; my $query = $in{'query'}; my $paginate = $in{'paginate'}; my $follow = $in{'follow'}; my $caseins = $in{'caseins'}; my $grepstring = $in{'grepstring'}; my $fsid = $in{'fsid'}; my $exclude = $in{'exclude'}; my $regex = $in{'regex'}; my $all_items = $in{'all_items'}; return "&page=$page&query=$query&paginate=$paginate&follow=$follow&caseins=$caseins&grepstring=$grepstring&fsid=$fsid&exclude=$exclude®ex=$regex&all_items=$all_items"; } sub set_response { my ($c) = @_; print "Set-Cookie: file-manager-response=" . $c . "; path=/\r\n"; } sub set_response_count { my ($c) = @_; print "Set-Cookie: file-manager-response_count=" . $c . "; path=/\r\n"; } sub fatal_errors { my @errors = @_; head(); print $text{'errors_occured'}; print "<ul>"; foreach my $error (@errors) { print("<li>$error</li>"); } print "</ul>"; } sub utf8_decode { my ($text) = @_; return decode('UTF-8', $text, Encode::FB_DEFAULT); } sub redirect_local { print "Location: $_[0]\n\n"; } sub convert_to_json_local { return JSON::PP->new->encode((@_ ? @_ : {})); } sub print_json_local { print "Content-type: application/json;\n\n"; print convert_to_json_local(@_); } sub print_error { my ($err_msg) = @_; my %err; $err{'error'} = $err_msg; print_json([\%err]); exit; } sub cache_search { my ($id, $searched_data) = @_; $id || return (); my $tmp_dir = tempname_dir(); my $fname = ".$remote_user-file-manager-scache"; my $fcached = $tmp_dir . "/$fname-$id"; my $dcached = read_file_contents($fcached); my $dcached_ready = $dcached ? unserialise_variable($dcached) : undef; my @data; # Clear previously cached data opendir(my $dir, $tmp_dir); my @tmps = grep {$_ =~ /$fname/} readdir($dir); closedir $dir; foreach (@tmps) { my $file = "$tmp_dir/$_"; my @stat = stat($file); if (@stat && $stat[9] < time() - (24 * 60 * 60)) { unlink_file($file); } } # Check if cache with requested id is available if (!$searched_data && -r $fcached && @$dcached_ready) { # Use cache for now @data = @$dcached_ready; if (@data) { return @data; } else { return (); } } elsif ($searched_data) { # Write cache my $fh = "cache"; open_tempfile($fh, ">$fcached"); print_tempfile($fh, serialise_variable($searched_data)); close_tempfile($fh); } else { return (); } } sub cache_search_delete { my ($id, $deleted_data) = @_; my @results_cached = cache_search($id); if (@results_cached) { @results_cached = grep { my $f = $_; !grep $f =~ /^\Q$_\E/, @$deleted_data } @results_cached; cache_search($id, \@results_cached); } } sub cache_search_rename { my ($id, $from, $to) = @_; my @results_cached = cache_search($id); if (@results_cached) { my @updated_cache; foreach my $file (@results_cached) { if ($file eq "/$from") { $file = "/$to"; } push(@updated_cache, $file); } cache_search($id, \@updated_cache); } } sub exec_search { my ($list) = @_; my $mask = $in{'query'}; my $grep = $in{'grepstring'}; my $fsid = $in{'fsid'}; my $exclude = $in{'exclude'}; my $replace = $in{'grepreplace'}; my $caseins = $in{'caseins'}; my $follow = ($in{'follow'} ? 1 : 0); my $regex = ($in{'regex'} ? 1 : 0); my @results; my @excludes; my @results_cached = cache_search($fsid); if (@results_cached) { return @results_cached; } find( { wanted => sub { my $found = $File::Find::name; if ($found ne $path) { my $found_text = $_; my $mask_text = $mask; if ($caseins) { $found_text = lc($found_text); $mask_text = lc($mask_text); $exclude = lc($exclude); } if ($exclude) { @excludes = split(';', $exclude); } if (($mask_text eq "*" || !$regex && (index($found_text, $mask_text) != -1)) || ($regex && $found_text =~ /$mask_text/)) { if (!$list) { $found =~ s/^\Q$cwd\E//g; } if ($follow || (!$follow && !-l $_)) { my $excluded; my $found_ = $found; $found_ = lc($found_) if ($caseins); if (@excludes) { foreach my $e (@excludes) { if ((!$regex && index($found_, $e) != -1) || ($regex && $found_ =~ /$e/)) { $excluded = 1; } } } if (!$exclude || (@excludes && !$excluded)) { push(@results, $found); } } } } }, follow => $follow, follow_skip => 2, }, $cwd); my @replaces; if (length($grep) || length($replace)) { if (length($grep)) { @results = map {&simplify_path("$cwd/$_")} @results; my @matched; fdo { my ($file, $line, $text) = @_; if ($caseins) { $text = lc($text); $grep = lc($grep); } if ((!$regex && index($text, $grep) != -1) || ($regex && $text =~ /$grep/)) { if (!grep(/^\Q$results[$file]\E$/, @replaces)) { push(@replaces, $results[$file]); } (my $sfile = $results[$file]) =~ s/^\Q$cwd\E//g; if (!grep(/^\Q$sfile\E$/, @matched)) { push(@matched, $sfile); } } } @results; undef(@results); @results = @matched; } if (length($replace)) { foreach my $file (@replaces) { if (-r $file) { if ($caseins) { (my $fc = read_file_contents($file)) =~ s/$grep/$replace/gi; write_file_contents($file, $fc); } else { (my $fc = read_file_contents($file)) =~ s/$grep/$replace/g; write_file_contents($file, $fc); } } } } } cache_search($fsid, \@results) if ($fsid && !length($replace)); return @results; } sub server_pagination_enabled { my ($totals, $max_allowed, $query) = @_; return ($totals > $max_allowed || ($query && $totals)); } sub print_content { my %list_data; my $query = $in{'query'}; my @list; # In case of search trim the list accordingly if ($query) { @list = exec_search(); } else { unless (opendir(DIR, $cwd)) { print_error("$text{'theme_xhred_global_error'}: [tt]`$cwd`[/tt]- $!."); exit; } @list = grep {$_ ne '.' && $_ ne '..'} readdir(DIR); closedir(DIR); } # Filter out not allowed entries if ($remote_user_info[0] ne 'root' && $allowed_paths[0] ne '$ROOT') { # Leave only allowed my @tmp_list; for $path (@allowed_paths) { my $slashed = $path; $slashed .= "/" if ($slashed !~ /\/$/); push @tmp_list, grep { string_starts_with($slashed, "$cwd/$_/") || index($slashed, $_) != -1 || index("$cwd/$_", $slashed) != -1 } @list; } # Remove duplicates my %hash = map {$_, 1} @tmp_list; @list = keys %hash; } my $page = 1; my $pagelimit = 4294967295; my $pages = 0; my $max_allowed = int($userconfig{'max_allowed'}); if ($max_allowed !~ /^[0-9,.E]+$/ || $max_allowed < 100 || $max_allowed > 10000) { $max_allowed = 1000; } my $totals = scalar(@list); my $totals_spliced = $totals; my $tuconfig_per_page = get_user_config('config_portable_module_filemanager_records_per_page'); if (server_pagination_enabled($totals, $max_allowed, $query)) { $page = int($in{'page'}) || 1; $pagelimit = int($in{'paginate'}) || int($tuconfig_per_page) || 30; $pages = ceil(($totals) / $pagelimit); if ($page > $pages) { $page = $pages; $in{'page'} = $page; } my $splice_start = $pagelimit * ($page - 1); my $splice_end = $pagelimit; if ($totals > 100000) { @list = sort {$a cmp $b} @list; } else { @list = map $_->[0], sort {$a->[1] <=> $b->[1] || $a->[0] cmp $b->[0]} map [$_, -f "$cwd/$_"], @list; } @list = splice(@list, $splice_start, $splice_end); $totals_spliced = scalar(@list); } @list = map {&simplify_path("$cwd/$_")} @list; my $setype = get_selinux_command_type(); my %secontext; my %attributes; # List attributes if ($userconfig{'columns'} =~ /attributes/ && get_attr_status()) { my $command = get_attr_command() . join(' ', map {quotemeta("$_")} @list); my $output = `$command`; my @attributesArr = map {[split(/\s+/, $_, 2)]} split(/\n/, $output); %attributes = map {$_->[1] => ('<span data-attributes="s">' . $_->[0] . '</span>')} @attributesArr; } # List security context if ($userconfig{'columns'} =~ /selinux/ && get_selinux_status()) { my $command = get_selinux_command() . join(' ', map {quotemeta("$_")} @list); my $output = `$command`; (!$setype && ($output =~ s/\n//g, $output =~ s/,\s/,/g)); my $delimiter = ($setype ? '\n' : ','); my @searray = map {[split(/\s+/, $_, 2)]} split(/$delimiter/, $output); %secontext = map {$_->[1] => ($_->[0] eq "?" ? undef : ('<span data-secontext>' . $_->[0] . '</span>'))} @searray; } # Get info about directory entries my @info = map {[$_, lstat($_), &mimetype($_), -d, -l $_, $secontext{$_}, $attributes{$_}]} @list; my @folders = map {$_} grep {$_->[15] == 1} @info; my @files = map {$_} grep {$_->[15] != 1} @info; if (server_pagination_enabled($totals, $max_allowed, $query)) { undef(@list); push(@list, @info); } else { @folders = sort {"\L$a->[0]" cmp "\L$b->[0]"} @folders; @files = sort {"\L$a->[0]" cmp "\L$b->[0]"} @files; undef(@list); push(@list, @folders, @files); } my $info_total; my $info_files = scalar(@files); my $info_folders = scalar(@folders); my @allowed_for_edit = split(/\s+/, $access{'allowed_for_edit'}); my %allowed_for_edit = map {$_ => 1} @allowed_for_edit; # Set icons variables my $edit_icon = "<i class='fa fa-edit' alt='$text{'edit'}'></i>"; my $rename_icon = "<i class='fa fa-font' title='$text{'rename'}'></i>"; my $extract_icon = "<i class='fa fa-external-link' alt='$text{'extract_archive'}'></i>"; my $goto_icon = "<i class='fa fa-arrow-right' alt='$text{'goto_folder'}'></i>"; my $server_pagination = undef; $list_data{'pagination_limit'} = undef; if (server_pagination_enabled($totals, $max_allowed, $query)) { $page = int($in{'page'}) || 1; $pagelimit = int($in{'paginate'}) || int($tuconfig_per_page) || 30; $pages = ceil(($totals) / $pagelimit); if ($page > $pages) { $page = $pages; $in{'page'} = $page; } $server_pagination = get_pagination($page, $pages, $query); $list_data{'pagination_limit'} = $in{'paginate'} || undef; my $pagination_text = $text{'theme_xhred_datatable_sinfo'}; my $start = $page * $pagelimit - $pagelimit + 1; my $end = $page * $pagelimit; if ($end > $totals) { $end = $totals; } $pagination_text =~ s/_START_/@{[nice_number($start, ",")]}/ig; $pagination_text =~ s/_END_/@{[nice_number($end, ",")]}/ig; $pagination_text =~ s/_TOTAL_/@{[nice_number($totals, ",")]}/ig; $list_data{'pagination_text'} = $pagination_text; } $list_data{'pagination'} = $server_pagination; my $total_with_pagination; if ($server_pagination) { $total_with_pagination = "_paginated"; } if ($info_files eq 1 && $info_folders eq 1) { $info_total = ('filemanager_global_info' . $total_with_pagination . '_total1'); } elsif ($info_files ne 1 && $info_folders eq 1) { $info_total = ('filemanager_global_info' . $total_with_pagination . '_total2'); } elsif ($info_files eq 1 && $info_folders ne 1) { $info_total = ('filemanager_global_info' . $total_with_pagination . '_total3'); } else { $info_total = ('filemanager_global_info' . $total_with_pagination . '_total4'); } $list_data{'total'} = "<div class='total'>" . ( $query ? (trim($text{'filemanager_global_search_results'}) . ": ") : ($server_pagination ? (trim($text{'filemanager_global_paginated_results'}) . ": ") : undef) ) . "" . utf8_decode( text(nice_number($info_total, ","), nice_number($info_files, ","), nice_number($info_folders, ","), nice_number($totals, ","), nice_number($pages, ",") ) ) . "</div>"; # Render current directory entries $list_data{'form'} = &ui_form_start("", "post", undef, "id='list_form'"); my @ui_columns = ('<input class="_select-unselect_" type="checkbox" onclick="selectUnselect(this)" />', ''); push @ui_columns, ('<span data-head-name>' . $text{'name'} . '</span>'); push @ui_columns, ('<span data-head-type>' . $text{'type'} . '</span>') if ($userconfig{'columns'} =~ /type/); push @ui_columns, ('<span data-head-actions>' . $text{'actions'} . '</span>'); push @ui_columns, ('<span data-head-size>' . $text{'size'} . '</span>') if ($userconfig{'columns'} =~ /size/); push @ui_columns, ('<span data-head-owner_user>' . $text{'ownership'} . '</span>') if ($userconfig{'columns'} =~ /owner_user/); push @ui_columns, ('<span data-head-permissions>' . $text{'permissions'} . '</span>') if ($userconfig{'columns'} =~ /permissions/); push @ui_columns, ('<span data-head-attributes>' . $text{'attributes'} . '</span>') if (get_attr_status() && $userconfig{'columns'} =~ /attributes/); push @ui_columns, ('<span data-head-selinux>' . $text{'selinux'} . '</span>') if (get_selinux_status() && $userconfig{'columns'} =~ /selinux/); push @ui_columns, ('<span data-head-last_mod_time>' . $text{'last_mod_time'} . '</span>') if ($userconfig{'columns'} =~ /last_mod_time/); $list_data{'rows'} = ''; for (my $count = 1; $count <= $totals_spliced; $count++) { if ($count > $totals) {last;} my $class = $count & 1 ? "odd" : "even"; my $link = $list[$count - 1][0]; $link =~ s/\Q$cwd\E\///; $link =~ s/^\///g; my $vlink = html_escape($link); $vlink = utf8_decode($vlink); my $hlink = html_escape($vlink); my $filename = $link; $filename =~ /\/([^\/]+)$/; if ($1 && $list[$count - 1][15] == 0) { $filename = $1; } my $hlink_path = $hlink; if ($query) { if (!string_contains($hlink_path, '/') && $list[$count - 1][15] == 0) { $hlink_path = undef; } $hlink_path =~ s/\/\Q$filename\E$//; } my $type = $list[$count - 1][14]; $type =~ s/\//\-/g; my $img = "images/icons/mime/$type.png"; unless (-e $request_uri{'module'} . '/' . $img) { $img = "images/icons/mime/unknown.png"; } my $actions = "<a class='action-link' href='javascript:void(0)' onclick='renameDialog(\"$hlink\")' title='$text{'rename'}' data-container='body'>$rename_icon</a>"; my $href; my $is_archive = 0; my $is_file = 1; my $is_gpg = 0; my $is_img = 0; if ($list[$count - 1][15] == 1) { $is_file = 0; $href = "index.cgi?path=" . &urlize("$path/$link"); } else { my ($fname, $fpath, $fsuffix) = fileparse($list[$count - 1][0]); if ($base ne '/') { $fpath =~ s/^\Q$base\E//g; } $href = "download.cgi?file=" . &urlize($link) . "&path=" . &urlize($fpath); if ($0 =~ /search.cgi/) { $actions = "$actions<a class='action-link' " . "href='index.cgi?path=" . &urlize($fpath) . "' " . "title='$text{'goto_folder'}'>$goto_icon</a>"; } if ($type =~ /text-/ || $type =~ /svg\+xml/ || exists($allowed_for_edit{$type})) { $actions = "$actions<a class='action-link' href='edit_file.cgi?file=" . &urlize($link) . "&path=" . &urlize($path) . "' title='$text{'edit'}' data-container='body'>$edit_icon</a>"; } my $type_archive = $type; if ($type =~ /^image/) { $is_img = 1; } if ($type =~ /application-pgp-encrypted/) { my $link_gpg = $link; $link_gpg =~ s/\.(gpg|pgp)$//; $type_archive = mimetype($link_gpg); $is_gpg = 1; } if (($type_archive =~ /application-zip/ && has_command('unzip')) || ($type_archive =~ /application-x-7z-compressed/ && has_command('7z')) || ($type_archive =~ /application-x-rar|application-vnd\.rar/ && (has_command('unrar') || has_command('unar'))) || ($type_archive =~ /application-x-rpm/ && has_command('rpm2cpio') && has_command('cpio')) || ($type_archive =~ /application-x-deb/ && has_command('dpkg')) || ( ($type_archive =~ /x-compressed-tar/ || $type_archive =~ /-x-tar/ || ($type_archive =~ /-x-bzip/ && has_command('bzip2')) || ($type_archive =~ /-gzip/ && (has_command('gzip') || has_command('gunzip'))) || ($type_archive =~ /-x-xz/ && has_command('xz')) ) && has_command('tar'))) { $is_archive = 1; $actions = "$actions <a class='action-link' href='extract.cgi?path=" . &urlize($path) . "&file=" . &urlize($link) . "' title='$text{'extract_archive'}' data-container='body'>$extract_icon</a> "; } } my @row_data = ("<a href='$href' data-filemin-link=\"$hlink\"" . ($query ? " data-filemin-flink=\"$hlink_path\"" : undef) . "><img src=\"$img\"></a>", "<a href=\"$href\" data-filemin-link=\"$hlink\"" . ($query ? " data-filemin-flink=\"$hlink_path\"" : undef) . ">$vlink</a>"); my @td_tags = (undef, 'class="col-icon"', 'class="col-name" data-xarchive="' . $is_archive . '" data-xfile="' . $is_file . '" data-gpg="' . $is_gpg . '" data-img="' . $is_img . '" data-order="' . ($is_file ? 1 : 0) . html_escape($filename) . '"'); if ($userconfig{'columns'} =~ /type/) { push(@row_data, $type); push(@td_tags, 'class="col-type"'); } push @row_data, $actions; push(@td_tags, 'class="col-actions"'); if ($userconfig{'columns'} =~ /size/) { my $size = &theme_nice_size_local($list[$count - 1][8]); push @row_data, ( "<span data-toggle=\"tooltip\" data-html=\"true\" data-title=\"$text{'theme_xhred_filemanager_global_size_in_bytes'}<br>@{[nice_number($list[$count - 1][8])]}\">" . $size . "</span>"); push(@td_tags, undef); } if ($userconfig{'columns'} =~ /owner_user/) { my $user; my $group; if (supports_users()) { my $uid = getpwuid($list[$count - 1][5]); my $gid = getgrgid($list[$count - 1][6]); $user = $uid ? $uid : $list[$count - 1][5]; $group = $gid ? $gid : $list[$count - 1][6]; } else { $user = $list[$count - 1][5]; $group = $list[$count - 1][6]; } push @row_data, ( "<span data-toggle=\"tooltip\" data-html=\"true\" data-title=\"$text{'filemanager_global_user_group_id'}<br>$list[$count - 1][5]:$list[$count - 1][6]\">" . $user . ':' . $group . "</span>"); push(@td_tags, 'class="col-ownership"'); } if ($userconfig{'columns'} =~ /permissions/) { my $permissions = sprintf("%04o", $list[$count - 1][3] & 07777); push @row_data, $permissions; push(@td_tags, 'class=col-permissions'); } if (get_attr_status() && $userconfig{'columns'} =~ /attributes/) { push @row_data, $list[$count - 1][18]; push(@td_tags, 'class="col-attrs"'); } if (get_selinux_status() && $userconfig{'columns'} =~ /selinux/) { push @row_data, $list[$count - 1][17]; push(@td_tags, 'class="col-selinux"'); } if ($userconfig{'columns'} =~ /last_mod_time/) { my $access_time = POSIX::strftime('%Y/%m/%d - %T', localtime($list[$count - 1][9])); my $mod_time = POSIX::strftime('%Y/%m/%d - %T', localtime($list[$count - 1][10])); my $change_time = POSIX::strftime('%Y/%m/%d - %T', localtime($list[$count - 1][11])); push @row_data, ( "<span data-toggle=\"tooltip\" data-html=\"true\" data-title=\"$text{'filemanager_global_access_change_time'}<br>$access_time<br>$change_time\">" . $mod_time . "</span>"); push(@td_tags, 'data-order="' . ($list[$count - 1][10]) . '" class="col-time"'); } $list_data{'rows'} .= &ui_checked_columns_row(\@row_data, \@td_tags, "name", $vlink); } $list_data{'form'} .= &ui_hidden("path", $path), "\n"; $list_data{'form'} .= '</form>'; $list_data{'success'} = (length $in{'success'} ? $in{'success'} : undef); $list_data{'error'} = (length $in{'error'} ? utf8_decode($in{'error'}) : undef); $list_data{'error_fatal'} = (length $in{'error_fatal'} ? $in{'error_fatal'} : undef); $list_data{'output'} = (length $in{'output'} ? $in{'output'} : undef); $list_data{'page_requested'} = $page; $list_data{'pagination_requested'} = $in{'paginate'}; $list_data{'totals'} = $totals; $list_data{'searched'} = $query ? 1 : 0; $list_data{'flush'} = test_all_items_query() ? 1 : 0; $list_data{'flush_reset'} = $in{'flush_reset'} ? 1 : 0; $list_data{'udata'} = { user => $remote_user_info[0], home => $remote_user_info[7], uid => $remote_user_info[2], guid => $remote_user_info[3], access => $access{'work_as_user'} }; print_json_local([\%list_data]); } sub get_tree { my ($p, $d, $e) = @_; my $df = int($d); my %r; my @r; my $wanted = sub { my $td = $File::Find::name; if (-d $td && !-l $td) { $td =~ s|^\Q$p\E/?||; if ($r{$td} || !$td) { return; } my ($pd, $cd) = $td =~ m|^ (.+) / ([^/]+) \z|x; my $pp = $p ne '/' ? $p : undef; my $c = $r{$td} = { key => html_escape("$pp/$td"), title => (defined($cd) ? html_escape($cd) : html_escape($td)) }; defined $pd ? (push @{ $r{$pd}{children} }, $c) : (push @r, $c); } }; my $preprocess = sub { my $td = $File::Find::name; my $d = $td =~ tr[/][]; if ($e && $p eq "/" && $d == 1) { if ($td =~ /^\/(cdrom|dev|lib|lost\+found|mnt|proc|run|snaps|sys|tmp|.trash)/i) { return; } } my $dd = ($df > 0 ? ($df + 1) : 0); if ($dd) { if ($d < $dd) { return sort {"\L$a" cmp "\L$b"} @_; } return; } sort {"\L$a" cmp "\L$b"} @_; }; find( { wanted => $wanted, preprocess => $preprocess }, $p); return \@r; } sub paster { my ($c, $f, $s, $d, $r, $m) = @_; my $x; my $j = $c . ($f =~ m/^\// ? undef : '/') . $f; if (!$r && -f $j ne -d $j) { for (my $t = 1;; $t += 1) { my ($jn, $je) = $j =~ /(.*)\.(.*)/; if (!-e ($jn . '(' . $t . ')' . ".$je") && (!-e ($j . '(' . $t . ')'))) { $x = $t; last; } } } $s =~ s/\/\//\//g; if ($m && -d $j && $j =~ /^\Q$s\E/) { set_response('merr'); return; } if (-d $j) { $j = $j . (!$x ? '' : '(' . $x . ')'); } else { my ($jn, $je) = $j =~ /(.*)\.(.*)/; if ($je) { $j = $jn . (!$x ? '' : '(' . $x . ')') . ".$je"; } else { $j = $j . (!$x ? '' : '(' . $x . ')'); } } my ($o, $e) = copy_source_dest($s, $j); if ($x) { set_response('cc'); } if ($m) { unlink_file($s); } return $e; } sub get_element_index { my ($arr, $elem) = @_; my $idx; for my $i (0 .. $#$arr) { if ($arr->[$i] eq $elem) { $idx = $i; last; } } return $idx; } sub get_gpg_version { my ($gpg) = @_; $gpg = "gpg" if (!$gpg); $gpg = quotemeta($gpg); $gpg = `$gpg --version`; $gpg =~ /(\*|\d+(\.\d+){0,2})/; return $1; } sub get_gpg_path { my $gnupg = 'gnupg'; my $gnupg_target = foreign_available($gnupg) ? $gnupg : get_product_name(); my %gpgconfig = foreign_config($gnupg_target); my $gpgpath = quotemeta($gpgconfig{'gpg'} || "gpg"); return $gpgpath; } sub switch_to_user { if (!supports_users()) { return undef; } my ($username) = @_; my @uinfo = getpwnam($username); if (@uinfo) { switch_to_unix_user(\@uinfo); } } sub is_linux { return $gconfig{'os_type'} =~ /-linux$/; } sub is_root { return ($base_remote_user eq 'root' ? 1 : 0); } sub get_env { my ($key) = @_; return $ENV{ uc($key) }; } sub set_env { my ($k, $v) = @_; $ENV{ uc($k) } = $v; } 1;