Files
webmin/nftables/nftables-lib.pl
2026-05-17 22:25:03 -05:00

4080 lines
107 KiB
Perl

# nftables-lib.pl
# Functions for reading and writing nftables rules
BEGIN { push(@INC, ".."); }; ## no critic
use WebminCore;
use strict;
use warnings;
our (%config, %access, $module_config_directory, $module_var_directory,
$module_root_directory);
our ($last_config_change_flag, $last_restart_time_flag);
init_config();
%access = get_module_acl();
$last_config_change_flag = $module_var_directory."/config-flag";
$last_restart_time_flag = $module_var_directory."/restart-flag";
# check_acl(action)
# Returns true if the current Webmin user can perform an action
sub check_acl
{
my ($action) = @_;
return $access{$action} ? 1 : 0;
}
# assert_acl(action)
# Fails if the current Webmin user cannot perform an action
sub assert_acl
{
my ($action) = @_;
check_acl($action) || error(text('acl_ecannot'));
}
# check_quick_acl([quick-action])
# Returns true if the current user can use a quick action
sub check_quick_acl
{
my ($action) = @_;
return 0 if (!check_acl('quick'));
return 1 if (!$action);
my $key = "quick_".$action;
return !defined($access{$key}) || $access{$key} ? 1 : 0;
}
# assert_quick_acl([quick-action])
# Fails if the current user cannot use a quick action
sub assert_quick_acl
{
my ($action) = @_;
check_quick_acl($action) || error(text('acl_ecannot'));
}
# table_acl_name(&table)
# Returns the ACL token for a table
sub table_acl_name
{
my ($table) = @_;
return ($table->{'family'} || '').":".($table->{'name'} || '');
}
# check_table_acl(&table)
# Returns true if the current Webmin user can manage a table
sub check_table_acl
{
my ($table) = @_;
return 0 if (!$table);
my $tables = defined($access{'tables'}) ? $access{'tables'} : '*';
return 1 if ($tables eq '*');
my $name = table_acl_name($table);
my @tokens = grep { $_ ne '' } split(/\s+/, $tables);
if (@tokens && $tokens[0] eq '!') {
my %deny = map { $_ => 1 } @tokens[1 .. $#tokens];
return !$deny{$name};
}
my %allow = map { $_ => 1 } @tokens;
return $allow{$name} ? 1 : 0;
}
# assert_table_acl(&table)
# Fails if the current Webmin user cannot manage a table
sub assert_table_acl
{
my ($table) = @_;
check_table_acl($table) ||
error(text('acl_etable', html_escape(nft_table_spec($table))));
}
# check_unrestricted_table_acl()
# Returns true if the current Webmin user can manage every saved table
sub check_unrestricted_table_acl
{
my $tables = defined($access{'tables'}) ? $access{'tables'} : '*';
return $tables eq '*';
}
# check_manual_acl()
# Returns true if the current user can edit the full saved rules file
sub check_manual_acl
{
return check_acl('manual') && check_unrestricted_table_acl();
}
# assert_manual_acl()
# Fails if the current user cannot edit the full saved rules file
sub assert_manual_acl
{
check_acl('manual') || error(text('acl_ecannot'));
check_unrestricted_table_acl() || error(text('manual_etables'));
}
# restart_button()
# Returns HTML for the header apply button
sub restart_button
{
return "" if (!check_acl('apply'));
my @tables = get_nftables_save();
return "" if (!@tables);
my $args = "redir=".urlize(this_url());
my $needs = needs_config_restart();
my $apply = text('index_apply_changes');
my $label = $needs ? "<b>$apply</b>" : $apply;
my $url = "restart.cgi?$args";
$url .= "&newconfig=1" if ($needs);
return ui_link($url, $label);
}
# this_url()
# Returns the URL in the nftables module for the current script
sub this_url
{
my $url = $ENV{'SCRIPT_NAME'} || "";
my $query = $ENV{'QUERY_STRING'} || "";
$url .= "?$query" if ($query ne "");
return $url;
}
# update_last_config_change()
# Updates the flag file indicating when the saved config was changed
sub update_last_config_change
{
open_lock_tempfile(my $fh, ">$last_config_change_flag", 0, 1);
close_tempfile($fh);
}
# restart_last_restart_time()
# Updates the flag file indicating when the saved config was applied
sub restart_last_restart_time
{
open_lock_tempfile(my $fh, ">$last_restart_time_flag", 0, 1);
close_tempfile($fh);
}
# needs_config_restart()
# Returns 1 if saved config changes still need to be applied
sub needs_config_restart
{
my @cst = stat($last_config_change_flag);
my @rst = stat($last_restart_time_flag);
return 0 if (!@cst);
return 1 if (!@rst);
return $cst[9] > $rst[9] ? 1 : 0;
}
# get_nft_command()
# Returns the configured nft command path, or finds it in PATH
sub get_nft_command
{
my $cmd = $config{'nft_cmd'} || "nft";
return has_command($cmd);
}
# nft_version_text()
# Returns a friendly nftables version string for page subtitles
sub nft_version_text
{
my $cmd = get_nft_command();
return if (!$cmd);
my $out = backquote_command(quotemeta($cmd)." --version 2>&1");
return if ($? || !$out);
$out =~ s/\r?\n.*$//s;
$out =~ s/^\s+|\s+$//g;
if ($out =~ /^nftables\s+v?(\S+)(?:\s+(.*))?$/i) {
my $details = $2 || "";
$details =~ s/^\s+|\s+$//g;
return text('index_version', $1.($details ne "" ? " ".$details : ""));
}
return $out;
}
# check_nftables()
# Returns an error message if nftables is not installed, undef if all is OK
sub check_nftables
{
return if (get_nft_command());
return text('index_ecommand', "<tt>nft</tt>");
}
# nftables_rules_file()
# Returns the Webmin-managed nftables rules file
sub nftables_rules_file
{
return "$module_config_directory/rules.conf";
}
# nftables_boot_action()
# Returns the init action name for applying nftables rules at boot
sub nftables_boot_action
{
return "webmin-nftables";
}
# nftables_boot_wrapper()
# Returns the generated wrapper used by the boot action
sub nftables_boot_wrapper
{
return "$module_config_directory/apply-boot.pl";
}
# nftables_started_at_boot()
# Returns true if Webmin-managed nftables rules are enabled at boot
sub nftables_started_at_boot
{
return 0 if (!foreign_check("init"));
foreign_require("init", "init-lib.pl");
return init::action_status(nftables_boot_action()) == 2 ? 1 : 0;
}
# create_nftables_init()
# Creates or enables the boot action for Webmin-managed nftables rules
sub create_nftables_init
{
foreign_require("init", "init-lib.pl");
chmod(0755, "$module_root_directory/apply-boot.pl");
create_wrapper(nftables_boot_wrapper(), "nftables", "apply-boot.pl");
my $action = nftables_boot_action();
{
no warnings 'once';
if (($init::init_mode || "") eq "systemd") {
my $unit = init::action_unit($action);
my $unit_file = init::get_systemd_root($unit)."/".$unit;
if (-r $unit_file) {
init::disable_at_boot($action);
init::delete_systemd_service($unit);
}
}
}
init::enable_at_boot(
$action,
"Load Webmin nftables rules",
nftables_boot_wrapper(),
undef, undef,
{
'exit' => 1,
'opts' => {
'after' => 'local-fs.target systemd-modules-load.service',
'before' => 'network-pre.target network.target',
'wants' => 'network-pre.target',
}
});
}
# disable_nftables_init()
# Disables the boot action for Webmin-managed nftables rules
sub disable_nftables_init
{
foreign_require("init", "init-lib.pl");
my $action = nftables_boot_action();
init::disable_at_boot($action);
{
no warnings 'once';
if (($init::init_mode || "") eq "systemd") {
init::delete_systemd_service(init::action_unit($action));
}
}
unlink_file(nftables_boot_wrapper());
}
# get_nftables_config_files()
# Returns files that can be manually edited by this module
sub get_nftables_config_files
{
my @files;
push(@files, nftables_rules_file());
foreach my $sysfile ("/etc/nftables.conf", "/etc/sysconfig/nftables.conf") {
push(@files, $sysfile) if (-f $sysfile);
}
if (-d "/etc/nftables") {
opendir(my $dir, "/etc/nftables");
if ($dir) {
foreach my $name (sort readdir($dir)) {
next if ($name =~ /^\./);
next if ($name !~ /\.(?:nft|conf)$/);
my $path = "/etc/nftables/$name";
push(@files, $path) if (-f $path);
}
closedir($dir);
}
}
my %seen;
return grep { !$seen{$_}++ } @files;
}
# list_foreign_firewall_modules()
# Returns other configured Webmin firewall modules that may manage rules
sub list_foreign_firewall_modules
{
my @mods = qw(firewalld firewall firewall6 shorewall shorewall6 csf);
my @rv;
foreach my $mod (@mods) {
next if (!foreign_check($mod));
my $installed = eval { foreign_installed($mod, 1) };
next if ($@ || $installed != 2);
my %minfo = get_module_info($mod);
push(@rv, {
'module' => $mod,
'desc' => $minfo{'desc'} } );
}
return @rv;
}
# validate_nftables_text(text)
# Returns an error if nft rejects the supplied ruleset text
sub validate_nftables_text
{
my ($text) = @_;
my $cmd = get_nft_command();
return text('index_ecommand', "<tt>nft</tt>") if (!$cmd);
my $tmp = tempname();
open_tempfile(my $fh, ">$tmp");
print_tempfile($fh, $text);
close_tempfile($fh);
my $out = backquote_logged("$cmd -c -f $tmp 2>&1");
unlink_file($tmp);
return $? ? "<pre>$out</pre>" : undef;
}
# get_nftables_save([file])
# Returns a list of tables and their chains/rules
sub get_nftables_save
{
my ($file) = @_;
if (!$file) {
$file = nftables_rules_file();
}
return () if (!$file);
return () if ($file !~ /\|\s*$/ && !-r $file);
my @rv;
my $table;
my $chain;
my $set;
my $set_depth = 0;
my $set_elem_open = 0;
my $set_elem_buf = '';
my $lnum = 0;
my $content;
my $fh;
my $is_pipe = $file =~ /\|\s*$/;
if ($is_pipe) {
(my $pipe_cmd = $file) =~ s/\|\s*$//;
open($fh, '-|', $pipe_cmd);
}
else {
lock_file($file);
open($fh, '<', $file);
}
$content = do { local $/; <$fh> };
close($fh);
unlock_file($file) if (!$is_pipe);
my @lines = split /\r?\n/, $content;
for (my $i = 0 ; $i < @lines ; $i++) {
my $line = $lines[$i];
$lnum++;
$line =~ s/#.*$//; # Ignore comments for now
if ($set) {
my $sline = $line;
$sline =~ s/^\s+//;
$sline =~ s/\s+$//;
if ($set_elem_open) {
if ($sline =~ /(.*)\}/) {
$set_elem_buf .= " ".$1;
$set_elem_open = 0;
$set_elem_buf =~ s/;\s*$//;
$set->{'elements'} =
parse_set_elements_string($set_elem_buf);
$set_elem_buf = '';
}
else {
$set_elem_buf .= " ".$sline if ($sline ne '');
}
}
else {
if ($sline =~ /^type\s+(\S+)\s*;?$/) {
$set->{'type'} = $1;
$set->{'type'} =~ s/;\s*$//;
}
elsif ($sline =~ /^flags\s+(.+?)\s*;?$/) {
$set->{'flags'} = $1;
}
elsif ($sline =~ /^elements\s*=\s*\{(.*)$/) {
my $rest = $1;
if ($rest =~ /(.*)\}/) {
my $content = $1;
$content =~ s/;\s*$//;
$set->{'elements'} =
parse_set_elements_string($content);
}
else {
$set_elem_open = 1;
$set_elem_buf = $rest;
}
}
elsif ($sline ne '' && $sline ne '}') {
push(@{$set->{'raw_lines'}}, $sline);
}
}
my $opens = () = $line =~ /\{/g;
my $closes = () = $line =~ /\}/g;
$set_depth += $opens - $closes;
if ($set_depth <= 0) {
$set = undef;
$set_depth = 0;
$set_elem_open = 0;
$set_elem_buf = '';
}
next;
}
if ($line =~ /^table\s+(\S+)\s+(\S+)\s+\{/) {
# Start of a table
$table = {
'name' => $2,
'family' => $1,
'line' => $lnum,
'rules' => [ ],
'chains' => {},
'sets' => {}
};
push(@rv, $table);
$chain = undef;
}
elsif ($line =~ /^\s*flags\s+(.+?)\s*;?$/ && $table && !$chain) {
$table->{'flags'} = $1;
}
elsif ($line =~ /^\s*set\s+(\S+)\s+\{/) {
# Start of a set
if ($table) {
my $setname = $1;
$set = {
'name' => $setname,
'line' => $lnum,
'elements' => [ ],
'raw_lines' => [ ],
};
$table->{'sets'}->{$setname} = $set;
$set_depth = () = $line =~ /\{/g;
$set_depth -= () = $line =~ /\}/g;
$set_elem_open = 0;
$set_elem_buf = '';
}
}
elsif ($line =~ /^\s*chain\s+(\S+)\s+\{/) {
# Start of a chain
if ($table) {
$chain = $1;
$table->{'chains'}->{$chain} = {};
# Look at next line for chain definition
if ($lines[$i + 1] =~
/^\s*type\s+(\S+)\s+hook\s+(\S+)\s+priority\s+(.+?);\s+policy\s+(\S+);/) {
$table->{'chains'}->{$chain}->{'type'} = $1;
$table->{'chains'}->{$chain}->{'hook'} = $2;
$table->{'chains'}->{$chain}->{'priority'} = $3;
$table->{'chains'}->{$chain}->{'policy'} = $4;
$i++; # Skip next line
}
}
}
elsif ($line =~ /^\s*(.*?)$/ && $table && $chain && $1 ne "}") {
# A rule
my $rule_str = $1;
if ($rule_str =~ /\S/) {
my $rule = {
'text' => $rule_str,
'chain' => $chain,
'index' => scalar(@{$table->{'rules'}}),
'line' => $lnum
};
my $parsed = parse_rule_text($rule_str);
if ($parsed) {
foreach my $k (keys %$parsed) {
$rule->{$k} = $parsed->{$k};
}
}
push(@{$table->{'rules'}}, $rule);
}
}
}
return @rv;
}
# get_active_nftables_save()
# Returns an array ref of tables from the active ruleset, and an optional error
sub get_active_nftables_save
{
my $cmd = get_nft_command();
return (undef, text('index_ecommand', "<tt>nft</tt>")) if (!$cmd);
my $out = backquote_command("$cmd list ruleset 2>&1");
return (undef, "<pre>$out</pre>") if ($?);
my $tmp = tempname();
open_tempfile(my $fh, ">$tmp");
print_tempfile($fh, $out);
close_tempfile($fh);
my @tables = get_nftables_save($tmp);
unlink_file($tmp);
return (\@tables, undef);
}
# tokenize_nft_rule(rule-text)
# Splits an nftables rule line into parser tokens
sub tokenize_nft_rule
{
my ($line) = @_;
my @tokens;
my $i = 0;
my $len = length($line);
while ($i < $len) {
my $ch = substr($line, $i, 1);
if ($ch =~ /\s/) {
$i++;
next;
}
if ($ch eq '"' || $ch eq "'") {
my $q = $ch;
my $j = $i + 1;
my $esc = 0;
while ($j < $len) {
my $c = substr($line, $j, 1);
if ($esc) {
$esc = 0;
}
elsif ($c eq "\\") {
$esc = 1;
}
elsif ($c eq $q) {
$j++;
last;
}
$j++;
}
push(@tokens, substr($line, $i, $j - $i));
$i = $j;
next;
}
if ($ch eq '{') {
my $j = $i + 1;
my $depth = 1;
while ($j < $len && $depth > 0) {
my $c = substr($line, $j, 1);
if ($c eq '{') {
$depth++;
}
elsif ($c eq '}') {
$depth--;
}
$j++;
}
push(@tokens, substr($line, $i, $j - $i));
$i = $j;
next;
}
my $j = $i;
while ($j < $len && substr($line, $j, 1) !~ /\s/) {
$j++;
}
push(@tokens, substr($line, $i, $j - $i));
$i = $j;
}
return @tokens;
}
# unquote_nft_string(string)
# Removes nftables-style quotes and escapes from a string token
sub unquote_nft_string
{
my ($s) = @_;
return $s if (!defined($s));
if ($s =~ /^"(.*)"$/s) {
$s = $1;
$s =~ s/\\(["\\])/$1/g;
}
elsif ($s =~ /^'(.*)'$/s) {
$s = $1;
$s =~ s/\\(['\\])/$1/g;
}
return $s;
}
# escape_nft_string(string)
# Escapes a string for use inside nftables double quotes
sub escape_nft_string
{
my ($s) = @_;
return "" if (!defined($s));
$s =~ s/\\/\\\\/g;
$s =~ s/"/\\"/g;
return $s;
}
# guess_addr_family(address, [fallback])
# Returns ip or ip6 based on an address-like value
sub guess_addr_family
{
my ($addr, $fallback) = @_;
return $fallback if ($fallback);
return "ip6" if (defined($addr) && $addr =~ /:/);
return "ip";
}
# validate_chain_base(type, hook, priority, policy)
# Returns true if a chain has a complete or empty base-chain definition
sub validate_chain_base
{
my ($type, $hook, $priority, $policy) = @_;
if (defined($type) || defined($hook) ||
defined($priority) || defined($policy)) {
return 0 if (!defined($type) || !defined($hook) ||
!defined($priority) || !defined($policy));
}
return 1;
}
# reindex_table_rules(&table)
# Updates rule index fields to match their array positions
sub reindex_table_rules
{
my ($table) = @_;
return
if (!$table ||
ref($table) ne 'HASH' ||
!$table->{'rules'} ||
ref($table->{'rules'}) ne 'ARRAY');
for (my $i = 0 ; $i < @{$table->{'rules'}} ; $i++) {
my $r = $table->{'rules'}->[$i];
$r->{'index'} = $i if ($r && ref($r) eq 'HASH');
}
return;
}
# find_input_chain(&table)
# Returns the best input chain for adding inbound quick rules
sub find_input_chain
{
my ($table) = @_;
return
if (!$table ||
ref($table) ne 'HASH' ||
!$table->{'chains'} ||
ref($table->{'chains'}) ne 'HASH');
# Quick IP rules must live in the selected table's input chain. A separate
# table cannot reliably allow traffic, because another input chain can still
# drop the packet later.
foreach my $c (sort keys %{$table->{'chains'}}) {
my $chain = $table->{'chains'}->{$c} || {};
return $c if ($c eq 'input' && ($chain->{'hook'} || '') eq 'input');
}
foreach my $c (sort keys %{$table->{'chains'}}) {
my $chain = $table->{'chains'}->{$c} || {};
return $c if (($chain->{'hook'} || '') eq 'input');
}
return $table->{'chains'}->{'input'} ? 'input' : undef;
}
# parse_ip_cidr(string)
# Returns address, nftables family and optional error for an IPv4/IPv6 CIDR
sub parse_ip_cidr
{
my ($ip) = @_;
$ip = "" if (!defined($ip));
$ip =~ s/^\s+//;
$ip =~ s/\s+$//;
return (undef, undef, text('quick_eip')) if ($ip eq '' || $ip =~ /\s/);
return (undef, undef, text('quick_eip')) if ($ip =~ tr/\/// > 1);
my $mask;
my $addr = $ip;
if ($addr =~ s/\/(\d+)$//) {
$mask = $1;
}
elsif ($addr =~ /\//) {
return (undef, undef, text('quick_eip'));
}
if (check_ipaddress($addr)) {
return (undef, undef, text('quick_eip'))
if (defined($mask) && $mask > 32);
return ($addr.(defined($mask) ? "/".$mask : ""), 'ip', undef);
}
if (check_ip6address($addr)) {
return (undef, undef, text('quick_eip'))
if (defined($mask) && $mask > 128);
return ($addr.(defined($mask) ? "/".$mask : ""), 'ip6', undef);
}
return (undef, undef, text('quick_eip'));
}
# quick_rule_type(&rule)
# Returns allow or block if this rule was created by the quick IP controls
sub quick_rule_type
{
my ($rule) = @_;
return if (!$rule || ref($rule) ne 'HASH');
return 'allow' if (($rule->{'comment'} || '') eq 'Webmin quick allow');
return 'block' if (($rule->{'comment'} || '') eq 'Webmin quick block');
return 'port' if (($rule->{'comment'} || '') eq 'Webmin quick port');
return 'service' if (($rule->{'comment'} || '') =~ /^Webmin quick service\b/);
return 'forward' if (($rule->{'comment'} || '') eq 'Webmin quick forward');
return;
}
# quick_rule_rank(type)
# Returns the insertion priority for generated quick rules
sub quick_rule_rank
{
my ($type) = @_;
return 0 if ($type && $type eq 'allow');
return 1 if ($type && $type eq 'block');
return 2 if ($type && ($type eq 'port' ||
$type eq 'service' ||
$type eq 'forward'));
return 9;
}
# insert_quick_rule(&table, chain, &rule, rank)
# Inserts a quick rule before normal rules but after lower ranked quick rules
sub insert_quick_rule
{
my ($table, $chain, $rule, $rank) = @_;
$table->{'rules'} ||= [ ];
my $insert = scalar(@{$table->{'rules'} || [ ]});
for (my $i = 0 ; $i < @{$table->{'rules'} || [ ]} ; $i++) {
my $r = $table->{'rules'}->[$i];
next if (!$r || ref($r) ne 'HASH');
next if (($r->{'chain'} || '') ne $chain);
my $rrank = quick_rule_rank(quick_rule_type($r));
if ($rrank > $rank) {
$insert = $i;
last;
}
}
splice(@{$table->{'rules'}}, $insert, 0, $rule);
reindex_table_rules($table);
return;
}
# table_supports_quick_l4(&table)
# Returns an error if quick TCP/UDP rules cannot be added to this table
sub table_supports_quick_l4
{
my ($table) = @_;
return text('quick_etable') if (!$table || ref($table) ne 'HASH');
return if (($table->{'family'} || '') =~ /^(inet|ip|ip6)$/);
return text('quick_efamily', nft_table_spec($table));
}
# parse_quick_port(port|range)
# Returns a validated port expression and optional error
sub parse_quick_port
{
my ($port) = @_;
$port = "" if (!defined($port));
$port =~ s/^\s+//;
$port =~ s/\s+$//;
return (undef, text('quick_eport')) if ($port eq '');
if ($port =~ /^(\d+)$/) {
my $p = $1;
return valid_profile_port_number($p)
? ($p, undef)
: (undef, text('quick_eport'));
}
if ($port =~ /^(\d+)-(\d+)$/) {
my ($from, $to) = ($1, $2);
return (undef, text('quick_eport'))
if (!valid_profile_port_number($from) ||
!valid_profile_port_number($to));
return (undef, text('quick_eportrange')) if ($from >= $to);
return ("$from-$to", undef);
}
return (undef, text('quick_eport'));
}
# normalize_quick_proto(proto)
# Returns a supported transport protocol for quick port operations
sub normalize_quick_proto
{
my ($proto) = @_;
$proto = lc($proto || '');
return $proto if ($proto eq 'tcp' || $proto eq 'udp');
return;
}
# port_interval(port|range)
# Returns numeric start and end values for a port expression
sub port_interval
{
my ($port) = @_;
return ($1, $1) if (defined($port) && $port =~ /^(\d+)$/);
return ($1, $2) if (defined($port) && $port =~ /^(\d+)-(\d+)$/);
return;
}
# port_expr_covers(existing, wanted)
# Returns true if one port expression covers another
sub port_expr_covers
{
my ($existing, $wanted) = @_;
return 1 if (defined($existing) && defined($wanted) && $existing eq $wanted);
my ($es, $ee) = port_interval($existing);
my ($ws, $we) = port_interval($wanted);
return defined($es) && defined($ws) && $es <= $ws && $ee >= $we;
}
# set_contains_port(&set, port|range)
# Returns true if an inet_service set covers a port expression
sub set_contains_port
{
my ($set, $port) = @_;
return 0 if (!$set || ref($set) ne 'HASH');
return 0 if (!$set->{'elements'} || ref($set->{'elements'}) ne 'ARRAY');
foreach my $e (@{$set->{'elements'}}) {
return 1 if (port_expr_covers($e, $port));
}
return 0;
}
# add_ports_to_set(&set, ports...)
# Adds ports to an inet_service set and returns true if it changed
sub add_ports_to_set
{
my ($set, @ports) = @_;
return 0 if (!$set || ref($set) ne 'HASH' || !@ports);
my @old = $set->{'elements'} && ref($set->{'elements'}) eq 'ARRAY'
? @{$set->{'elements'}}
: ( );
my @new = normalize_port_set_elements(@old, @ports);
return 0 if (join("\0", @old) eq join("\0", @new));
$set->{'elements'} = \@new;
if (grep { /-/ } @new) {
my $flags = $set->{'flags'} || '';
if ($flags !~ /(?:^|[,\s])interval(?:$|[,\s])/) {
$set->{'flags'} = $flags ? $flags.", interval" : "interval";
}
}
return 1;
}
# find_accept_port_set(&table, chain, proto)
# Finds a port set already accepted by an input rule for a protocol
sub find_accept_port_set
{
my ($table, $chain, $proto) = @_;
return if (!$table || ref($table) ne 'HASH');
return if (!$table->{'rules'} || ref($table->{'rules'}) ne 'ARRAY');
foreach my $r (@{$table->{'rules'}}) {
next if (!$r || ref($r) ne 'HASH');
next if (($r->{'chain'} || '') ne $chain);
next if (($r->{'action'} || '') ne 'accept');
next if (($r->{'proto'} || '') ne $proto);
my $setname = set_name_from_value($r->{'dport'});
next if (!$setname);
my $sets = $table->{'sets'} && ref($table->{'sets'}) eq 'HASH'
? $table->{'sets'}
: {};
my $set = $sets->{$setname};
next if (!$set || set_type_kind($set->{'type'}) ne 'port');
return $setname;
}
return;
}
# quick_accept_port_covered(&table, chain, proto, port|range)
# Returns true if an existing accept rule already covers a port
sub quick_accept_port_covered
{
my ($table, $chain, $proto, $port) = @_;
return 0 if (!$table || ref($table) ne 'HASH');
foreach my $r (@{$table->{'rules'} || [ ]}) {
next if (!$r || ref($r) ne 'HASH');
next if (($r->{'chain'} || '') ne $chain);
next if (($r->{'action'} || '') ne 'accept');
next if (($r->{'proto'} || '') ne $proto);
my $dport = $r->{'dport'};
next if (!defined($dport) || $dport eq '');
my $setname = set_name_from_value($dport);
if ($setname) {
my $sets = $table->{'sets'} && ref($table->{'sets'}) eq 'HASH'
? $table->{'sets'}
: {};
return 1
if (set_contains_port($sets->{$setname}, $port));
next;
}
return 1 if (port_expr_covers($dport, $port));
}
return 0;
}
# add_quick_accept_port(&table, port|range, proto, comment)
# Adds or merges an accepted destination-port rule. Returns changed and error.
sub add_quick_accept_port
{
my ($table, $port, $proto, $comment) = @_;
my $err = table_supports_quick_l4($table);
return (0, $err) if ($err);
$proto = normalize_quick_proto($proto);
return (0, text('quick_eproto')) if (!$proto);
($port, $err) = parse_quick_port($port);
return (0, $err) if ($err);
my $chain = find_input_chain($table);
return (0, text('quick_echain', nft_table_spec($table))) if (!$chain);
my $setname = find_accept_port_set($table, $chain, $proto);
if ($setname) {
my $changed = add_ports_to_set($table->{'sets'}->{$setname}, $port);
return ($changed, undef);
}
return (0, undef)
if (quick_accept_port_covered($table, $chain, $proto, $port));
my $rule = {
'chain' => $chain,
'proto' => $proto,
'dport' => $port,
'action' => 'accept',
'comment' => $comment || 'Webmin quick port',
};
$rule->{'text'} = format_rule_text($rule);
insert_quick_rule($table, $chain, $rule,
quick_rule_rank(quick_rule_type($rule)));
return (1, undef);
}
# add_quick_port_rule(&table, port|range, proto)
# Adds an accepted destination-port quick rule
sub add_quick_port_rule
{
my ($table, $port, $proto) = @_;
my ($changed, $err) =
add_quick_accept_port($table, $port, $proto, 'Webmin quick port');
return $err if ($err);
return text('quick_edup', $port) if (!$changed);
return;
}
# add_quick_ip_rule(&table, ip-cidr, action)
# Adds an allow or block source-address rule to the table's input chain
sub add_quick_ip_rule
{
my ($table, $ip, $action) = @_;
return text('quick_etable') if (!$table || ref($table) ne 'HASH');
$action = $action eq 'allow' ? 'allow' : $action eq 'block' ? 'block' : '';
return text('quick_eaction') if (!$action);
my ($source, $family, $err) = parse_ip_cidr($ip);
return $err if ($err);
if (($table->{'family'} || '') eq 'ip' && $family ne 'ip' ||
($table->{'family'} || '') eq 'ip6' && $family ne 'ip6' ||
($table->{'family'} || '') !~ /^(inet|ip|ip6)$/) {
return text('quick_efamily', nft_table_spec($table));
}
my $chain = find_input_chain($table);
return text('quick_echain', nft_table_spec($table)) if (!$chain);
foreach my $r (@{$table->{'rules'} || [ ]}) {
next if (!$r || ref($r) ne 'HASH');
next if (($r->{'chain'} || '') ne $chain);
next if (($r->{'saddr'} || '') ne $source);
next
if (($r->{'action'} || '') ne
($action eq 'allow' ? 'accept' : 'drop'));
next if (!quick_rule_type($r));
return text('quick_edup', $source);
}
my $rule = {
'chain' => $chain,
'saddr' => $source,
'saddr_family' => $family,
'action' => $action eq 'allow' ? 'accept' : 'drop',
'comment' => $action eq 'allow'
? 'Webmin quick allow'
: 'Webmin quick block',
};
$rule->{'text'} = format_rule_text($rule);
insert_quick_rule($table, $chain, $rule,
$action eq 'allow' ? quick_rule_rank('allow') : quick_rule_rank('block'));
return;
}
# builtin_quick_service_defs()
# Returns built-in service definitions used when /etc/services is unavailable
sub builtin_quick_service_defs
{
my @defs = (
[ 'ssh', [ [ 'tcp', '22' ] ] ],
[ 'http', [ [ 'tcp', '80' ] ] ],
[ 'https', [ [ 'tcp', '443' ] ] ],
[ 'dns', [ [ 'tcp', '53' ], [ 'udp', '53' ] ], [ 'domain' ] ],
[ 'smtp', [ [ 'tcp', '25' ] ] ],
[ 'submission', [ [ 'tcp', '587' ] ], [ 'msa' ] ],
[ 'smtps', [ [ 'tcp', '465' ] ], [ 'submissions' ] ],
[ 'imap', [ [ 'tcp', '143' ] ] ],
[ 'imaps', [ [ 'tcp', '993' ] ] ],
[ 'pop3', [ [ 'tcp', '110' ] ] ],
[ 'pop3s', [ [ 'tcp', '995' ] ] ],
[ 'ftp', [ [ 'tcp', '21' ] ] ],
[ 'ntp', [ [ 'udp', '123' ] ] ],
);
my @rv;
foreach my $d (@defs) {
my ($id, $ports, $aliases) = @$d;
my $svc = {
'id' => $id,
'aliases' => $aliases || [ ],
'ports' => {},
'source_ports' => {},
'protocols' => [ ],
};
foreach my $p (@$ports) {
push(@{$svc->{'ports'}->{$p->[0]}}, $p->[1]);
}
$svc->{'label'} = quick_service_label($svc);
push(@rv, $svc);
}
return @rv;
}
# read_etc_service_defs([services-file])
# Returns service definitions from /etc/services canonical names and aliases
sub read_etc_service_defs
{
my ($file) = @_;
$file ||= "/etc/services";
my %defs;
if (open(my $fh, "<", $file)) {
while(my $line = <$fh>) {
$line =~ s/#.*$//;
$line =~ s/^\s+//;
$line =~ s/\s+$//;
next if ($line eq '');
my ($name, $portproto, @aliases) = split(/\s+/, $line);
next if (!$name || !$portproto);
next if ($name !~ /^[A-Za-z0-9_.+-]+$/);
next if ($portproto !~ /^(\d+)\/(tcp|udp)$/i);
my ($port, $proto) = ($1, lc($2));
next if (!valid_profile_port_number($port));
@aliases = grep { /^[A-Za-z0-9_.+-]+$/ } @aliases;
$defs{$name} ||= {
'id' => $name,
'aliases' => [ ],
'ports' => {},
'source_ports' => {},
'protocols' => [ ],
};
push(@{$defs{$name}->{'ports'}->{$proto}}, $port);
push(@{$defs{$name}->{'aliases'}}, @aliases);
}
close($fh);
}
foreach my $id (keys %defs) {
my %aliases_seen;
$defs{$id}->{'aliases'} =
[ grep { !$aliases_seen{$_}++ } @{$defs{$id}->{'aliases'}} ];
foreach my $proto (keys %{$defs{$id}->{'ports'}}) {
my %seen;
$defs{$id}->{'ports'}->{$proto} =
[ normalize_port_set_elements(grep { !$seen{$_}++ }
@{$defs{$id}->{'ports'}->{$proto}}) ];
}
$defs{$id}->{'label'} = quick_service_label($defs{$id});
}
return values %defs;
}
# setup_quick_service_defs()
# Returns quick service definitions from the module's dynamic profile services
sub setup_quick_service_defs
{
my @rv;
foreach my $svc (setup_services()) {
my $id = $svc->{'id'};
next if (!$id);
my $def = {
'id' => $id,
'ports' => {},
'source_ports' => {},
'protocols' => [ ],
'rules' => [ @{$svc->{'rules'} || [ ]} ],
};
foreach my $rule (@{$def->{'rules'}}) {
if ($rule =~ /^(tcp|udp)\s+dport\s+(\S+)\s+accept$/) {
push(@{$def->{'ports'}->{$1}}, $2);
}
elsif ($rule =~ /^(tcp|udp)\s+sport\s+(\S+)\s+accept$/) {
push(@{$def->{'source_ports'}->{$1}}, $2);
}
}
$def->{'label'} = quick_service_label($def);
push(@rv, $def);
}
return @rv;
}
# quick_service_label(&service)
# Returns a service label with ports and protocols
sub quick_service_label
{
my ($svc) = @_;
my $name = $svc->{'id'} || "";
my @details;
foreach my $kind ('ports', 'source_ports') {
my $ports = $svc->{$kind} || {};
my %by_ports;
foreach my $proto (sort keys %$ports) {
my @ports = @{$ports->{$proto} || [ ]};
next if (!@ports);
push(@{$by_ports{join(", ", @ports)}}, uc($proto));
}
foreach my $plist (sort port_sort keys %by_ports) {
my $prefix = $kind eq 'source_ports' ? text('quick_source_ports')." " : "";
push(@details, $prefix.$plist." ".join("/", @{$by_ports{$plist}}));
}
}
if ($svc->{'protocols'} && @{$svc->{'protocols'}}) {
push(@details, map { uc($_) } @{$svc->{'protocols'}});
}
return @details ? $name." (".join("; ", @details).")" : $name;
}
# merge_quick_service(&defs, &service)
# Adds or replaces one quick service definition
sub merge_quick_service
{
my ($defs, $svc) = @_;
return if (!$defs || !$svc || ref($svc) ne 'HASH' || !$svc->{'id'});
$defs->{$svc->{'id'}} = $svc;
return;
}
# quick_services([services-file])
# Returns available service definitions for the quick service selector
sub quick_services
{
my ($services_file) = @_;
my %defs;
foreach my $svc (builtin_quick_service_defs()) {
merge_quick_service(\%defs, $svc);
}
foreach my $svc (read_etc_service_defs($services_file)) {
merge_quick_service(\%defs, $svc);
}
foreach my $svc (setup_quick_service_defs()) {
merge_quick_service(\%defs, $svc);
}
my @sorted = sort { lc($a->{'label'}) cmp lc($b->{'label'}) } values %defs;
return @sorted;
}
# service_search_text(&service)
# Returns searchable service names and aliases
sub service_search_text
{
my ($svc) = @_;
return lc(join(" ",
$svc->{'id'} || "",
$svc->{'label'} || "",
@{$svc->{'aliases'} || [ ]}
));
}
# search_quick_services(query, [limit], [services-file])
# Returns quick service definitions matching a short search string
sub search_quick_services
{
my ($query, $limit, $services_file) = @_;
$query = "" if (!defined($query));
$query =~ s/^\s+//;
$query =~ s/\s+$//;
return ( ) if ($query eq "");
$limit ||= 20;
$limit = 1 if ($limit < 1);
$limit = 50 if ($limit > 50);
my $q = lc($query);
my @ranked;
foreach my $svc (&quick_services($services_file)) {
my $id = lc($svc->{'id'} || "");
my $label = lc($svc->{'label'} || $svc->{'id'} || "");
my @aliases = map { lc($_) } @{$svc->{'aliases'} || [ ]};
my $search = service_search_text($svc);
my $rank;
if ($id eq $q) {
$rank = 0;
}
elsif (grep { $_ eq $q } @aliases) {
$rank = 1;
}
elsif ($id =~ /^\Q$q\E/) {
$rank = 2;
}
elsif (grep { /^\Q$q\E/ } @aliases) {
$rank = 3;
}
elsif ($label =~ /^\Q$q\E/) {
$rank = 4;
}
elsif ($id =~ /\Q$q\E/) {
$rank = 5;
}
elsif ($search =~ /\Q$q\E/) {
$rank = 6;
}
else {
next;
}
push(@ranked, [ $rank, $label, $svc ]);
}
@ranked = sort { $a->[0] <=> $b->[0] || $a->[1] cmp $b->[1] } @ranked;
my @rv;
foreach my $r (@ranked) {
push(@rv, $r->[2]);
last if (@rv >= $limit);
}
return @rv;
}
# quick_service_by_id(service-id, [services-file])
# Returns a quick service definition by ID
sub quick_service_by_id
{
my ($id, $services_file) = @_;
$id = "" if (!defined($id));
$id =~ s/^\s+//;
$id =~ s/\s+$//;
return if ($id !~ /^[A-Za-z0-9_.+-]+$/);
foreach my $svc (quick_services($services_file)) {
return $svc if ($svc->{'id'} eq $id);
foreach my $alias (@{$svc->{'aliases'} || [ ]}) {
return $svc if ($alias eq $id);
}
}
return;
}
# quick_service_rules(&service)
# Returns nftables accept rule texts for a quick service definition
sub quick_service_rules
{
my ($svc) = @_;
return if (!$svc || ref($svc) ne 'HASH');
return @{$svc->{'rules'}} if ($svc->{'rules'} && @{$svc->{'rules'}});
my @rules;
foreach my $proto (sort keys %{$svc->{'ports'} || {}}) {
foreach my $port (@{$svc->{'ports'}->{$proto} || [ ]}) {
push(@rules, "$proto dport $port accept");
}
}
foreach my $proto (sort keys %{$svc->{'source_ports'} || {}}) {
foreach my $port (@{$svc->{'source_ports'}->{$proto} || [ ]}) {
push(@rules, "$proto sport $port accept");
}
}
foreach my $proto (@{$svc->{'protocols'} || [ ]}) {
push(@rules, "meta l4proto $proto accept");
}
return @rules;
}
# quick_rule_text_compatible(&table, rule-text)
# Returns true if a rule text does not conflict with the table family
sub quick_rule_text_compatible
{
my ($table, $text) = @_;
my $family = $table->{'family'} || '';
return 0 if ($family eq 'ip' && $text =~ /(?:^|\s)ip6\s/);
return 0 if ($family eq 'ip6' && $text =~ /(?:^|\s)ip\s/);
return 1;
}
# quick_add_raw_input_rule(&table, chain, rule-text, comment)
# Adds a raw quick input rule if it does not already exist
sub quick_add_raw_input_rule
{
my ($table, $chain, $text, $comment) = @_;
$text =~ s/^\s+//;
$text =~ s/\s+$//;
return 0 if ($text eq '');
return 0 if (!quick_rule_text_compatible($table, $text));
my $out = quick_rule_with_comment($text, $comment);
foreach my $r (@{$table->{'rules'} || [ ]}) {
next if (!$r || ref($r) ne 'HASH');
next if (($r->{'chain'} || '') ne $chain);
return 0 if (($r->{'text'} || '') eq $out || ($r->{'text'} || '') eq $text);
}
my $rule = parse_rule_text($out);
$rule->{'chain'} = $chain;
$rule->{'text'} = $out;
insert_quick_rule($table, $chain, $rule,
quick_rule_rank(quick_rule_type($rule)));
return 1;
}
# quick_rule_with_comment(rule-text, comment)
# Appends a comment to a rule text if it does not already have one
sub quick_rule_with_comment
{
my ($text, $comment) = @_;
return $text if ($text =~ /(?:^|\s)comment\s+/);
my $c = escape_nft_string($comment || "");
return $c ne "" ? $text." comment \"".$c."\"" : $text;
}
# add_quick_service_rule(&table, service-id)
# Adds accept rules for a known service to the table's input chain
sub add_quick_service_rule
{
my ($table, $service_id) = @_;
my $err = table_supports_quick_l4($table);
return $err if ($err);
my $chain = find_input_chain($table);
return text('quick_echain', nft_table_spec($table)) if (!$chain);
my $svc = quick_service_by_id($service_id);
return text('quick_eservice') if (!$svc);
my @rules = quick_service_rules($svc);
return text('quick_eservice_empty', $service_id) if (!@rules);
my $changed = 0;
my $comment = "Webmin quick service: ".$svc->{'id'};
foreach my $text (@rules) {
if ($text =~ /^(tcp|udp)\s+dport\s+(\S+)\s+accept$/) {
my ($ok, $perr) =
add_quick_accept_port($table, $2, $1, $comment);
return $perr if ($perr);
$changed ||= $ok;
}
else {
my $ok = quick_add_raw_input_rule(
$table, $chain, $text, $comment);
$changed ||= $ok;
}
}
return $changed ? undef : text('quick_edup', $svc->{'label'});
}
# parse_quick_forward_addr(address, &table)
# Validates a port-forward destination address
sub parse_quick_forward_addr
{
my ($addr, $table) = @_;
$addr = "" if (!defined($addr));
$addr =~ s/^\s+//;
$addr =~ s/\s+$//;
return (undef, undef, undef) if ($addr eq '');
return (undef, undef, text('quick_eforward_addr')) if ($addr =~ m{/});
my $family;
if (check_ipaddress($addr)) {
$family = 'ip';
}
elsif (check_ip6address($addr)) {
$family = 'ip6';
}
else {
return (undef, undef, text('quick_eforward_addr'));
}
if (($table->{'family'} || '') eq 'ip' && $family ne 'ip' ||
($table->{'family'} || '') eq 'ip6' && $family ne 'ip6') {
return (undef, undef,
text('quick_eforward_family', nft_table_spec($table)));
}
return ($addr, $family, undef);
}
# find_base_chain(&table, hook, [type])
# Finds the best base chain for a hook and optional type
sub find_base_chain
{
my ($table, $hook, $type) = @_;
return if (!$table || !$table->{'chains'} || ref($table->{'chains'}) ne 'HASH');
foreach my $name (sort keys %{$table->{'chains'}}) {
my $chain = $table->{'chains'}->{$name} || {};
next if (($chain->{'hook'} || '') ne $hook);
next if (defined($type) && ($chain->{'type'} || '') ne $type);
return $name if ($name eq $hook);
}
foreach my $name (sort keys %{$table->{'chains'}}) {
my $chain = $table->{'chains'}->{$name} || {};
next if (($chain->{'hook'} || '') ne $hook);
next if (defined($type) && ($chain->{'type'} || '') ne $type);
return $name;
}
return;
}
# unique_chain_name(&table, base-name)
# Returns an unused chain name in a table
sub unique_chain_name
{
my ($table, $base) = @_;
$base ||= "chain";
my $name = $base;
my $i = 1;
while ($table->{'chains'}->{$name}) {
$name = $base."_".$i++;
}
return $name;
}
# ensure_prerouting_nat_chain(&table)
# Finds or creates a NAT prerouting chain for quick port forwards
sub ensure_prerouting_nat_chain
{
my ($table) = @_;
my $chain = find_base_chain($table, 'prerouting', 'nat');
return $chain if ($chain);
my $base = $table->{'chains'}->{'prerouting'} ? 'prerouting_nat' : 'prerouting';
$chain = unique_chain_name($table, $base);
$table->{'chains'}->{$chain} = {
'type' => 'nat',
'hook' => 'prerouting',
'priority' => '-100',
'policy' => 'accept',
};
return $chain;
}
# quick_text_rule_exists(&table, chain, rule-text)
# Returns true if exact rule text already exists in a chain
sub quick_text_rule_exists
{
my ($table, $chain, $text) = @_;
foreach my $r (@{$table->{'rules'} || [ ]}) {
next if (!$r || ref($r) ne 'HASH');
next if (($r->{'chain'} || '') ne $chain);
return 1 if (($r->{'text'} || '') eq $text);
}
return 0;
}
# format_forward_target(&table, addr, addr-family, port)
# Formats the dnat or redirect statement for a quick port forward
sub format_forward_target
{
my ($table, $addr, $addr_family, $port) = @_;
if ($addr) {
my $target = format_nat_target($addr, $port);
my $stmt = "dnat ";
$stmt .= $addr_family." " if (($table->{'family'} || '') eq 'inet');
return $stmt."to ".$target;
}
return "redirect to :".$port;
}
# parse_nat_target(target)
# Splits a redirect/dnat target into address and port parts
sub parse_nat_target
{
my ($target) = @_;
return (undef, undef) if (!defined($target) || $target eq '');
return (undef, $1) if ($target =~ /^:(.+)$/);
return ($1, $2) if ($target =~ /^\[([^\]]+)\]:(.+)$/);
return ($1, $2) if ($target =~ /^([^:]+):([^:]+)$/);
return ($target, undef);
}
# format_nat_target(address, port)
# Formats a redirect/dnat target from address and port parts
sub format_nat_target
{
my ($addr, $port) = @_;
$addr = undef if (defined($addr) && $addr eq '');
$port = undef if (defined($port) && $port eq '');
if (defined($addr)) {
my $target = $addr;
if (defined($port)) {
$target = $addr =~ /:/ ? "[".$addr."]:".$port : $addr.":".$port;
}
return $target;
}
return defined($port) ? ":".$port : "";
}
# format_nat_expr(&rule)
# Formats a redirect or dnat expression from a structured rule
sub format_nat_expr
{
my ($rule) = @_;
my $action = $rule->{'action'} || '';
return if ($action !~ /^(redirect|dnat)$/);
my $target = format_nat_target($rule->{'nat_addr'}, $rule->{'nat_port'});
my $out = $action;
if ($action eq 'dnat' && $rule->{'nat_family'}) {
$out .= " ".$rule->{'nat_family'};
}
$out .= " to ".$target if ($target ne '');
return $out;
}
# quick_forward_has_established(&table, chain)
# Returns true if a forward chain already accepts established traffic
sub quick_forward_has_established
{
my ($table, $chain) = @_;
foreach my $r (@{$table->{'rules'} || [ ]}) {
next if (!$r || ref($r) ne 'HASH');
next if (($r->{'chain'} || '') ne $chain);
next if (($r->{'action'} || '') ne 'accept');
return 1 if (($r->{'ct_state'} || '') =~ /\bestablished\b/);
}
return 0;
}
# quick_forward_accept_covered(&table, chain, proto, port, addr)
# Returns true if a forward accept rule already covers a DNAT destination
sub quick_forward_accept_covered
{
my ($table, $chain, $proto, $port, $addr) = @_;
foreach my $r (@{$table->{'rules'} || [ ]}) {
next if (!$r || ref($r) ne 'HASH');
next if (($r->{'chain'} || '') ne $chain);
next if (($r->{'action'} || '') ne 'accept');
next if (($r->{'proto'} || '') ne $proto);
next if (($r->{'daddr'} || '') ne $addr);
return 1 if (port_expr_covers($r->{'dport'}, $port));
}
return 0;
}
# add_quick_forward_filter(&table, proto, port, addr, addr-family)
# Adds matching forward-chain accepts for remote DNAT destinations
sub add_quick_forward_filter
{
my ($table, $proto, $port, $addr, $addr_family) = @_;
my $chain = find_base_chain($table, 'forward', 'filter');
$chain ||= find_base_chain($table, 'forward');
return 0 if (!$chain);
my $changed = 0;
if (!quick_forward_has_established($table, $chain)) {
my $est = {
'chain' => $chain,
'ct_state' => 'established,related',
'action' => 'accept',
'comment' => 'Webmin quick forward',
};
$est->{'text'} = format_rule_text($est);
insert_quick_rule($table, $chain, $est,
quick_rule_rank(quick_rule_type($est)));
$changed = 1;
}
if (!quick_forward_accept_covered($table, $chain, $proto, $port, $addr)) {
my $rule = {
'chain' => $chain,
'daddr' => $addr,
'daddr_family' => $addr_family,
'proto' => $proto,
'dport' => $port,
'action' => 'accept',
'comment' => 'Webmin quick forward',
};
$rule->{'text'} = format_rule_text($rule);
insert_quick_rule($table, $chain, $rule,
quick_rule_rank(quick_rule_type($rule)));
$changed = 1;
}
return $changed;
}
# add_quick_forward_rule(&table, src-port, proto, dst-port, dst-addr)
# Adds a simple port forward to the selected table
sub add_quick_forward_rule
{
my ($table, $src_port, $proto, $dst_port, $dst_addr) = @_;
my $err = table_supports_quick_l4($table);
return $err if ($err);
$proto = normalize_quick_proto($proto);
return text('quick_eproto') if (!$proto);
($src_port, $err) = parse_quick_port($src_port);
return $err if ($err);
$dst_port = "" if (!defined($dst_port));
$dst_port =~ s/^\s+//;
$dst_port =~ s/\s+$//;
if ($dst_port ne "") {
($dst_port, $err) = parse_quick_port($dst_port);
return $err if ($err);
}
else {
$dst_port = undef;
}
my ($addr, $addr_family, $addr_err) =
parse_quick_forward_addr($dst_addr, $table);
return $addr_err if ($addr_err);
return text('quick_eforward_target') if (!$addr && !$dst_port);
my $chain = ensure_prerouting_nat_chain($table);
my $target = format_forward_target($table, $addr, $addr_family, $dst_port);
my $rule_text = "$proto dport $src_port $target";
$rule_text = quick_rule_with_comment($rule_text, 'Webmin quick forward');
my $changed = 0;
if (!quick_text_rule_exists($table, $chain, $rule_text)) {
my $rule = parse_rule_text($rule_text);
$rule->{'chain'} = $chain;
$rule->{'text'} = $rule_text;
insert_quick_rule($table, $chain, $rule,
quick_rule_rank(quick_rule_type($rule)));
$changed = 1;
}
my $filter_port = $dst_port || $src_port;
if ($addr) {
my $ok = add_quick_forward_filter(
$table, $proto, $filter_port, $addr, $addr_family);
$changed ||= $ok;
}
else {
my ($ok, $perr) =
add_quick_accept_port($table, $filter_port, $proto,
'Webmin quick forward');
return $perr if ($perr);
$changed ||= $ok;
}
return $changed ? undef : text('quick_edup', $src_port);
}
# move_rule_in_chain(&table, chain, index, direction)
# Moves one rule within its chain and returns true if changed
sub move_rule_in_chain
{
my ($table, $chain, $idx, $dir) = @_;
return if (!defined($table) || ref($table) ne 'HASH');
return if (!defined($idx) || $idx !~ /^\d+$/);
return if (!defined($chain) || $chain eq '');
return if (!$table->{'rules'} || ref($table->{'rules'}) ne 'ARRAY');
return if ($idx > $#{$table->{'rules'}});
my $rule = $table->{'rules'}->[$idx];
return if (!$rule || $rule->{'chain'} ne $chain);
my @chain_idxs;
for (my $i = 0 ; $i < @{$table->{'rules'}} ; $i++) {
my $r = $table->{'rules'}->[$i];
next if (!$r || ref($r) ne 'HASH');
push(@chain_idxs, $i) if ($r->{'chain'} && $r->{'chain'} eq $chain);
}
my $pos;
for (my $i = 0 ; $i <= $#chain_idxs ; $i++) {
if ($chain_idxs[$i] == $idx) {
$pos = $i;
last;
}
}
return if (!defined($pos));
my $swap;
if ($dir eq 'up') {
return 0 if ($pos == 0);
$swap = $chain_idxs[$pos - 1];
}
elsif ($dir eq 'down') {
return 0 if ($pos == $#chain_idxs);
$swap = $chain_idxs[$pos + 1];
}
else {
return;
}
($table->{'rules'}->[$idx], $table->{'rules'}->[$swap]) =
($table->{'rules'}->[$swap], $table->{'rules'}->[$idx]);
reindex_table_rules($table);
return 1;
}
# format_addr_expr(direction, &rule)
# Formats a source or destination address expression
sub format_addr_expr
{
my ($dir, $rule) = @_;
my $val = $rule->{$dir};
return if (!defined($val) || $val eq '');
my $fam = guess_addr_family($val, $rule->{$dir."_family"});
return $fam." ".$dir." ".$val;
}
# format_l4proto_expr(&rule)
# Formats a layer-4 protocol expression
sub format_l4proto_expr
{
my ($rule) = @_;
my $proto = $rule->{'l4proto'};
return if (!defined($proto) || $proto eq '');
my $fam = $rule->{'l4proto_family'} || 'meta';
if ($fam eq 'ip' || $fam eq 'ip6') {
return $fam." protocol ".$proto;
}
return "meta l4proto ".$proto;
}
# format_port_expr(direction, &rule)
# Formats a source or destination port expression
sub format_port_expr
{
my ($dir, $rule) = @_;
my $val = $rule->{$dir};
return if (!defined($val) || $val eq '');
my $proto;
if ($dir eq 'sport') {
$proto =
$rule->{'sport_proto'} || $rule->{'proto'} || $rule->{'l4proto'};
}
else {
$proto = $rule->{'proto'} || $rule->{'l4proto'};
}
return if (!defined($proto) || $proto eq '');
return $proto." ".$dir." ".$val;
}
# format_tcp_flags_expr(&rule)
# Formats a TCP flags expression
sub format_tcp_flags_expr
{
my ($rule) = @_;
return if (!defined($rule->{'tcp_flags'}) || $rule->{'tcp_flags'} eq '');
my $val = $rule->{'tcp_flags'};
if (defined($rule->{'tcp_flags_mask'}) && $rule->{'tcp_flags_mask'} ne '') {
return "tcp flags & ".$rule->{'tcp_flags_mask'}." == ".$val;
}
return "tcp flags ".$val;
}
# format_limit_expr(&rule)
# Formats a rate limit expression
sub format_limit_expr
{
my ($rule) = @_;
return if (!defined($rule->{'limit_rate'}) || $rule->{'limit_rate'} eq '');
my $out = "limit rate ".$rule->{'limit_rate'};
if (defined($rule->{'limit_burst'}) && $rule->{'limit_burst'} ne '') {
my $burst = $rule->{'limit_burst'};
$out .= " burst ".$burst;
$out .= " packets" if ($burst =~ /^\d+$/);
}
return $out;
}
# format_log_expr(&rule)
# Formats a log expression
sub format_log_expr
{
my ($rule) = @_;
return if (!$rule->{'log'} && !$rule->{'log_prefix'} && !$rule->{'log_level'});
my @p = ("log");
if (defined($rule->{'log_prefix'}) && $rule->{'log_prefix'} ne '') {
my $pfx = escape_nft_string($rule->{'log_prefix'});
push(@p, "prefix", "\"".$pfx."\"");
}
if (defined($rule->{'log_level'}) && $rule->{'log_level'} ne '') {
push(@p, "level", $rule->{'log_level'});
}
return join(" ", @p);
}
# parse_rule_text(rule-text)
# Parses one nftables rule line into structured fields where possible
sub parse_rule_text
{
my ($line) = @_;
return {} if (!defined($line));
my %rule;
my @tokens = tokenize_nft_rule($line);
my @exprs;
my $i = 0;
while ($i < @tokens) {
my $tok = $tokens[$i];
if ($tok eq 'comment' && $i + 1 < @tokens) {
my $raw = $tokens[$i]." ".$tokens[$i + 1];
$rule{'comment'} = unquote_nft_string($tokens[$i + 1]);
push(@exprs, {'type' => 'comment', 'text' => $raw});
$i += 2;
next;
}
if (($tok eq 'iif' || $tok eq 'iifname') && $i + 1 < @tokens) {
my $raw = $tok." ".$tokens[$i + 1];
$rule{'iif'} = unquote_nft_string($tokens[$i + 1]);
$rule{'iif_type'} = $tok;
push(@exprs, {'type' => 'iif', 'text' => $raw});
$i += 2;
next;
}
if (($tok eq 'oif' || $tok eq 'oifname') && $i + 1 < @tokens) {
my $raw = $tok." ".$tokens[$i + 1];
$rule{'oif'} = unquote_nft_string($tokens[$i + 1]);
$rule{'oif_type'} = $tok;
push(@exprs, {'type' => 'oif', 'text' => $raw});
$i += 2;
next;
}
if (($tok eq 'ip' || $tok eq 'ip6') && $i + 2 < @tokens &&
($tokens[$i + 1] eq 'saddr' || $tokens[$i + 1] eq 'daddr')) {
my $which = $tokens[$i + 1];
my $val = $tokens[$i + 2];
my $raw = $tok." ".$which." ".$val;
$rule{$which} = $val;
$rule{$which."_family"} = $tok;
push(@exprs, {'type' => $which, 'text' => $raw});
$i += 3;
next;
}
if (($tok eq 'ip' || $tok eq 'ip6') && $i + 2 < @tokens &&
$tokens[$i + 1] eq 'protocol') {
my $val = $tokens[$i + 2];
my $raw = $tok." protocol ".$val;
$rule{'l4proto'} = $val;
$rule{'l4proto_family'} = $tok;
push(@exprs, {'type' => 'l4proto', 'text' => $raw});
$i += 3;
next;
}
if ($tok eq 'meta' && $i + 2 < @tokens &&
$tokens[$i + 1] eq 'l4proto') {
my $val = $tokens[$i + 2];
my $raw = "meta l4proto ".$val;
$rule{'l4proto'} = $val;
$rule{'l4proto_family'} = 'meta';
push(@exprs, {'type' => 'l4proto', 'text' => $raw});
$i += 3;
next;
}
if ($tok eq 'tcp' && $i + 1 < @tokens && $tokens[$i + 1] eq 'flags') {
my $j = $i + 2;
my $mask;
my $val;
if ($j < @tokens && $tokens[$j] eq '&' && $j + 1 < @tokens) {
$mask = $tokens[$j + 1];
$j += 2;
}
if ($j < @tokens && $tokens[$j] eq '==' && $j + 1 < @tokens) {
$val = $tokens[$j + 1];
$j += 2;
}
elsif ($j < @tokens) {
$val = $tokens[$j];
$j++;
}
my $raw = join(" ", @tokens[$i .. ($j - 1)]);
$rule{'tcp_flags'} = $val if (defined($val));
$rule{'tcp_flags_mask'} = $mask if (defined($mask));
push(@exprs, {'type' => 'tcp_flags', 'text' => $raw});
$i = $j;
next;
}
if (($tok eq 'tcp' || $tok eq 'udp') && $i + 2 < @tokens &&
($tokens[$i + 1] eq 'dport' || $tokens[$i + 1] eq 'sport')) {
my $dir = $tokens[$i + 1];
my $val = $tokens[$i + 2];
my $raw = $tok." ".$dir." ".$val;
if ($dir eq 'dport') {
$rule{'proto'} = $tok;
$rule{'dport'} = $val;
}
else {
$rule{'sport'} = $val;
$rule{'sport_proto'} = $tok;
}
push(@exprs, {'type' => $dir, 'text' => $raw, 'proto' => $tok});
$i += 3;
next;
}
if (($tok eq 'icmp' || $tok eq 'icmpv6') && $i + 2 < @tokens &&
$tokens[$i + 1] eq 'type') {
my $val = $tokens[$i + 2];
my $raw = $tok." type ".$val;
if ($tok eq 'icmp') {
$rule{'icmp_type'} = $val;
}
else {
$rule{'icmpv6_type'} = $val;
}
push(@exprs, {'type' => $tok, 'text' => $raw});
$i += 3;
next;
}
if ($tok eq 'ct' && $i + 2 < @tokens && $tokens[$i + 1] eq 'state') {
my $val = $tokens[$i + 2];
my $raw = "ct state ".$val;
$rule{'ct_state'} = $val;
push(@exprs, {'type' => 'ct_state', 'text' => $raw});
$i += 3;
next;
}
if ($tok eq 'limit') {
my $j = $i + 1;
my @lt = ($tok);
if ($j < @tokens && $tokens[$j] eq 'rate' && $j + 1 < @tokens) {
push(@lt, $tokens[$j], $tokens[$j + 1]);
$rule{'limit_rate'} = $tokens[$j + 1];
$j += 2;
if ($j < @tokens && $tokens[$j] eq 'burst' &&
$j + 1 < @tokens) {
push(@lt, $tokens[$j], $tokens[$j + 1]);
$rule{'limit_burst'} = $tokens[$j + 1];
$j += 2;
if ($j < @tokens && $tokens[$j] eq 'packets') {
push(@lt, $tokens[$j]);
$j++;
}
}
}
my $raw = join(" ", @lt);
push(@exprs, {'type' => 'limit', 'text' => $raw});
$i = $j;
next;
}
if ($tok eq 'log') {
my $j = $i + 1;
my @lt = ($tok);
while ($j < @tokens) {
if ($tokens[$j] eq 'prefix' && $j + 1 < @tokens) {
$rule{'log_prefix'} =
unquote_nft_string($tokens[$j + 1]);
push(@lt, $tokens[$j], $tokens[$j + 1]);
$j += 2;
next;
}
if ($tokens[$j] eq 'level' && $j + 1 < @tokens) {
$rule{'log_level'} = $tokens[$j + 1];
push(@lt, $tokens[$j], $tokens[$j + 1]);
$j += 2;
next;
}
last;
}
$rule{'log'} = 1;
my $raw = join(" ", @lt);
push(@exprs, {'type' => 'log', 'text' => $raw});
$i = $j;
next;
}
if ($tok eq 'counter') {
$rule{'counter'} = 1;
push(@exprs, {'type' => 'counter', 'text' => $tok});
$i++;
next;
}
if ($tok =~ /^(redirect|dnat)$/) {
my $action = $tok;
my $j = $i + 1;
my @nt = ($tok);
if ($action eq 'dnat' && $j < @tokens &&
($tokens[$j] eq 'ip' || $tokens[$j] eq 'ip6')) {
$rule{'nat_family'} = $tokens[$j];
push(@nt, $tokens[$j]);
$j++;
}
if ($j < @tokens && $tokens[$j] eq 'to') {
push(@nt, $tokens[$j]);
$j++;
if ($j < @tokens) {
my ($addr, $port) = parse_nat_target($tokens[$j]);
$rule{'nat_addr'} = $addr if (defined($addr));
$rule{'nat_port'} = $port if (defined($port));
push(@nt, $tokens[$j]);
$j++;
}
}
$rule{'action'} = $action;
push(@exprs, {'type' => 'nat', 'text' => join(" ", @nt)});
$i = $j;
next;
}
if ($tok =~ /^(accept|drop|reject|return)$/) {
$rule{'action'} = $tok;
push(@exprs, {'type' => 'action', 'text' => $tok});
$i++;
next;
}
if (($tok eq 'jump' || $tok eq 'goto') && $i + 1 < @tokens) {
my $raw = $tok." ".$tokens[$i + 1];
$rule{$tok} = $tokens[$i + 1];
push(@exprs, {'type' => $tok, 'text' => $raw});
$i += 2;
next;
}
push(@exprs, {'type' => 'raw', 'text' => $tok});
$i++;
}
$rule{'exprs'} = \@exprs;
return \%rule;
}
# format_rule_text(&rule)
# Formats a structured rule hash into nftables rule text
sub format_rule_text
{
my ($rule) = @_;
return "" if (!$rule || ref($rule) ne 'HASH');
my @parts;
my %used;
my $exprs = $rule->{'exprs'};
if ($exprs && ref($exprs) eq 'ARRAY' && @$exprs) {
foreach my $e (@$exprs) {
my $type = $e->{'type'} || 'raw';
if ($type eq 'action' || $type eq 'comment') {
next;
}
if ($type eq 'iif') {
if (!$used{'iif'} && defined($rule->{'iif'}) &&
$rule->{'iif'} ne '') {
my $iftype = $rule->{'iif_type'} || 'iif';
my $ival = escape_nft_string($rule->{'iif'});
push(@parts, $iftype." \"".$ival."\"");
$used{'iif'} = 1;
}
next;
}
if ($type eq 'oif') {
if (!$used{'oif'} && defined($rule->{'oif'}) &&
$rule->{'oif'} ne '') {
my $oftype = $rule->{'oif_type'} || 'oif';
my $oval = escape_nft_string($rule->{'oif'});
push(@parts, $oftype." \"".$oval."\"");
$used{'oif'} = 1;
}
next;
}
if ($type eq 'saddr') {
if (!$used{'saddr'}) {
my $addr = format_addr_expr('saddr', $rule);
if ($addr) {
push(@parts, $addr);
$used{'saddr'} = 1;
}
}
next;
}
if ($type eq 'daddr') {
if (!$used{'daddr'}) {
my $addr = format_addr_expr('daddr', $rule);
if ($addr) {
push(@parts, $addr);
$used{'daddr'} = 1;
}
}
next;
}
if ($type eq 'l4proto') {
if (!$used{'l4proto'}) {
my $lp = format_l4proto_expr($rule);
if ($lp) {
push(@parts, $lp);
$used{'l4proto'} = 1;
}
}
next;
}
if ($type eq 'sport') {
if (!$used{'sport'}) {
my $sp = format_port_expr('sport', $rule);
if ($sp) {
push(@parts, $sp);
$used{'sport'} = 1;
}
}
next;
}
if ($type eq 'dport') {
if (!$used{'dport'} && $rule->{'proto'} &&
$rule->{'dport'}) {
my $dp = format_port_expr('dport', $rule);
if ($dp) {
push(@parts, $dp);
$used{'dport'} = 1;
}
}
next;
}
if ($type eq 'icmp') {
if (!$used{'icmp'} && $rule->{'icmp_type'}) {
push(@parts, "icmp type ".$rule->{'icmp_type'});
$used{'icmp'} = 1;
}
next;
}
if ($type eq 'icmpv6') {
if (!$used{'icmpv6'} && $rule->{'icmpv6_type'}) {
push(@parts,
"icmpv6 type ".$rule->{'icmpv6_type'});
$used{'icmpv6'} = 1;
}
next;
}
if ($type eq 'ct_state') {
if (!$used{'ct_state'} && $rule->{'ct_state'}) {
push(@parts, "ct state ".$rule->{'ct_state'});
$used{'ct_state'} = 1;
}
next;
}
if ($type eq 'tcp_flags') {
if (!$used{'tcp_flags'}) {
my $tf = format_tcp_flags_expr($rule);
if ($tf) {
push(@parts, $tf);
$used{'tcp_flags'} = 1;
}
}
next;
}
if ($type eq 'limit') {
if (!$used{'limit'}) {
my $lim = format_limit_expr($rule);
if ($lim) {
push(@parts, $lim);
$used{'limit'} = 1;
}
}
next;
}
if ($type eq 'log') {
if (!$used{'log'}) {
my $lg = format_log_expr($rule);
if ($lg) {
push(@parts, $lg);
$used{'log'} = 1;
}
}
next;
}
if ($type eq 'counter') {
if (!$used{'counter'} && $rule->{'counter'}) {
push(@parts, "counter");
$used{'counter'} = 1;
}
next;
}
if ($type eq 'nat') {
if (!$used{'nat'}) {
my $nat = format_nat_expr($rule);
if ($nat) {
push(@parts, $nat);
$used{'nat'} = 1;
}
}
next;
}
if ($type eq 'jump') {
if (!$used{'jump'} && $rule->{'jump'}) {
push(@parts, "jump ".$rule->{'jump'});
$used{'jump'} = 1;
}
next;
}
if ($type eq 'goto') {
if (!$used{'goto'} && $rule->{'goto'}) {
push(@parts, "goto ".$rule->{'goto'});
$used{'goto'} = 1;
}
next;
}
push(@parts, $e->{'text'}) if ($e->{'text'});
}
}
if (!$used{'iif'} && defined($rule->{'iif'}) && $rule->{'iif'} ne '') {
my $iftype = $rule->{'iif_type'} || 'iif';
my $ival = escape_nft_string($rule->{'iif'});
push(@parts, $iftype." \"".$ival."\"");
}
if (!$used{'oif'} && defined($rule->{'oif'}) && $rule->{'oif'} ne '') {
my $oftype = $rule->{'oif_type'} || 'oif';
my $oval = escape_nft_string($rule->{'oif'});
push(@parts, $oftype." \"".$oval."\"");
}
if (!$used{'saddr'}) {
my $addr = format_addr_expr('saddr', $rule);
push(@parts, $addr) if ($addr);
}
if (!$used{'daddr'}) {
my $addr = format_addr_expr('daddr', $rule);
push(@parts, $addr) if ($addr);
}
if (!$used{'l4proto'}) {
my $lp = format_l4proto_expr($rule);
push(@parts, $lp) if ($lp);
}
if (!$used{'sport'}) {
my $sp = format_port_expr('sport', $rule);
push(@parts, $sp) if ($sp);
}
if (!$used{'dport'}) {
my $dp = format_port_expr('dport', $rule);
push(@parts, $dp) if ($dp);
}
if (!$used{'icmp'} && $rule->{'icmp_type'}) {
push(@parts, "icmp type ".$rule->{'icmp_type'});
}
if (!$used{'icmpv6'} && $rule->{'icmpv6_type'}) {
push(@parts, "icmpv6 type ".$rule->{'icmpv6_type'});
}
if (!$used{'tcp_flags'}) {
my $tf = format_tcp_flags_expr($rule);
push(@parts, $tf) if ($tf);
}
if (!$used{'ct_state'} && $rule->{'ct_state'}) {
push(@parts, "ct state ".$rule->{'ct_state'});
}
if (!$used{'limit'}) {
my $lim = format_limit_expr($rule);
push(@parts, $lim) if ($lim);
}
if (!$used{'log'}) {
my $lg = format_log_expr($rule);
push(@parts, $lg) if ($lg);
}
if (!$used{'counter'} && $rule->{'counter'}) {
push(@parts, "counter");
}
if (!$used{'jump'} && $rule->{'jump'}) {
push(@parts, "jump ".$rule->{'jump'});
}
if (!$used{'goto'} && $rule->{'goto'}) {
push(@parts, "goto ".$rule->{'goto'});
}
if (!$used{'nat'}) {
my $nat = format_nat_expr($rule);
push(@parts, $nat) if ($nat);
}
if ($rule->{'action'} && !$rule->{'jump'} && !$rule->{'goto'} &&
$rule->{'action'} !~ /^(redirect|dnat)$/) {
push(@parts, $rule->{'action'});
}
if (defined($rule->{'comment'}) && $rule->{'comment'} ne '') {
my $c = escape_nft_string($rule->{'comment'});
push(@parts, "comment \"".$c."\"");
}
my $text = join(" ", grep { defined($_) && $_ ne '' } @parts);
$text =~ s/^\s+//;
$text =~ s/\s+$//;
return $text;
}
# parse_set_elements_string(string)
# Parses a comma-separated nftables set elements string
sub parse_set_elements_string
{
my ($text) = @_;
return [ ] if (!defined($text));
$text =~ s/^\s+//;
$text =~ s/\s+$//;
return [ ] if ($text eq '');
my @vals = split(/\s*,\s*/, $text);
@vals = grep { defined($_) && $_ ne '' } @vals;
return \@vals;
}
# parse_set_elements_input(string)
# Parses set elements from textarea input
sub parse_set_elements_input
{
my ($text) = @_;
return [ ] if (!defined($text));
$text =~ s/\r//g;
$text =~ s/^\s+//;
$text =~ s/\s+$//;
return [ ] if ($text eq '');
$text =~ s/\n/,/g;
return parse_set_elements_string($text);
}
# set_elements_text(&set)
# Returns set elements formatted for textarea editing
sub set_elements_text
{
my ($set) = @_;
return "" if (!$set || ref($set) ne 'HASH');
return "" if (!$set->{'elements'} || ref($set->{'elements'}) ne 'ARRAY');
return join("\n", @{$set->{'elements'}});
}
# set_elements_summary(&set)
# Returns a short set elements summary for table listings
sub set_elements_summary
{
my ($set) = @_;
return "-" if (!$set || ref($set) ne 'HASH');
return "-" if (!$set->{'elements'} || ref($set->{'elements'}) ne 'ARRAY');
my @elems = @{$set->{'elements'}};
return "-" if (!@elems);
my $max = 20;
my $preview =
join(", ", @elems[0 .. ($#elems < $max - 1 ? $#elems : $max - 1)]);
if (@elems > $max) {
$preview .= ", ...";
}
return $preview;
}
# normalize_port_set_elements(elements)
# Removes overlaps from port set elements so interval sets are valid
sub normalize_port_set_elements
{
my (@elements) = @_;
my (@ranges, @other);
foreach my $e (@elements) {
if ($e =~ /^(\d+)-(\d+)$/) {
my ($start, $end) = ($1, $2);
($start, $end) = ($end, $start) if ($start > $end);
push(@ranges, [$start, $end]);
}
elsif ($e =~ /^(\d+)$/) {
push(@ranges, [$1, $1]);
}
else {
push(@other, $e);
}
}
@ranges = sort { $a->[0] <=> $b->[0] || $a->[1] <=> $b->[1] } @ranges;
my @merged;
foreach my $r (@ranges) {
if (@merged && $r->[0] <= $merged[-1]->[1]) {
$merged[-1]->[1] = $r->[1] if ($r->[1] > $merged[-1]->[1]);
}
else {
push(@merged, [@$r]);
}
}
return (map { $_->[0] == $_->[1] ? $_->[0] : $_->[0]."-".$_->[1] } @merged,
sort port_sort @other);
}
# port_sort(a, b)
# Sorts nftables service ports and ranges by starting port number
sub port_sort
{
my ($aa) = $a =~ /^(\d+)/;
my ($bb) = $b =~ /^(\d+)/;
return ($aa || 0) <=> ($bb || 0) || $a cmp $b;
}
# setup_profiles()
# Returns available ruleset profiles and their default policies/services
sub setup_profiles
{
return (
{
'id' => 'allow_all',
'name' => text('setup_profile_allow_all'),
'desc' => text('setup_profile_allow_all_desc'),
'input' => 'accept',
'forward' => 'accept',
'output' => 'accept',
'services' => [ ]
},
{
'id' => 'management',
'name' => text('setup_profile_management'),
'desc' => text('setup_profile_management_desc'),
'input' => 'drop',
'forward' => 'drop',
'output' => 'accept',
'services' => [qw(ssh webmin)]
},
{
'id' => 'web',
'name' => text('setup_profile_web'),
'desc' => text('setup_profile_web_desc'),
'input' => 'drop',
'forward' => 'drop',
'output' => 'accept',
'services' => [qw(ssh webmin http https)]
},
{
'id' => 'mail',
'name' => text('setup_profile_mail'),
'desc' => text('setup_profile_mail_desc'),
'input' => 'drop',
'forward' => 'drop',
'output' => 'accept',
'services' => [
qw(ssh usermin smtp submission smtps pop3 pop3s imap imaps)
]
},
{
'id' => 'dns',
'name' => text('setup_profile_dns'),
'desc' => text('setup_profile_dns_desc'),
'input' => 'drop',
'forward' => 'drop',
'output' => 'accept',
'services' => [qw(ssh webmin dhcpv6 dns dot mdns)]
},
{
'id' => 'virtualmin',
'name' => text('setup_profile_virtualmin'),
'desc' => text('setup_profile_virtualmin_desc'),
'input' => 'drop',
'forward' => 'drop',
'output' => 'accept',
'services' => [
qw(ssh webmin dhcpv6 dns dot ftp http https imap imaps
mdns pop3 pop3s smtp submission smtps ftp_data
ssh_alt webmin_range usermin passive_ftp)
]
},
{
'id' => 'locked',
'name' => text('setup_profile_locked'),
'desc' => text('setup_profile_locked_desc'),
'input' => 'drop',
'forward' => 'drop',
'output' => 'drop',
'services' => [ ]
},
{
'id' => 'custom',
'name' => text('setup_profile_custom'),
'desc' => text('setup_profile_custom_desc'),
'input' => 'drop',
'forward' => 'drop',
'output' => 'accept',
'services' => [ ]
},
);
}
# profile_ports_or_default(&ports, proto, &service-names, &fallback-ports)
# Returns valid nftables port expressions from config, /etc/services or fallback
sub profile_ports_or_default
{
my ($ports, $proto, $service_names, $fallbacks) = @_;
my @ports = clean_profile_ports($proto, @$ports);
if (!@ports && $service_names && @$service_names) {
@ports = clean_profile_ports(
$proto,
map { get_etc_service_port($_, $proto) } @$service_names
);
}
if (!@ports && $fallbacks) {
@ports = clean_profile_ports($proto, @$fallbacks);
}
return @ports;
}
# clean_profile_ports(proto, port|service|range, ...)
# Expands service names and removes invalid profile port expressions
sub clean_profile_ports
{
my ($proto, @ports) = @_;
my %seen;
foreach my $port (@ports) {
next if (!defined($port));
foreach my $p (split(/[\s,]+/, $port)) {
foreach my $e (expand_profile_port($p, $proto)) {
$seen{$e} = 1;
}
}
}
return normalize_port_set_elements(keys %seen);
}
# expand_profile_port(port|service|range, proto)
# Converts one configured value to one or more nftables port expressions
sub expand_profile_port
{
my ($port, $proto) = @_;
return ( ) if (!defined($port));
$port =~ s/^\s+//;
$port =~ s/\s+$//;
return ( ) if ($port eq '');
if ($port =~ /^(\d+)$/) {
my $p = $1;
return valid_profile_port_number($p) ? ($p) : ( );
}
if ($port =~ /^(\d+)-(\d+)$/) {
my ($from, $to) = ($1, $2);
return valid_profile_port_number($from) &&
valid_profile_port_number($to) ? ("$from-$to") : ( );
}
my $svcport = get_etc_service_port($port, $proto);
return defined($svcport) ? ($svcport) : ( );
}
# valid_profile_port_number(port)
# Returns true for a valid TCP/UDP port number
sub valid_profile_port_number
{
return defined($_[0]) && $_[0] =~ /^\d+$/ && $_[0] >= 1 && $_[0] <= 65535;
}
# profile_port_number(port|service, proto)
# Returns a single numeric port number for a configured value
sub profile_port_number
{
my ($port, $proto) = @_;
my @ports = expand_profile_port($port, $proto);
return @ports && $ports[0] =~ /^\d+$/ ? $ports[0] : undef;
}
# profile_accept_rules(proto, ports...)
# Returns simple inbound accept rules for the given ports
sub profile_accept_rules
{
my ($proto, @ports) = @_;
return map { "$proto dport $_ accept" } @ports;
}
# profile_ports_label(ports...)
# Formats a port list for the setup UI
sub profile_ports_label
{
return @_ ? join(", ", @_) : "-";
}
# get_etc_service_port(service|&services, proto, [services-file])
# Looks up a default service port in /etc/services
sub get_etc_service_port
{
my ($services, $proto, $file) = @_;
my @services = ref($services) eq 'ARRAY' ? @$services : ($services);
my $map = read_etc_services($file);
foreach my $service (@services) {
next if (!defined($service));
my $port = $map->{lc($proto || '')}->{lc($service)};
return $port if (defined($port));
}
return;
}
# read_etc_services([services-file])
# Parses /etc/services into a protocol/name to port map
sub read_etc_services
{
my ($file) = @_;
$file ||= "/etc/services";
our %profile_etc_services_cache;
return $profile_etc_services_cache{$file}
if (defined($profile_etc_services_cache{$file}));
my %map;
if (open(my $fh, "<", $file)) {
while(my $line = <$fh>) {
$line =~ s/#.*$//;
$line =~ s/^\s+//;
$line =~ s/\s+$//;
next if ($line eq '');
my ($name, $portproto, @aliases) = split(/\s+/, $line);
next if (!$name || !$portproto);
next if ($portproto !~ /^(\d+)\/([A-Za-z0-9_+-]+)$/);
my ($port, $proto) = ($1, lc($2));
next if (!valid_profile_port_number($port));
foreach my $n ($name, @aliases) {
$map{$proto}->{lc($n)} ||= $port;
}
}
close($fh);
}
$profile_etc_services_cache{$file} = \%map;
return \%map;
}
# foreign_require_quiet(module, [config-key...])
# Loads a foreign module API, returning false on any failure
sub foreign_require_quiet
{
my ($mod, @config_keys) = @_;
my $ok = eval {
if (foreign_check($mod) &&
foreign_config_has_readable_file($mod, @config_keys)) {
local $main::error_must_die = 1;
foreign_require($mod);
1;
}
else {
0;
}
};
return $ok && !$@ ? 1 : 0;
}
# foreign_config_has_readable_file(module, config-key...)
# Returns true if no keys are required, or a configured file exists
sub foreign_config_has_readable_file
{
my ($mod, @keys) = @_;
return 1 if (!@keys);
my %fconfig = foreign_config($mod);
foreach my $key (@keys) {
foreach my $file (split(/\s+/, $fconfig{$key} || '')) {
return 1 if (-r $file);
}
}
return 0;
}
# configured_port_from_address(value, [default-port])
# Extracts a port from address:port, [address]:port or bare port values
sub configured_port_from_address
{
my ($value, $default) = @_;
return if (!defined($value) || $value eq '');
return $1 if ($value =~ /^(\d+)$/);
return $1 if ($value =~ /^\[[^\]]+\]:(\d+)$/);
return $1 if ($value =~ /^[^:]+:(\d+)$/);
return $default if (defined($default) && $value =~ /\S/);
return;
}
# address_is_loopback(address)
# Returns true if an address is loopback-only
sub address_is_loopback
{
my ($addr) = @_;
return 0 if (!defined($addr) || $addr eq '' || $addr eq '*' ||
$addr eq '0.0.0.0' || $addr eq '::' || $addr eq '[::]');
$addr =~ s/^\[//;
$addr =~ s/\]$//;
return 1 if (lc($addr) eq 'localhost' || $addr eq '::1' ||
$addr =~ /^127\./);
return 0;
}
# get_sshd_ports()
# Returns configured SSH server ports from the sshd module
sub get_sshd_ports
{
return ( ) if (!foreign_require_quiet('sshd', 'sshd_config'));
my @ports = eval {
my $conf = sshd::get_sshd_config();
my @rv;
foreach my $p (sshd::find('Port', $conf)) {
push(@rv, @{$p->{'values'} || [ ]});
}
foreach my $l (sshd::find('ListenAddress', $conf)) {
my $listen = $l->{'values'}->[0];
my $port = configured_port_from_address($listen);
push(@rv, $port) if ($port);
}
clean_profile_ports('tcp', @rv);
};
return $@ ? ( ) : @ports;
}
# miniserv_config_ports(&miniserv-config)
# Extracts configured miniserv listener ports
sub miniserv_config_ports
{
my ($miniserv) = @_;
my @ports;
push(@ports, $miniserv->{'port'})
if (valid_profile_port_number($miniserv->{'port'}));
foreach my $sock (split(/\s+/, $miniserv->{'sockets'} || '')) {
my $port = configured_port_from_address($sock);
push(@ports, $port) if ($port);
}
return clean_profile_ports('tcp', @ports);
}
# get_webmin_ports()
# Returns configured Webmin listener ports
sub get_webmin_ports
{
my %miniserv;
if (get_miniserv_config(\%miniserv)) {
return miniserv_config_ports(\%miniserv);
}
return ( );
}
# get_usermin_ports()
# Returns configured Usermin listener ports
sub get_usermin_ports
{
return ( ) if (!foreign_require_quiet('usermin'));
my @ports = eval {
my %miniserv;
usermin::get_usermin_miniserv_config(\%miniserv);
miniserv_config_ports(\%miniserv);
};
return $@ ? ( ) : @ports;
}
# get_bind_ports(tls)
# Returns configured BIND DNS or DNS-over-TLS listener ports
sub get_bind_ports
{
my ($want_tls) = @_;
return ( ) if (!foreign_require_quiet('bind8', 'named_conf'));
my @ports = eval {
my $conf = bind8::get_config();
my $options = bind8::find('options', $conf);
my @rv;
if ($options) {
foreach my $l (bind8::find('listen-on', $options->{'members'}),
bind8::find('listen-on-v6',
$options->{'members'})) {
my $vals = $l->{'values'} || [ ];
my $has_tls = scalar(grep { $_ eq 'tls' } @$vals) ? 1 : 0;
next if ($want_tls != $has_tls);
my $port;
for(my $i = 0; $i < @$vals; $i++) {
if ($vals->[$i] eq 'port') {
$port = $vals->[$i + 1];
last;
}
}
$port ||= get_etc_service_port('domain', 'tcp') || 53;
push(@rv, $port);
}
}
clean_profile_ports('tcp', @rv);
};
return $@ ? ( ) : @ports;
}
# get_apache_ports(https)
# Returns configured Apache HTTP or HTTPS listener ports
sub get_apache_ports
{
my ($https) = @_;
return ( ) if (!foreign_require_quiet('apache', 'httpd_conf'));
my @ports = eval {
my $conf = apache::get_config();
my $defport = profile_port_number(
apache::find_directive('Port', $conf, 1), 'tcp');
$defport ||= get_etc_service_port('http', 'tcp') || 80;
my (%http_vhost, %https_vhost, @rv);
foreach my $v (apache::find_directive_struct('VirtualHost', $conf)) {
my $vm = $v->{'members'} || [ ];
my $ssl = lc(apache::find_vdirective(
'SSLEngine', $vm, $conf, 1) || '') eq 'on';
foreach my $word (@{$v->{'words'} || [ ]}) {
my $port = configured_port_from_address($word, $defport);
next if (!$port || $port eq '*');
if ($ssl || $port == 443) {
$https_vhost{$port} = 1;
}
else {
$http_vhost{$port} = 1;
}
}
}
foreach my $port (keys %http_vhost, keys %https_vhost) {
push(@rv, $port)
if ($https ? $https_vhost{$port} : $http_vhost{$port});
}
foreach my $listen (apache::find_directive('Listen', $conf)) {
my ($first) = split(/\s+/, $listen);
my $port = configured_port_from_address($first, $defport);
next if (!$port);
if ($https) {
push(@rv, $port)
if ($https_vhost{$port} ||
(!$http_vhost{$port} && $port == 443));
}
else {
push(@rv, $port)
if ($http_vhost{$port} ||
(!$https_vhost{$port} && $port != 443));
}
}
if (!@rv && !$https && $defport != 443) {
push(@rv, $defport);
}
elsif (!@rv && $https && $defport == 443) {
push(@rv, $defport);
}
clean_profile_ports('tcp', @rv);
};
return $@ ? ( ) : @ports;
}
# get_nginx_ports(https)
# Returns configured Nginx HTTP or HTTPS listener ports
sub get_nginx_ports
{
my ($https) = @_;
return ( ) if (!foreign_require_quiet('nginx', 'nginx_config'));
my @ports = eval {
my $conf = nginx::get_config();
my $http = nginx::find('http', $conf);
my @rv;
if ($http) {
foreach my $server (nginx::find('server', $http)) {
my @listen = nginx::find('listen', $server);
@listen = ({ 'words' => [ '80' ] }) if (!@listen);
my $server_ssl = lc(nginx::find_value('ssl', $server) || '')
eq 'on';
foreach my $l (@listen) {
my @words = @{$l->{'words'} || [ ]};
next if (!@words || $words[0] =~ /^unix:/);
my (undef, $port) = nginx::split_ip_port($words[0]);
next if (!valid_profile_port_number($port));
my $ssl = $server_ssl ||
scalar(grep { lc($_) eq 'ssl' } @words);
if ($https ? ($ssl || $port == 443) :
(!$ssl && $port != 443)) {
push(@rv, $port);
}
}
}
}
clean_profile_ports('tcp', @rv);
};
return $@ ? ( ) : @ports;
}
# get_dovecot_ports(listener)
# Returns configured Dovecot IMAP/POP3 listener ports
sub get_dovecot_ports
{
my ($listener) = @_;
return ( ) if (!foreign_require_quiet('dovecot', 'dovecot_config'));
my @ports = eval {
my $conf = dovecot::get_config();
my @rv;
foreach my $p (dovecot::find('port', $conf, 0,
'inet_listener', $listener)) {
push(@rv, $p->{'value'}) if (($p->{'value'} || '') ne '0');
}
clean_profile_ports('tcp', @rv);
};
return $@ ? ( ) : @ports;
}
# get_proftpd_ports()
# Returns configured ProFTPD control listener ports
sub get_proftpd_ports
{
return ( ) if (!foreign_require_quiet('proftpd', 'proftpd_conf'));
my @ports = eval {
my $conf = proftpd::get_config();
my @rv = proftpd::find_directive('Port', $conf);
foreach my $v (proftpd::find_directive_struct('VirtualHost', $conf)) {
push(@rv, proftpd::find_directive('Port',
$v->{'members'} || [ ]));
}
clean_profile_ports('tcp', @rv);
};
return $@ ? ( ) : @ports;
}
# get_proftpd_passive_ports()
# Returns configured ProFTPD passive port ranges
sub get_proftpd_passive_ports
{
return ( ) if (!foreign_require_quiet('proftpd', 'proftpd_conf'));
my @ports = eval {
my $conf = proftpd::get_config();
my @dirs = proftpd::find_directive_struct('PassivePorts', $conf);
foreach my $v (proftpd::find_directive_struct('VirtualHost', $conf)) {
push(@dirs, proftpd::find_directive_struct(
'PassivePorts', $v->{'members'} || [ ]));
}
my @rv;
foreach my $d (@dirs) {
my @w = @{$d->{'words'} || [ ]};
push(@rv, "$w[0]-$w[1]")
if (valid_profile_port_number($w[0]) &&
valid_profile_port_number($w[1]));
}
clean_profile_ports('tcp', @rv);
};
return $@ ? ( ) : @ports;
}
# get_postfix_ports(service)
# Returns configured Postfix SMTP listener ports for smtp/submission/smtps
sub get_postfix_ports
{
my ($service) = @_;
return ( ) if (!foreign_require_quiet('postfix', 'postfix_master'));
my @ports = eval {
my $masters = postfix::get_master_config();
my @rv;
foreach my $m (@$masters) {
next if (!$m->{'enabled'} || $m->{'type'} ne 'inet');
next if (($m->{'command'} || '') !~ /(^|\s)smtpd(\s|$)/);
my ($port, $addr) = postfix_master_port($m->{'name'});
next if (!$port || address_is_loopback($addr));
push(@rv, $port)
if (mail_listener_matches_service(
$service, $m->{'name'}, $port));
}
clean_profile_ports('tcp', @rv);
};
return $@ ? ( ) : @ports;
}
# postfix_master_port(service-name)
# Returns port and optional bind address from a Postfix master.cf service name
sub postfix_master_port
{
my ($name) = @_;
return (undef, undef) if (!defined($name));
if ($name =~ /^\[([^\]]+)\]:(\S+)$/ || $name =~ /^([^:]+):(\S+)$/) {
my ($addr, $svc) = ($1, $2);
return (profile_port_number($svc, 'tcp'), $addr);
}
return (profile_port_number($name, 'tcp'), undef);
}
# get_sendmail_ports(service)
# Returns configured Sendmail listener ports for smtp/submission/smtps
sub get_sendmail_ports
{
my ($service) = @_;
return ( ) if (!foreign_require_quiet('sendmail', 'sendmail_cf'));
my @ports = eval {
my $conf = sendmail::get_sendmailcf();
my @rv;
{
no warnings 'once';
local @sendmail::rv;
foreach my $dpo (sendmail::find_options(
'DaemonPortOptions', $conf)) {
my %opts;
foreach my $o (split(/\s*,\s*/, $dpo->[1])) {
if ($o =~ /^([^=]+)=(\S+)$/) {
$opts{$1} = $2;
}
}
foreach my $k (qw(Name Address Port Modifiers Family)) {
my $short = substr($k, 0, 1);
$opts{$k} ||= $opts{$short};
}
$opts{'Address'} ||= $opts{'Addr'};
next if (address_is_loopback($opts{'Address'}));
my $name = $opts{'Name'} || 'MTA';
my $port = $opts{'Port'} ?
profile_port_number($opts{'Port'}, 'tcp') :
default_mail_service_port($name);
next if (!$port);
push(@rv, $port)
if (mail_listener_matches_service(
$service, $name, $port));
}
}
clean_profile_ports('tcp', @rv);
};
return $@ ? ( ) : @ports;
}
# default_mail_service_port(listener-name)
# Returns the default port implied by a Sendmail daemon name
sub default_mail_service_port
{
my ($name) = @_;
my $lname = lc($name || '');
return get_etc_service_port('submission', 'tcp') || 587
if ($lname eq 'msa' || $lname eq 'submission');
return get_etc_service_port([ 'submissions', 'smtps' ], 'tcp') || 465
if ($lname eq 'smtps' || $lname eq 'submissions');
return get_etc_service_port('smtp', 'tcp') || 25;
}
# mail_listener_matches_service(service, listener-name, port)
# Classifies MTA listener ports into smtp/submission/smtps profile services
sub mail_listener_matches_service
{
my ($service, $name, $port) = @_;
my $lname = lc($name || '');
my $smtp = get_etc_service_port('smtp', 'tcp') || 25;
my $submission = get_etc_service_port('submission', 'tcp') || 587;
my $smtps = get_etc_service_port([ 'submissions', 'smtps' ], 'tcp') || 465;
if ($service eq 'submission') {
return $lname eq 'submission' || $lname eq 'msa' ||
$port == $submission;
}
if ($service eq 'smtps') {
return $lname eq 'smtps' || $lname eq 'submissions' ||
$port == $smtps;
}
return $lname eq 'smtp' || $lname eq 'mta' || $port == $smtp ||
($port != $submission && $port != $smtps);
}
# setup_services()
# Returns selectable services and ports used by ruleset profiles
sub setup_services
{
my @ssh_ports = profile_ports_or_default([ get_sshd_ports() ],
'tcp', [ 'ssh' ], [ 22 ]);
my @webmin_ports = profile_ports_or_default([ get_webmin_ports() ],
'tcp', [ 'webmin' ], [ 10000 ]);
my @usermin_ports = profile_ports_or_default([ get_usermin_ports() ],
'tcp', [ 'usermin' ], [ 20000 ]);
my @dhcpv6_ports = profile_ports_or_default([ ],
'udp', [ 'dhcpv6-client' ], [ 546 ]);
my @dns_ports = profile_ports_or_default([ get_bind_ports(0) ],
'tcp', [ 'domain', 'dns' ], [ 53 ]);
my @dot_ports = profile_ports_or_default([ get_bind_ports(1) ],
'tcp', [ 'domain-s', 'dns-over-tls' ], [ 853 ]);
my @ftp_ports = profile_ports_or_default([ get_proftpd_ports() ],
'tcp', [ 'ftp' ], [ 21 ]);
my @http_ports = profile_ports_or_default(
[ get_apache_ports(0), get_nginx_ports(0) ],
'tcp', [ 'http', 'www', 'www-http' ], [ 80 ]);
my @https_ports = profile_ports_or_default(
[ get_apache_ports(1), get_nginx_ports(1) ],
'tcp', [ 'https' ], [ 443 ]);
my @imap_ports = profile_ports_or_default(
[ get_dovecot_ports('imap') ],
'tcp', [ 'imap2', 'imap' ], [ 143 ]);
my @imaps_ports = profile_ports_or_default(
[ get_dovecot_ports('imaps') ],
'tcp', [ 'imaps' ], [ 993 ]);
my @mdns_ports = profile_ports_or_default([ ],
'udp', [ 'mdns' ], [ 5353 ]);
my @pop3_ports = profile_ports_or_default(
[ get_dovecot_ports('pop3') ],
'tcp', [ 'pop3' ], [ 110 ]);
my @pop3s_ports = profile_ports_or_default(
[ get_dovecot_ports('pop3s') ],
'tcp', [ 'pop3s' ], [ 995 ]);
my @smtp_ports = profile_ports_or_default(
[ get_postfix_ports('smtp'), get_sendmail_ports('smtp') ],
'tcp', [ 'smtp' ], [ 25 ]);
my @submission_ports = profile_ports_or_default(
[ get_postfix_ports('submission'), get_sendmail_ports('submission') ],
'tcp', [ 'submission' ], [ 587 ]);
my @smtps_ports = profile_ports_or_default(
[ get_postfix_ports('smtps'), get_sendmail_ports('smtps') ],
'tcp', [ 'submissions', 'smtps' ], [ 465 ]);
my @ftp_data_ports = profile_ports_or_default([ ],
'tcp', [ 'ftp-data' ], [ 20 ]);
my @passive_ftp_ports = profile_ports_or_default(
[ get_proftpd_passive_ports() ],
'tcp', [ ], [ '49152-65535' ]);
return (
{
'id' => 'ssh',
'label' => text('setup_svc_ssh'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@ssh_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @ssh_ports) ]
},
{
'id' => 'webmin',
'label' => text('setup_svc_webmin'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@webmin_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @webmin_ports) ]
},
{
'id' => 'dhcpv6',
'label' => text('setup_svc_dhcpv6'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@dhcpv6_ports),
'proto' => 'UDP',
'rules' => [
map { "ip6 daddr fe80::/64 udp dport $_ accept" }
@dhcpv6_ports
]
},
{
'id' => 'dns',
'label' => text('setup_svc_dns'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@dns_ports),
'proto' => 'TCP/UDP',
'rules' => [
profile_accept_rules('tcp', @dns_ports),
profile_accept_rules('udp', @dns_ports)
]
},
{
'id' => 'dot',
'label' => text('setup_svc_dot'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@dot_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @dot_ports) ]
},
{
'id' => 'ftp',
'label' => text('setup_svc_ftp'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@ftp_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @ftp_ports) ]
},
{
'id' => 'http',
'label' => text('setup_svc_http'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@http_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @http_ports) ]
},
{
'id' => 'https',
'label' => text('setup_svc_https'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@https_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @https_ports) ]
},
{
'id' => 'imap',
'label' => text('setup_svc_imap'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@imap_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @imap_ports) ]
},
{
'id' => 'imaps',
'label' => text('setup_svc_imaps'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@imaps_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @imaps_ports) ]
},
{
'id' => 'mdns',
'label' => text('setup_svc_mdns'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@mdns_ports),
'proto' => 'UDP',
'rules' => [
map {
(
"ip daddr 224.0.0.251 udp dport $_ accept",
"ip6 daddr ff02::fb udp dport $_ accept"
)
} @mdns_ports
]
},
{
'id' => 'pop3',
'label' => text('setup_svc_pop3'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@pop3_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @pop3_ports) ]
},
{
'id' => 'pop3s',
'label' => text('setup_svc_pop3s'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@pop3s_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @pop3s_ports) ]
},
{
'id' => 'smtp',
'label' => text('setup_svc_smtp'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@smtp_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @smtp_ports) ]
},
{
'id' => 'submission',
'label' => text('setup_svc_submission'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@submission_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @submission_ports) ]
},
{
'id' => 'smtps',
'label' => text('setup_svc_smtps'),
'type' => text('setup_type_service'),
'port' => profile_ports_label(@smtps_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @smtps_ports) ]
},
{
'id' => 'ftp_data',
'label' => text('setup_port_ftp_data'),
'type' => text('setup_type_port'),
'port' => profile_ports_label(@ftp_data_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @ftp_data_ports) ]
},
{
'id' => 'ssh_alt',
'label' => text('setup_port_ssh_alt'),
'type' => text('setup_type_port'),
'port' => '2222',
'proto' => 'TCP',
'rules' => ['tcp dport 2222 accept']
},
{
'id' => 'webmin_range',
'label' => text('setup_port_webmin_range'),
'type' => text('setup_type_port'),
'port' => '10000-10100',
'proto' => 'TCP',
'rules' => ['tcp dport 10000-10100 accept']
},
{
'id' => 'usermin',
'label' => text('setup_port_usermin'),
'type' => text('setup_type_port'),
'port' => profile_ports_label(@usermin_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @usermin_ports) ]
},
{
'id' => 'passive_ftp',
'label' => text('setup_port_passive_ftp'),
'type' => text('setup_type_port'),
'port' => profile_ports_label(@passive_ftp_ports),
'proto' => 'TCP',
'rules' => [ profile_accept_rules('tcp', @passive_ftp_ports) ]
},
);
}
# create_profile_ruleset(table-name, profile-id, allowed-service-ids|'*')
# Builds an inet table for a selected profile and service list
sub create_profile_ruleset
{
my ($table_name, $profile_id, $allow_ids) = @_;
my %profiles = map { $_->{'id'} => $_ } setup_profiles();
my $profile = $profiles{$profile_id} || error(text('setup_invalid_type'));
my @services = setup_services();
my %services = map { $_->{'id'} => $_ } @services;
my @allow_ids;
if (!defined($allow_ids) || $allow_ids eq '*') {
@allow_ids = @{$profile->{'services'} || [ ]};
}
elsif (ref($allow_ids) eq 'ARRAY') {
@allow_ids = @$allow_ids;
}
else {
@allow_ids = grep { $_ ne '' } split(/\s*,\s*|\s+/, $allow_ids);
}
my %allow;
foreach my $id (@allow_ids) {
$services{$id} || error(text('setup_eservice', $id));
$allow{$id} = 1;
}
my $table = {
'name' => $table_name,
'family' => 'inet',
'rules' => [ ],
'sets' => {},
'chains' => {
'input' => {
'type' => 'filter',
'hook' => 'input',
'priority' => 0,
'policy' => $profile->{'input'}
},
'forward' => {
'type' => 'filter',
'hook' => 'forward',
'priority' => 0,
'policy' => $profile->{'forward'}
},
'output' => {
'type' => 'filter',
'hook' => 'output',
'priority' => 0,
'policy' => $profile->{'output'}
}
}
};
return $table if ($profile_id eq 'allow_all');
add_profile_rule($table, 'input', 'ct state established,related accept');
add_profile_rule($table, 'input', 'iif "lo" accept');
add_profile_rule($table, 'input', 'meta l4proto { icmp, ipv6-icmp } accept');
if ($profile->{'output'} eq 'drop') {
add_profile_rule($table, 'output',
'ct state established,related accept');
add_profile_rule($table, 'output', 'oif "lo" accept');
add_profile_rule($table, 'output',
'meta l4proto { icmp, ipv6-icmp } accept');
}
my %seen;
my %ports;
my @special_rules;
foreach my $id (map { $_->{'id'} } @services) {
next if (!$allow{$id});
foreach my $rule (@{$services{$id}->{'rules'}}) {
next if ($seen{$rule}++);
if ($rule =~ /^(tcp|udp)\s+dport\s+(\S+)\s+accept$/) {
$ports{$1}->{$2} = 1;
}
else {
push(@special_rules, $rule);
}
}
}
add_profile_port_set($table, $profile_id, \%ports);
foreach my $rule (@special_rules) {
add_profile_rule($table, 'input', $rule);
}
return $table;
}
# save_profile_ruleset(table-name, profile-id, allowed-service-ids|'*')
# Saves or replaces a Webmin-managed profile table and returns an error
sub save_profile_ruleset
{
my ($table_name, $profile_id, $allow_ids) = @_;
return text('create_ename')
if (!defined($table_name) || $table_name !~ /^\w[\w-]*$/);
my $table = create_profile_ruleset($table_name, $profile_id, $allow_ids);
my ($active, $active_err) = get_active_nftables_save();
if (!$active_err) {
foreach my $t (@$active) {
if ($t->{'family'} eq 'inet' && $t->{'name'} eq $table_name &&
table_is_externally_managed($t)) {
return text('create_eexternal', nft_table_spec($t));
}
}
}
my @tables = get_nftables_save();
my $done;
foreach my $i (0 .. $#tables) {
if ($tables[$i]->{'family'} eq 'inet' &&
$tables[$i]->{'name'} eq $table_name) {
$tables[$i] = $table;
$done = 1;
last;
}
}
push(@tables, $table) if (!$done);
return save_configuration(@tables);
}
# add_profile_port_set(&table, profile-id, &proto-ports)
# Adds profile service port sets and their input accept rules
sub add_profile_port_set
{
my ($table, $profile_id, $ports) = @_;
# Keep TCP and UDP ports in separate sets when they differ, otherwise a UDP
# accept rule would also allow TCP-only service ports.
my @protos = grep { keys %{$ports->{$_}} } sort keys %$ports;
return if (!@protos);
foreach my $proto (@protos) {
next if (!keys %{$ports->{$proto}});
my $set_name =
profile_port_set_name($profile_id, $proto, scalar(@protos));
my @elements = normalize_port_set_elements(keys %{$ports->{$proto}});
$table->{'sets'}->{$set_name} = {
'name' => $set_name,
'type' => 'inet_service',
'flags' => (grep { /-/ } @elements) ? 'interval' : undef,
'elements' => \@elements,
'raw_lines' => [ ],
};
add_profile_rule($table, 'input', "$proto dport \@$set_name accept");
}
return;
}
# add_profile_rule(&table, chain, rule-text)
# Appends a generated rule to a profile table
sub add_profile_rule
{
my ($table, $chain, $text) = @_;
push(
@{$table->{'rules'}},
{
'text' => $text,
'chain' => $chain,
'index' => scalar(@{$table->{'rules'}}),
}
);
return;
}
# profile_table_name(profile-id)
# Returns an unused default table name for a profile
sub profile_table_name
{
my ($profile) = @_;
my $base = profile_base_table_name($profile);
my @tables = get_nftables_save();
my %used = map { $_->{'family'} eq 'inet' ? ($_->{'name'} => 1) : () } @tables;
my $name = $base;
my $i = 1;
while ($used{$name}) {
$name = $base."_".$i++;
}
return $name;
}
# profile_base_table_name(profile-id)
# Returns the base table name for a profile before uniquifying
sub profile_base_table_name
{
my ($profile) = @_;
my %names = (
'allow_all' => 'profile_allow_all',
'management' => 'profile_management',
'web' => 'profile_web',
'mail' => 'profile_mail',
'dns' => 'profile_dns',
'virtualmin' => 'profile_hosting',
'locked' => 'profile_locked',
'custom' => 'profile_custom',
);
return $names{$profile} || 'profile_custom';
}
# profile_port_set_name(profile, proto, proto-count)
# Returns the set name used for profile-generated service ports
sub profile_port_set_name
{
my ($profile, $proto, $proto_count) = @_;
my $name = profile_base_table_name($profile);
$name .= "_".$proto if ($proto_count && $proto_count > 1);
$name .= "_ports";
$name =~ s/[^\w-]/_/g;
return $name;
}
# default_profile_table_name()
# Returns the default table name for the default profile
sub default_profile_table_name
{
return profile_table_name('virtualmin');
}
# set_type_kind(type)
# Returns addr, port or undef for a set type
sub set_type_kind
{
my ($type) = @_;
return if (!defined($type));
return 'addr' if ($type =~ /addr$/);
return 'port' if ($type =~ /(service|port)$/);
return;
}
# set_type_family(type)
# Returns ip or ip6 for address set types
sub set_type_family
{
my ($type) = @_;
return if (!defined($type));
return 'ip6' if ($type eq 'ipv6_addr');
return 'ip' if ($type eq 'ipv4_addr');
return;
}
# set_name_from_value(value)
# Returns the set name from an @set reference value
sub set_name_from_value
{
my ($val) = @_;
return if (!defined($val));
return $1 if ($val =~ /^\@(\S+)$/);
return;
}
# rule_uses_set(&rule, set-name)
# Returns true if a rule references a set
sub rule_uses_set
{
my ($rule, $setname) = @_;
return 0 if (!$rule || !$setname);
foreach my $k (qw(saddr daddr sport dport)) {
return 1 if (defined($rule->{$k}) && $rule->{$k} eq '@'.$setname);
}
return 1 if ($rule->{'text'} && $rule->{'text'} =~ /\@\Q$setname\E\b/);
return 0;
}
# count_set_references(&table, set-name)
# Returns the number of rules in a table that reference a set
sub count_set_references
{
my ($table, $setname) = @_;
return 0 if (!$table || ref($table) ne 'HASH' || !$setname);
return 0 if (!$table->{'rules'} || ref($table->{'rules'}) ne 'ARRAY');
my $count = 0;
foreach my $r (@{$table->{'rules'}}) {
next if (!$r || ref($r) ne 'HASH');
$count++ if (rule_uses_set($r, $setname));
}
return $count;
}
# validate_set_references(&table)
# Returns an error if any structured rule uses a set in an incompatible field
sub validate_set_references
{
my ($table) = @_;
return if (!$table || ref($table) ne 'HASH');
return if (!$table->{'sets'} || ref($table->{'sets'}) ne 'HASH');
return if (!$table->{'rules'} || ref($table->{'rules'}) ne 'ARRAY');
foreach my $r (@{$table->{'rules'}}) {
next if (!$r || ref($r) ne 'HASH');
foreach my $check (['saddr', 'addr', text('edit_saddr')],
['daddr', 'addr', text('edit_daddr')],
['sport', 'port', text('edit_sport')],
['dport', 'port', text('edit_dport')]) {
my ($field, $want, $label) = @$check;
my $setname = set_name_from_value($r->{$field});
next if (!$setname);
my $set = $table->{'sets'}->{$setname};
next if (!$set);
my $kind = set_type_kind($set->{'type'});
if (!$kind || $kind ne $want) {
my $type = $set->{'type'} || text('set_type_select');
return text(
'apply_esettype', $setname,
nft_table_spec($table), $type,
$r->{'chain'} || "-", $label
);
}
}
}
return;
}
# nftables_save_header()
# Returns the generated-file header for saved rules
sub nftables_save_header
{
return "# This file was auto-generated by the module.\n".
"# Manual changes may be overwritten.\n\n";
}
# dump_nftables_save(@tables)
# Returns a string representation of the firewall rules
sub dump_nftables_save
{
my (@tables) = @_;
my $rv = nftables_save_header();
foreach my $t (@tables) {
if ($t->{'family'}) {
$rv .= "table $t->{'family'} $t->{'name'} {\n";
}
else {
$rv .= "table $t->{'name'} {\n";
}
if ($t->{'sets'} && ref($t->{'sets'}) eq 'HASH') {
foreach my $s (sort keys %{$t->{'sets'}}) {
my $set = $t->{'sets'}->{$s};
next if (!$set || ref($set) ne 'HASH');
$rv .= "\tset $s {\n";
$rv .= "\t\ttype $set->{'type'};\n" if ($set->{'type'});
$rv .= "\t\tflags $set->{'flags'};\n"
if ($set->{'flags'});
if ($set->{'raw_lines'} &&
ref($set->{'raw_lines'}) eq 'ARRAY') {
foreach my $l (@{$set->{'raw_lines'}}) {
next if (!defined($l) || $l eq '');
$rv .= "\t\t$l\n";
}
}
if ($set->{'elements'} &&
ref($set->{'elements'}) eq 'ARRAY' &&
@{$set->{'elements'}}) {
my $el = join(", ", @{$set->{'elements'}});
$rv .= "\t\telements = { $el }\n";
}
$rv .= "\t}\n";
}
}
foreach my $c (keys %{$t->{'chains'}}) {
my $chain = $t->{'chains'}->{$c};
$rv .= "\tchain $c {\n";
if ($chain->{'type'}) {
$rv .=
"\t\ttype $chain->{'type'} hook $chain->{'hook'} priority $chain->{'priority'}; policy $chain->{'policy'};\n";
}
# Add rules for this chain
my @rules = sort { $a->{'index'} <=> $b->{'index'} }
grep { ref($_) eq 'HASH' && $_->{'chain'} eq $c }
@{$t->{'rules'}};
foreach my $r (@rules) {
$rv .= "\t\t$r->{'text'}\n";
}
$rv .= "\t}\n";
}
$rv .= "}\n";
}
return $rv;
}
# write_configuration(@tables)
# Writes the configuration to the save file
sub write_configuration
{
my (@tables) = @_;
my $out = dump_nftables_save(@tables);
my $file = nftables_rules_file();
open_lock_tempfile(my $fh, ">$file");
print_tempfile($fh, $out);
close_tempfile($fh);
sync_managed_metadata(@tables);
update_last_config_change();
return;
}
# save_table(&table)
# Saves a single table to the save file or applies it
sub save_table
{
my ($table) = @_;
# Re-read all tables to ensure we have the full picture if we are overwriting the file
# But here we probably just want to update the specific table in the list of tables we have.
# Since we usually operate on a list of tables, we might need to pass the full list or
# re-read the state.
# For simplicity, we usually load all, modify one, and save all.
}
# save_configuration(@tables)
# Writes the configuration to the save file
sub save_configuration
{
my (@tables) = @_;
write_configuration(@tables);
return;
}
# create_table_configuration(&table, @tables)
# Writes the full configuration after creating a table
sub create_table_configuration
{
my ($table, @tables) = @_;
write_configuration(@tables);
return;
}
# save_table_configuration(&table, @tables)
# Writes the full configuration after changing a table
sub save_table_configuration
{
my ($table, @tables) = @_;
write_configuration(@tables);
return;
}
# delete_table_configuration(&table, @tables)
# Writes the full configuration after deleting a table
sub delete_table_configuration
{
my ($table, @tables) = @_;
write_configuration(@tables);
return;
}
# apply_restore([file])
# Applies Webmin-managed tables from the save file
sub apply_restore
{
my ($file) = @_;
$file ||= nftables_rules_file();
my $cmd = get_nft_command();
return text('index_ecommand', "<tt>nft</tt>") if (!$cmd);
my @tables = get_nftables_save($file);
return text('apply_enone') if (!@tables);
my ($active, $active_err) = get_active_nftables_save();
return $active_err if ($active_err);
my %active;
foreach my $t (@$active) {
$active{table_key($t)} = $t;
}
foreach my $t (@tables) {
my $set_err = validate_set_references($t);
return $set_err if ($set_err);
my $active_table = $active{table_key($t)};
if ($active_table && table_is_externally_managed($active_table)) {
return text('apply_eexternal', nft_table_spec($t));
}
}
my $tmp = tempname();
open_tempfile(my $fh, ">$tmp");
foreach my $t (@tables) {
print_tempfile($fh, "delete table ".nft_table_spec($t)."\n")
if ($active{table_key($t)});
}
print_tempfile($fh, dump_nftables_save(@tables));
close_tempfile($fh);
my $out = backquote_logged("$cmd -c -f $tmp 2>&1");
if (!$?) {
$out = backquote_logged("$cmd -f $tmp 2>&1");
}
unlink_file($tmp);
if ($?) {
return "<pre>$out</pre>";
}
restart_last_restart_time();
return;
}
# delete_active_table(&table)
# Deletes one table from the active ruleset if it exists
sub delete_active_table
{
my ($table) = @_;
my $cmd = get_nft_command();
return text('index_ecommand', "<tt>nft</tt>") if (!$cmd);
my ($active, $active_err) = get_active_nftables_save();
return $active_err if ($active_err);
my $active_table;
foreach my $t (@$active) {
if (table_key($t) eq table_key($table)) {
$active_table = $t;
last;
}
}
return if (!$active_table);
if (table_is_externally_managed($active_table)) {
return text('apply_eexternal', nft_table_spec($table));
}
my $tmp = tempname();
open_tempfile(my $fh, ">$tmp");
print_tempfile($fh, "delete table ".nft_table_spec($table)."\n");
close_tempfile($fh);
my $out = backquote_logged("$cmd -c -f $tmp 2>&1");
if (!$?) {
$out = backquote_logged("$cmd -f $tmp 2>&1");
}
unlink_file($tmp);
if ($?) {
return "<pre>$out</pre>";
}
return;
}
# nft_table_spec(&table)
# Returns a table spec for nft commands
sub nft_table_spec
{
my ($table) = @_;
return $table->{'family'}
? "$table->{'family'} $table->{'name'}"
: $table->{'name'};
}
# table_key(&table)
# Returns a stable key for a table
sub table_key
{
my ($table) = @_;
return ($table->{'family'} || '')."\0".($table->{'name'} || '');
}
# table_is_externally_managed(&table)
# Returns true if an active table is marked as owned by another program
sub table_is_externally_managed
{
my ($table) = @_;
return 0 if (!$table || !$table->{'flags'});
my %flags =
map { $_ => 1 } grep { $_ ne '' } split(/[,\s]+/, $table->{'flags'});
return $flags{'owner'} || $flags{'persist'};
}
# table_is_webmin_managed(&table, [&saved_tables])
# Returns true if an active table is present in Webmin's saved config
sub table_is_webmin_managed
{
my ($table, $saved_tables) = @_;
if (!$saved_tables) {
my @tables = get_nftables_save();
$saved_tables = \@tables;
}
foreach my $t (@$saved_tables) {
return 1 if (table_key($t) eq table_key($table));
}
return 0;
}
# active_table_status(&table, [&saved_tables])
# Returns webmin, external or unclaimed for an active table
sub active_table_status
{
my ($table, $saved_tables) = @_;
return "external" if (table_is_externally_managed($table));
return "webmin" if (table_is_webmin_managed($table, $saved_tables));
return "unclaimed";
}
# managed_metadata_file()
# Returns the path to Webmin's nftables metadata file
sub managed_metadata_file
{
return "$module_config_directory/managed.json";
}
# managed_table_key(&table)
# Returns the key used for managed table metadata
sub managed_table_key
{
my ($table) = @_;
return nft_table_spec($table);
}
# read_managed_metadata()
# Returns metadata about tables managed by this module
sub read_managed_metadata
{
my $file = managed_metadata_file();
return parse_managed_metadata(undef) if (!-r $file);
lock_file($file);
my $json = read_file_contents($file);
unlock_file($file);
return parse_managed_metadata($json);
}
# parse_managed_metadata(json)
# Parses managed table metadata, returning an empty structure on failure
sub parse_managed_metadata
{
my ($json) = @_;
my $meta = eval { convert_from_json($json) };
if (!$meta || ref($meta) ne 'HASH') {
$meta = {};
}
if (!$meta->{'tables'} || ref($meta->{'tables'}) ne 'HASH') {
$meta->{'tables'} = {};
}
return $meta;
}
# sync_managed_metadata(@tables)
# Keeps managed metadata aligned with the saved Webmin config
sub sync_managed_metadata
{
my (@tables) = @_;
my $file = managed_metadata_file();
lock_file($file);
my $meta =
-r $file
? parse_managed_metadata(read_file_contents($file))
: {'tables' => {}};
my %old = %{$meta->{'tables'}};
my %new;
foreach my $t (@tables) {
my $key = managed_table_key($t);
my %entry =
$old{$key} && ref($old{$key}) eq 'HASH' ? %{$old{$key}} : ();
$entry{'family'} = $t->{'family'};
$entry{'name'} = $t->{'name'};
$entry{'source'} ||= 'webmin';
$entry{'managed_at'} ||= time();
$new{$key} = \%entry;
}
$meta->{'tables'} = \%new;
write_file_contents($file, convert_to_json($meta, 1));
unlock_file($file);
return;
}
# register_managed_table(&table, %info)
# Adds or updates metadata for a Webmin-managed table
sub register_managed_table
{
my ($table, %info) = @_;
my $file = managed_metadata_file();
lock_file($file);
my $meta =
-r $file
? parse_managed_metadata(read_file_contents($file))
: {'tables' => {}};
my $key = managed_table_key($table);
my %entry = $meta->{'tables'}->{$key} &&
ref($meta->{'tables'}->{$key}) eq 'HASH'
? %{$meta->{'tables'}->{$key}}
: ();
foreach my $k (keys %info) {
$entry{$k} = $info{$k};
}
$entry{'family'} = $table->{'family'};
$entry{'name'} = $table->{'name'};
$entry{'source'} ||= 'webmin';
$entry{'managed_at'} ||= time();
$meta->{'tables'}->{$key} = \%entry;
write_file_contents($file, convert_to_json($meta, 1));
unlock_file($file);
return;
}
# unregister_managed_table(&table)
# Removes metadata for a table no longer managed by this module
sub unregister_managed_table
{
my ($table) = @_;
my $file = managed_metadata_file();
lock_file($file);
my $meta =
-r $file
? parse_managed_metadata(read_file_contents($file))
: {'tables' => {}};
delete($meta->{'tables'}->{managed_table_key($table)});
write_file_contents($file, convert_to_json($meta, 1));
unlock_file($file);
return;
}
# describe_rule(&rule)
# Returns a human-readable rule summary for listings
sub describe_rule
{
my ($r) = @_;
my @conds;
if ($r->{'iif'}) {
push(@conds, text('index_rule_iif', html_escape($r->{'iif'})));
}
if ($r->{'oif'}) {
push(@conds, text('index_rule_oif', html_escape($r->{'oif'})));
}
if ($r->{'saddr'}) {
push(@conds, text('index_rule_saddr', html_escape($r->{'saddr'})));
}
if ($r->{'daddr'}) {
push(@conds, text('index_rule_daddr', html_escape($r->{'daddr'})));
}
if ($r->{'l4proto'} || ($r->{'proto'} && !$r->{'dport'} && !$r->{'sport'})) {
my $p = $r->{'l4proto'} || $r->{'proto'};
push(@conds, text('index_rule_proto', html_escape($p)));
}
if ($r->{'sport'}) {
push(@conds, text('index_rule_sport', html_escape($r->{'sport'})));
}
if ($r->{'dport'}) {
push(@conds, text('index_rule_dport', html_escape($r->{'dport'})));
}
if ($r->{'icmp_type'}) {
push(@conds, text('index_rule_icmp', html_escape($r->{'icmp_type'})));
}
if ($r->{'icmpv6_type'}) {
push(@conds,
text('index_rule_icmpv6', html_escape($r->{'icmpv6_type'})));
}
if ($r->{'ct_state'}) {
push(@conds, text('index_rule_ct', html_escape($r->{'ct_state'})));
}
if ($r->{'tcp_flags'}) {
my $tf = $r->{'tcp_flags'};
if ($r->{'tcp_flags_mask'}) {
$tf = $r->{'tcp_flags_mask'}."==".$r->{'tcp_flags'};
}
push(@conds, text('index_rule_tcpflags', html_escape($tf)));
}
if ($r->{'limit_rate'}) {
my $lim = $r->{'limit_rate'};
if ($r->{'limit_burst'}) {
$lim .= " burst ".$r->{'limit_burst'};
}
push(@conds, text('index_rule_limit', html_escape($lim)));
}
if ($r->{'log_prefix'}) {
push(@conds,
text('index_rule_log_prefix', html_escape($r->{'log_prefix'})));
}
if ($r->{'log_level'}) {
push(@conds,
text('index_rule_log_level', html_escape($r->{'log_level'})));
}
if ($r->{'log'} && !$r->{'log_prefix'} && !$r->{'log_level'}) {
push(@conds, text('index_rule_log'));
}
if ($r->{'counter'}) {
push(@conds, text('index_rule_counter'));
}
my $action_label;
if ($r->{'jump'}) {
$action_label = text('index_rule_jump', html_escape($r->{'jump'}));
}
elsif ($r->{'goto'}) {
$action_label = text('index_rule_goto', html_escape($r->{'goto'}));
}
elsif ($r->{'action'}) {
if ($r->{'action'} eq 'return') {
$action_label = text('index_return_action');
}
elsif ($r->{'action'} eq 'redirect') {
my $target = format_nat_target($r->{'nat_addr'}, $r->{'nat_port'});
$action_label = $target ne '' ?
text('index_redirect_to', html_escape($target)) :
text('index_redirect');
}
elsif ($r->{'action'} eq 'dnat') {
my $target = format_nat_target($r->{'nat_addr'}, $r->{'nat_port'});
$action_label = $target ne '' ?
text('index_dnat_to', html_escape($target)) :
text('index_dnat');
}
else {
$action_label = text('index_'.lc($r->{'action'}));
}
}
if ($action_label) {
if (@conds) {
return text('index_rule_desc_generic', $action_label,
join(", ", @conds));
}
return text('index_rule_desc_action', $action_label);
}
return html_escape($r->{'text'});
}
# interface_choice(name, value, blanktext)
# Returns HTML for an interface chooser menu
sub interface_choice
{
my ($name, $value, $blanktext) = @_;
if (foreign_check("net")) {
foreign_require("net", "net-lib.pl");
return net::interface_choice($name, $value, $blanktext, 0, 1);
}
else {
return ui_textbox($name, $value, 20);
}
}
# get_webmin_port()
# Returns the configured Webmin port, or 10000 if unknown
sub get_webmin_port
{
my %miniserv;
if (get_miniserv_config(\%miniserv) && $miniserv{'port'} =~ /^\d+$/) {
return $miniserv{'port'};
}
return 10000;
}
# get_usermin_port()
# Returns the configured Usermin port, or 20000 if unknown
sub get_usermin_port
{
my %miniserv;
if (foreign_installed("usermin")) {
foreign_require("usermin", "usermin-lib.pl");
usermin::get_usermin_miniserv_config(\%miniserv);
if ($miniserv{'port'} =~ /^\d+$/) {
return $miniserv{'port'};
}
}
return 20000;
}
1;