Server IP : 195.201.23.43 / Your IP : 3.146.176.29 Web Server : Apache System : Linux webserver2.vercom.be 5.4.0-192-generic #212-Ubuntu SMP Fri Jul 5 09:47:39 UTC 2024 x86_64 User : kdecoratie ( 1041) PHP Version : 7.1.33-63+ubuntu20.04.1+deb.sury.org+1 Disable Function : pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals, MySQL : OFF | cURL : ON | WGET : ON | Perl : ON | Python : OFF | Sudo : ON | Pkexec : ON Directory : /usr/share/usermin/xterm/ |
Upload File : |
#!/usr/bin/perl # Start a websocket server connected to a shell use lib ("$ENV{'PERLLIB'}/vendor_perl"); use Net::WebSocket::Server; use utf8; require './xterm-lib.pl'; my ($port, $user, $dir) = @ARGV; # Switch to the user we're running as my @uinfo = getpwnam($user); my ($uid, $gid); if ($user ne "root" && !$<) { if (!@uinfo) { &remove_miniserv_websocket($port); die "User $user does not exist!"; } $uid = $uinfo[2]; $gid = $uinfo[3]; } else { $uid = $gid = 0; } # Run the user's shell in a sub-process &foreign_require("proc"); &clean_environment(); # Set locale my $lang = $config{'locale'}; if ($lang) { my @opts = ('LC_ALL', 'LANG', 'LANGUAGE'); $lang = 'en_US.UTF-8' if ($lang == 1); foreach my $opt (@opts) { $ENV{$opt} = &trim($lang); } } # Set terminal $ENV{'USER'} = $user; my $config_xterm = $config{'xterm'}; $config_xterm = 'xterm-256color' if (!$config_xterm); $config_xterm =~ s/\+/-/; $ENV{'TERM'} = $config_xterm; $ENV{'HOME'} = $uinfo[7]; chdir($dir || $uinfo[7] || "/"); my $shellcmd = $uinfo[8]; my $shellname = $shellcmd; $shellname =~ s/^.*\///; my $shellexec = $shellcmd; my $shelllogin = "-".$shellname; # Check for initialization file if ($config{'rcfile'} ne '0') { # Load shell init default file from module root directory or custom file my $rcdir = "$module_root_directory/rc"; my $rcfile = $config{'rcfile'} eq '1' ? "$rcdir/.".$shellname."rc" : $config{'rcfile'}; if ($rcfile =~ /^\~\//) { $rcfile =~ s/^\~\///; $rcfile = "$uinfo[7]/$rcfile"; } if (-r $rcfile) { if ($shellname eq 'bash') { # Bash $shellexec = "$shellcmd --rcfile $rcfile"; } elsif ($shellname eq 'zsh') { # Zsh $ENV{'ZDOTDIR'} = $rcdir; } # Cannot use login shell while passing other parameters, # and it is not necessary as we already add init files $shelllogin = undef; } } my ($shellfh, $pid) = &proc::pty_process_exec($shellexec, $uid, $gid, $shelllogin); &reset_environment(); my $shcmd = "'$shellexec".($shelllogin ? " $shelllogin" : "")."'"; if (!$pid) { &remove_miniserv_websocket($port); die "Failed to run shell with $shcmd\n"; } else { print STDERR "Running shell $shcmd with pid $pid\n"; } # Detach from controlling terminal if (fork()) { exit(0); } untie(*STDIN); close(STDIN); # Clean up when socket is terminated $SIG{'ALRM'} = sub { &remove_miniserv_websocket($port); die "timeout waiting for connection"; }; alarm(60); print STDERR "listening on port $port\n"; my ($wsconn, $shellbuf); Net::WebSocket::Server->new( listen => $port, on_connect => sub { my ($serv, $conn) = @_; print STDERR "got websockets connection\n"; if ($wsconn) { print STDERR "Unexpected second connection to the same port\n"; $conn->disconnect(); return; } $wsconn = $conn; alarm(0); $conn->on( handshake => sub { # Is the key valid for this Webmin session? my ($conn, $handshake) = @_; my $key = $handshake->req->fields->{'sec-websocket-key'}; my $dsess = &encode_base64($main::session_id); $key =~ s/\s//g; $dsess =~ s/\s//g; if (!$key || !$dsess || $key ne $dsess) { print STDERR "Key $key does not match session ID $dsess\n"; $conn->disconnect(); } }, ready => sub { my ($conn) = @_; $conn->send_binary($shellbuf) if ($shellbuf); }, utf8 => sub { my ($conn, $msg) = @_; utf8::encode($msg) if (utf8::is_utf8($msg)); # Check for resize escape sequence explicitly if ($msg =~ /^\\033\[8;\((\d+)\);\((\d+)\)t$/) { my ($rows, $cols) = ($1, $2); print STDERR "got resize to $rows $cols\n"; eval { $shellfh->set_winsize($rows, $cols); }; # If failed make ioctl directly (TIOCSWINSZ) # https://manpages.ubuntu.com/manpages/man2/ioctl_list.2.html if ($@) { ioctl($shellfh, 0x00005414, pack("s2", $rows, $cols)); } kill('WINCH', $pid); return; } if (!syswrite($shellfh, $msg, length($msg))) { print STDERR "write to shell failed : $!\n"; &remove_miniserv_websocket($port); exit(1); } }, disconnect => sub { print STDERR "websocket connection closed\n"; &remove_miniserv_websocket($port); kill('KILL', $pid) if ($pid); exit(0); } ); }, watch_readable => [ $shellfh => sub { # Got something from the shell my $buf; my $ok = sysread($shellfh, $buf, 1024); if ($ok <= 0) { print STDERR "end of output from shell\n"; &remove_miniserv_websocket($port); exit(0); } if ($wsconn) { $wsconn->send_binary($buf); } else { $shellbuf .= $buf; } }, ], )->start; print STDERR "exited websockets server\n"; &remove_miniserv_websocket($port); &cleanup_miniserv_websockets([$port]);Private