mirror of
https://github.com/webmin/webmin.git
synced 2026-06-10 06:40:32 +01:00
4080 lines
107 KiB
Perl
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;
|