HEX
Server: Apache
System: Linux vps-cdc32557.vps.ovh.ca 5.15.0-156-generic #166-Ubuntu SMP Sat Aug 9 00:02:46 UTC 2025 x86_64
User: hanode (1017)
PHP: 7.4.33
Disabled: 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,pcntl_unshare,
Upload Files
File: //usr/share/usermin/authentic-theme/stats.pl
#!/usr/bin/perl

#
# Authentic Theme (https://github.com/authentic-theme/authentic-theme)
# Copyright Ilia Rostovtsev <ilia@virtualmin.com>
# Licensed under MIT (https://github.com/authentic-theme/authentic-theme/blob/master/LICENSE)
#
use strict;

use lib ("$ENV{'PERLLIB'}/vendor_perl");
use IO::Socket::INET;
use Net::WebSocket::Server;
use utf8;

our ($current_theme, $json);
require($ENV{'THEME_ROOT'} . "/stats-lib.pl");

# Get port number
my ($port) = @ARGV;

# Check if user is admin
if (!webmin_user_is_admin()) {
	remove_miniserv_websocket($port, $current_theme);
	error_stderr("WebSocket server cannot be accessed because the user is not a master administrator");
	exit(2);
}

# Clean up when socket is terminated
$SIG{'ALRM'} = sub {
	remove_miniserv_websocket($port, $current_theme);
	error_stderr("WebSocket server timeout waiting for a connection");
	exit(1);
	};
alarm(60);

# Log successful connection
error_stderr("WebSocket server is listening on port $port");

# Current stats within a period
my $stats_period;

# Create socket
my $server_socket = IO::Socket::INET->new(
	Listen    => 5,
	LocalPort => $port,
	ReuseAddr => 1,
	Proto     => 'tcp',
	LocalAddr => '127.0.0.1',
) or die "Failed to listen on port $port : $!";

# Start WebSocket server
Net::WebSocket::Server->new(
	listen     => $server_socket,
	tick_period => 1,
	on_tick => sub {
		my ($serv) = @_;
		# If asked to stop running, then shut down the server
		if ($serv->{'disable'}) {
			$serv->shutdown();
			return;
		}
		# Has any connection been unpaused?
		my $unpaused = grep {
				$serv->{'conns'}->{$_}->{'conn'}->{'pausing'} && 
				!$serv->{'conns'}->{$_}->{'conn'}->{'paused'} }
					keys %{$serv->{'conns'}};
		my $stats_history;
		# Return full stats for the given user who was unpaused
		# to make sure graphs are updated with most recent data
		$stats_history = get_stats_history(1) if ($unpaused);
		# Collect current stats and send them to all connected
		# clients unless paused for some client
		my $stats_now = get_stats_now();
		my $stats_now_graphs = $stats_now->{'graphs'};
		my $stats_now_json = $json->encode($stats_now);
		utf8::decode($stats_now_json);
		foreach my $conn_id (keys %{$serv->{'conns'}}) {
			my $conn = $serv->{'conns'}->{$conn_id}->{'conn'};
			if ($conn->{'verified'} && !$conn->{'paused'}) {
				# Unpaused connection needs full stats
				if ($conn->{'pausing'}) {
					$conn->{'pausing'} = 0;
					my $stats_updated;
					# Merge stats from both disk data
					# and currently cached data
					if ($stats_history && $stats_period) {
						$stats_now->{'graphs'} =
							merge_stats($stats_history->{'graphs'},
										$stats_period);
						$stats_updated++;
					# If no cached data then use history
					} elsif ($stats_history) {
						$stats_now->{'graphs'} = $stats_history->{'graphs'};
						$stats_updated++;
					# If no history then use cached data
					} elsif ($stats_period) {
						$stats_updated++;
						$stats_now->{'graphs'} = $stats_period;
					}
					# If stats were updated then merge
					# them with latest (now) data
					if ($stats_updated) {
						$stats_now->{'graphs'} =
							merge_stats($stats_now->{'graphs'},
										$stats_now_graphs);
					}
					$stats_now_json = $json->encode($stats_now);
					utf8::decode($stats_now_json);
				}
				$conn->send_utf8($stats_now_json);
			}
		}
		# Cache stats to server
		if (!defined($stats_period)) {
			$stats_period = $stats_now_graphs;
		} else {
			$stats_period = merge_stats($stats_period, $stats_now_graphs);
		}
		# Save stats to history and reset cache
		if ($serv->{'ticked'}++ % 15 == 0) {
			save_stats_history($stats_period)
				if (get_stats_option('status', 1) != 2);
			undef($stats_period);
		}
		# If interval is set then sleep minus one
		# second because tick_period is one second
		if ($serv->{'interval'} > 1) {
			sleep($serv->{'interval'}-1);
		}
		# Release memory
		undef($stats_now);
		undef($stats_now_graphs);
		undef($stats_now_json);
		undef($stats_history);
	},
	on_connect => sub {
		my ($serv, $conn) = @_;
		error_stderr("WebSocket connection $conn->{'port'} opened");
		$serv->{'clients_connected'}++;
		alarm(0);
		# Set post-connect activity timeout
		$SIG{'ALRM'} = sub {
			error_stderr("WebSocket connection $conn->{'port'} is closed due to inactivity");
			$conn->disconnect();
		};
		alarm(30);
		# Set maximum send size
		$conn->max_send_size(9216 * 1024); # Max 9 MiB to accomodate 24h of data
		# Handle connection events
		$conn->on(
			utf8 => sub {
				# Reset inactivity timer
				alarm(0);
				# Decode JSON message
				my ($conn, $msg) = @_;
				utf8::encode($msg) if (utf8::is_utf8($msg));
				my $data = $json->decode($msg);
				# Connection permission test unless already verified
				if (!$conn->{'verified'}) {
					my $user = verify_session_id($data->{'session'});
					if ($user && webmin_user_is_admin()) {
						# Set connection as verified and continue
						error_stderr("WebSocket connection for user $user is granted");
						$conn->{'verified'} = 1;
					} else {
						# Deny connection and disconnect
						error_stderr("WebSocket connection for user $user was denied");
						$conn->disconnect();
						return;
					}
				}
				# Update connection variables
				$conn->{'pausing'} = $conn->{'paused'} // 0;
				$conn->{'paused'} = $data->{'paused'} // 0;
				# Update WebSocket server variables
				$serv->{'interval'} = $data->{'interval'} // 1;
				$serv->{'disable'} = $data->{'disable'} // 0;
				$serv->{'shutdown'} = $data->{'shutdown'} // 0;
			},
			disconnect => sub {
				my ($conn) = @_;
				$serv->{'clients_connected'}--;
				error_stderr("WebSocket connection $conn->{'port'} closed");
				# If shutdown requested and no clients connected
				# then exit the server
				if ($serv->{'shutdown'} && $serv->{'clients_connected'} == 0) {
					error_stderr("WebSocket server is shutting down on last client disconnect");
					$serv->shutdown();
				}
			}
		);
	},
	on_shutdown => sub {
		# Shutdown the server and clean up
		my ($serv) = @_;
		error_stderr("WebSocket server has gracefully shut down");
		remove_miniserv_websocket($port, $current_theme);
		cleanup_miniserv_websockets([$port], $current_theme);
		exit(0);
	},
)->start;
error_stderr("WebSocket server failed");
remove_miniserv_websocket($port, $current_theme);
cleanup_miniserv_websockets([$port], $current_theme);

1;