#!/usr/bin/env python3 # # Copyright (C) 2021-2022 VyOS maintainers and contributors # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License version 2 or later as # published by the Free Software Foundation. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see <http://www.gnu.org/licenses/>. import os from json import loads from sys import exit from vyos.config import Config from vyos.configdict import dict_merge from vyos.template import render from vyos.util import cmd from vyos.util import dict_search_args from vyos.util import run from vyos.xml import defaults from vyos import ConfigError from vyos import airbag airbag.enable() nftables_conf = '/run/nftables_zone.conf' def get_config(config=None): if config: conf = config else: conf = Config() base = ['zone-policy'] zone_policy = conf.get_config_dict(base, key_mangling=('-', '_'), get_first_key=True, no_tag_node_value_mangle=True) zone_policy['firewall'] = conf.get_config_dict(['firewall'], key_mangling=('-', '_'), get_first_key=True, no_tag_node_value_mangle=True) if 'zone' in zone_policy: # We have gathered the dict representation of the CLI, but there are default # options which we need to update into the dictionary retrived. default_values = defaults(base + ['zone']) for zone in zone_policy['zone']: zone_policy['zone'][zone] = dict_merge(default_values, zone_policy['zone'][zone]) return zone_policy def verify(zone_policy): # bail out early - looks like removal from running config if not zone_policy: return None local_zone = False interfaces = [] if 'zone' in zone_policy: for zone, zone_conf in zone_policy['zone'].items(): if 'local_zone' not in zone_conf and 'interface' not in zone_conf: raise ConfigError(f'Zone "{zone}" has no interfaces and is not the local zone') if 'local_zone' in zone_conf: if local_zone: raise ConfigError('There cannot be multiple local zones') if 'interface' in zone_conf: raise ConfigError('Local zone cannot have interfaces assigned') if 'intra_zone_filtering' in zone_conf: raise ConfigError('Local zone cannot use intra-zone-filtering') local_zone = True if 'interface' in zone_conf: found_duplicates = [intf for intf in zone_conf['interface'] if intf in interfaces] if found_duplicates: raise ConfigError(f'Interfaces cannot be assigned to multiple zones') interfaces += zone_conf['interface'] if 'intra_zone_filtering' in zone_conf: intra_zone = zone_conf['intra_zone_filtering'] if len(intra_zone) > 1: raise ConfigError('Only one intra-zone-filtering action must be specified') if 'firewall' in intra_zone: v4_name = dict_search_args(intra_zone, 'firewall', 'name') if v4_name and not dict_search_args(zone_policy, 'firewall', 'name', v4_name): raise ConfigError(f'Firewall name "{v4_name}" does not exist') v6_name = dict_search_args(intra_zone, 'firewall', 'ipv6-name') if v6_name and not dict_search_args(zone_policy, 'firewall', 'ipv6-name', v6_name): raise ConfigError(f'Firewall ipv6-name "{v6_name}" does not exist') if not v4_name and not v6_name: raise ConfigError('No firewall names specified for intra-zone-filtering') if 'from' in zone_conf: for from_zone, from_conf in zone_conf['from'].items(): if from_zone not in zone_policy['zone']: raise ConfigError(f'Zone "{zone}" refers to a non-existent or deleted zone "{from_zone}"') v4_name = dict_search_args(from_conf, 'firewall', 'name') if v4_name: if 'name' not in zone_policy['firewall']: raise ConfigError(f'Firewall name "{v4_name}" does not exist') if not dict_search_args(zone_policy, 'firewall', 'name', v4_name): raise ConfigError(f'Firewall name "{v4_name}" does not exist') v6_name = dict_search_args(from_conf, 'firewall', 'v6_name') if v6_name: if 'ipv6_name' not in zone_policy['firewall']: raise ConfigError(f'Firewall ipv6-name "{v6_name}" does not exist') if not dict_search_args(zone_policy, 'firewall', 'ipv6_name', v6_name): raise ConfigError(f'Firewall ipv6-name "{v6_name}" does not exist') return None def has_ipv4_fw(zone_conf): if 'from' not in zone_conf: return False zone_from = zone_conf['from'] return any([True for fz in zone_from if dict_search_args(zone_from, fz, 'firewall', 'name')]) def has_ipv6_fw(zone_conf): if 'from' not in zone_conf: return False zone_from = zone_conf['from'] return any([True for fz in zone_from if dict_search_args(zone_from, fz, 'firewall', 'ipv6_name')]) def get_local_from(zone_policy, local_zone_name): # Get all zone firewall names from the local zone out = {} for zone, zone_conf in zone_policy['zone'].items(): if zone == local_zone_name: continue if 'from' not in zone_conf: continue if local_zone_name in zone_conf['from']: out[zone] = zone_conf['from'][local_zone_name] return out def cleanup_commands(): commands = [] for table in ['ip filter', 'ip6 filter']: json_str = cmd(f'nft -t -j list table {table}') obj = loads(json_str) if 'nftables' not in obj: continue for item in obj['nftables']: if 'rule' in item: chain = item['rule']['chain'] handle = item['rule']['handle'] if 'expr' not in item['rule']: continue for expr in item['rule']['expr']: target = dict_search_args(expr, 'jump', 'target') if not target: continue if target.startswith("VZONE") or target.startswith("VYOS_STATE_POLICY"): commands.append(f'delete rule {table} {chain} handle {handle}') for item in obj['nftables']: if 'chain' in item: if item['chain']['name'].startswith("VZONE"): chain = item['chain']['name'] commands.append(f'delete chain {table} {chain}') return commands def generate(zone_policy): data = zone_policy or {} if os.path.exists(nftables_conf): # Check to see if we've run before data['cleanup_commands'] = cleanup_commands() if 'zone' in data: for zone, zone_conf in data['zone'].items(): zone_conf['ipv4'] = has_ipv4_fw(zone_conf) zone_conf['ipv6'] = has_ipv6_fw(zone_conf) if 'local_zone' in zone_conf: zone_conf['from_local'] = get_local_from(data, zone) render(nftables_conf, 'zone_policy/nftables.j2', data) return None def apply(zone_policy): install_result = run(f'nft -f {nftables_conf}') if install_result != 0: raise ConfigError('Failed to apply zone-policy') return None if __name__ == '__main__': try: c = get_config() verify(c) generate(c) apply(c) except ConfigError as e: print(e) exit(1)