diff options
| -rw-r--r-- | data/templates/dns-dynamic/ddclient.conf.j2 | 7 | ||||
| -rw-r--r-- | interface-definitions/include/version/dns-dynamic-version.xml.i | 2 | ||||
| -rw-r--r-- | interface-definitions/service_dns_dynamic.xml.in | 47 | ||||
| -rw-r--r-- | op-mode-definitions/container.xml.in | 3 | ||||
| -rw-r--r-- | op-mode-definitions/dns-dynamic.xml.in | 25 | ||||
| -rw-r--r-- | op-mode-definitions/dns-forwarding.xml.in | 2 | ||||
| -rw-r--r-- | python/vyos/opmode.py | 5 | ||||
| -rwxr-xr-x | smoketest/scripts/cli/test_service_dns_dynamic.py | 42 | ||||
| -rwxr-xr-x | src/conf_mode/service_dns_dynamic.py | 49 | ||||
| -rwxr-xr-x | src/migration-scripts/dns-dynamic/3-to-4 | 76 | ||||
| -rwxr-xr-x | src/op_mode/dns.py | 128 | ||||
| -rwxr-xr-x | src/op_mode/dns_dynamic.py | 113 | 
12 files changed, 272 insertions, 227 deletions
| diff --git a/data/templates/dns-dynamic/ddclient.conf.j2 b/data/templates/dns-dynamic/ddclient.conf.j2 index 6c0653a55..5538ea56c 100644 --- a/data/templates/dns-dynamic/ddclient.conf.j2 +++ b/data/templates/dns-dynamic/ddclient.conf.j2 @@ -7,7 +7,7 @@ use{{ ipv }}={{ address if address == 'web' else 'if' }}{{ ipv }}, \  web{{ ipv }}={{ web_options.url }}, \  {%         endif %}  {%         if web_options.skip is vyos_defined %} -web-skip{{ ipv }}='{{ web_options.skip }}', \ +web{{ ipv }}-skip='{{ web_options.skip }}', \  {%         endif %}  {%     else %}  if{{ ipv }}={{ address }}, \ @@ -45,9 +45,12 @@ use=no                                         else ['']) %}  {%             set password = config.key if config.protocol == 'nsupdate'                                else config.password %} +{%             set address = 'web' if config.address.web is vyos_defined +                              else config.address.interface %} +{%             set web_options = config.address.web | default({}) %}  # Web service dynamic DNS configuration for {{ service }}: [{{ config.protocol }}, {{ host }}] -{{ render_config(host, config.address, config.web_options, ip_suffixes, +{{ render_config(host, address, web_options, ip_suffixes,                   protocol=config.protocol, server=config.server, zone=config.zone,                   login=config.username, password=password, ttl=config.ttl,                   min_interval=config.wait_time, max_interval=config.expiry_time) }} diff --git a/interface-definitions/include/version/dns-dynamic-version.xml.i b/interface-definitions/include/version/dns-dynamic-version.xml.i index 773a6ab51..346385ccb 100644 --- a/interface-definitions/include/version/dns-dynamic-version.xml.i +++ b/interface-definitions/include/version/dns-dynamic-version.xml.i @@ -1,3 +1,3 @@  <!-- include start from include/version/dns-dynamic-version.xml.i --> -<syntaxVersion component='dns-dynamic' version='3'></syntaxVersion> +<syntaxVersion component='dns-dynamic' version='4'></syntaxVersion>  <!-- include end --> diff --git a/interface-definitions/service_dns_dynamic.xml.in b/interface-definitions/service_dns_dynamic.xml.in index d1b0e90bb..75e5520b7 100644 --- a/interface-definitions/service_dns_dynamic.xml.in +++ b/interface-definitions/service_dns_dynamic.xml.in @@ -38,42 +38,29 @@                        </constraint>                      </properties>                    </leafNode> -                  <leafNode name="address"> +                  <node name="address">                      <properties>                        <help>Obtain IP address to send Dynamic DNS update for</help> -                      <valueHelp> -                        <format>txt</format> -                        <description>Use interface to obtain the IP address</description> -                      </valueHelp> -                      <valueHelp> -                        <format>web</format> -                        <description>Use HTTP(S) web request to obtain the IP address</description> -                      </valueHelp> -                      <completionHelp> -                        <script>${vyos_completion_dir}/list_interfaces</script> -                        <list>web</list> -                      </completionHelp> -                      <constraint> -                        #include <include/constraint/interface-name.xml.i> -                        <regex>web</regex> -                      </constraint> -                    </properties> -                  </leafNode> -                  <node name="web-options"> -                    <properties> -                      <help>Options when using HTTP(S) web request to obtain the IP address</help>                      </properties>                      <children> -                      #include <include/url-http-https.xml.i> -                      <leafNode name="skip"> +                      #include <include/generic-interface.xml.i> +                      <node name="web">                          <properties> -                          <help>Pattern to skip from the HTTP(S) respose</help> -                          <valueHelp> -                            <format>txt</format> -                            <description>Pattern to skip from the HTTP(S) respose to extract the external IP address</description> -                          </valueHelp> +                          <help>HTTP(S) web request to use</help>                          </properties> -                      </leafNode> +                        <children> +                          #include <include/url-http-https.xml.i> +                          <leafNode name="skip"> +                            <properties> +                              <help>Pattern to skip from the HTTP(S) respose</help> +                              <valueHelp> +                                <format>txt</format> +                                <description>Pattern to skip from the HTTP(S) respose to extract the external IP address</description> +                              </valueHelp> +                            </properties> +                          </leafNode> +                        </children> +                      </node>                      </children>                    </node>                    <leafNode name="ip-version"> diff --git a/op-mode-definitions/container.xml.in b/op-mode-definitions/container.xml.in index f581d39fa..96c582a83 100644 --- a/op-mode-definitions/container.xml.in +++ b/op-mode-definitions/container.xml.in @@ -154,6 +154,9 @@      </children>    </node>    <node name="update"> +    <properties> +      <help>Update data for a service</help> +    </properties>      <children>        <node name="container">          <properties> diff --git a/op-mode-definitions/dns-dynamic.xml.in b/op-mode-definitions/dns-dynamic.xml.in index 79478f392..45d58e2e8 100644 --- a/op-mode-definitions/dns-dynamic.xml.in +++ b/op-mode-definitions/dns-dynamic.xml.in @@ -4,7 +4,7 @@      <children>        <node name="dns">          <properties> -          <help>Clear Domain Name System</help> +          <help>Clear Domain Name System (DNS) related service state</help>          </properties>          <children>            <node name="dynamic"> @@ -30,7 +30,7 @@          <children>            <node name="dns">              <properties> -              <help>Monitor last lines of Domain Name System related services</help> +              <help>Monitor last lines of Domain Name System (DNS) related services</help>              </properties>              <children>                <node name="dynamic"> @@ -51,7 +51,7 @@          <children>            <node name="dns">              <properties> -              <help>Show log for Domain Name System related services</help> +              <help>Show log for Domain Name System (DNS) related services</help>              </properties>              <children>                <node name="dynamic"> @@ -66,7 +66,7 @@        </node>        <node name="dns">          <properties> -          <help>Show Domain Name System related information</help> +          <help>Show Domain Name System (DNS) related information</help>          </properties>          <children>            <node name="dynamic"> @@ -78,7 +78,7 @@                  <properties>                    <help>Show Dynamic DNS status</help>                  </properties> -                <command>sudo ${vyos_op_scripts_dir}/dns_dynamic.py --status</command> +                <command>sudo ${vyos_op_scripts_dir}/dns.py show_dynamic_status</command>                </leafNode>              </children>            </node> @@ -90,34 +90,31 @@      <children>        <node name="dns">          <properties> -          <help>Restart specific Domain Name System related service</help> +          <help>Restart specific Domain Name System (DNS) related service</help>          </properties>          <children>            <node name="dynamic">              <properties>                <help>Restart Dynamic DNS service</help>              </properties> -            <command>sudo ${vyos_op_scripts_dir}/dns_dynamic.py --update</command> +            <command>if cli-shell-api existsActive service dns dynamic; then sudo systemctl restart ddclient.service; else echo "Dynamic DNS not configured"; fi</command>            </node>          </children>        </node>      </children>    </node> -  <node name="update"> -    <properties> -      <help>Update data for a service</help> -    </properties> +  <node name="reset">      <children>        <node name="dns">          <properties> -          <help>Update Domain Name System related information</help> +          <help>Reset Domain Name System (DNS) related service state</help>          </properties>          <children>            <node name="dynamic">              <properties> -              <help>Update Dynamic DNS information</help> +              <help>Reset Dynamic DNS information</help>              </properties> -            <command>sudo ${vyos_op_scripts_dir}/dns_dynamic.py --update</command> +            <command>sudo ${vyos_op_scripts_dir}/dns.py reset_dynamic</command>            </node>          </children>        </node> diff --git a/op-mode-definitions/dns-forwarding.xml.in b/op-mode-definitions/dns-forwarding.xml.in index ebedae6eb..29bfc61cf 100644 --- a/op-mode-definitions/dns-forwarding.xml.in +++ b/op-mode-definitions/dns-forwarding.xml.in @@ -11,7 +11,7 @@              <children>                <node name="forwarding">                  <properties> -                  <help>Monitor last lines of DNS Forwarding</help> +                  <help>Monitor last lines of DNS Forwarding service</help>                  </properties>                  <command>journalctl --no-hostname --follow --boot --unit pdns-recursor.service</command>                </node> diff --git a/python/vyos/opmode.py b/python/vyos/opmode.py index 230a85541..e1af1a682 100644 --- a/python/vyos/opmode.py +++ b/python/vyos/opmode.py @@ -1,4 +1,4 @@ -# Copyright 2022-2023 VyOS maintainers and contributors <maintainers@vyos.io> +# Copyright 2022-2024 VyOS maintainers and contributors <maintainers@vyos.io>  #  # This library is free software; you can redistribute it and/or  # modify it under the terms of the GNU Lesser General Public @@ -81,7 +81,7 @@ class InternalError(Error):  def _is_op_mode_function_name(name): -    if re.match(r"^(show|clear|reset|restart|add|delete|generate|set)", name): +    if re.match(r"^(show|clear|reset|restart|add|update|delete|generate|set)", name):          return True      else:          return False @@ -275,4 +275,3 @@ def run(module):          # Other functions should not return anything,          # although they may print their own warnings or status messages          func(**args) - diff --git a/smoketest/scripts/cli/test_service_dns_dynamic.py b/smoketest/scripts/cli/test_service_dns_dynamic.py index ae46b18ba..c39d4467a 100755 --- a/smoketest/scripts/cli/test_service_dns_dynamic.py +++ b/smoketest/scripts/cli/test_service_dns_dynamic.py @@ -1,6 +1,6 @@  #!/usr/bin/env python3  # -# Copyright (C) 2019-2023 VyOS maintainers and contributors +# Copyright (C) 2019-2024 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 @@ -62,7 +62,7 @@ class TestServiceDDNS(VyOSUnitTestSHIM.TestCase):                      'zoneedit': {'protocol': 'zoneedit1', 'username': username}}          for svc, details in services.items(): -            self.cli_set(name_path + [svc, 'address', interface]) +            self.cli_set(name_path + [svc, 'address', 'interface', interface])              self.cli_set(name_path + [svc, 'host-name', hostname])              self.cli_set(name_path + [svc, 'password', password])              for opt, value in details.items(): @@ -118,7 +118,7 @@ class TestServiceDDNS(VyOSUnitTestSHIM.TestCase):          expiry_time_bad = '360'          self.cli_set(base_path + ['interval', interval]) -        self.cli_set(svc_path + ['address', interface]) +        self.cli_set(svc_path + ['address', 'interface', interface])          self.cli_set(svc_path + ['ip-version', ip_version])          self.cli_set(svc_path + ['protocol', proto])          self.cli_set(svc_path + ['server', server]) @@ -156,7 +156,7 @@ class TestServiceDDNS(VyOSUnitTestSHIM.TestCase):          ip_version = 'both'          for name, details in services.items(): -            self.cli_set(name_path + [name, 'address', interface]) +            self.cli_set(name_path + [name, 'address', 'interface', interface])              self.cli_set(name_path + [name, 'host-name', hostname])              self.cli_set(name_path + [name, 'password', password])              for opt, value in details.items(): @@ -201,7 +201,7 @@ class TestServiceDDNS(VyOSUnitTestSHIM.TestCase):          with tempfile.NamedTemporaryFile(prefix='/config/auth/') as key_file:              key_file.write(b'S3cretKey') -            self.cli_set(svc_path + ['address', interface]) +            self.cli_set(svc_path + ['address', 'interface', interface])              self.cli_set(svc_path + ['protocol', proto])              self.cli_set(svc_path + ['server', server])              self.cli_set(svc_path + ['zone', zone]) @@ -229,7 +229,7 @@ class TestServiceDDNS(VyOSUnitTestSHIM.TestCase):          hostnames = ['@', 'www', hostname, f'@.{hostname}']          for name in hostnames: -            self.cli_set(svc_path + ['address', interface]) +            self.cli_set(svc_path + ['address', 'interface', interface])              self.cli_set(svc_path + ['protocol', proto])              self.cli_set(svc_path + ['server', server])              self.cli_set(svc_path + ['username', username]) @@ -251,38 +251,38 @@ class TestServiceDDNS(VyOSUnitTestSHIM.TestCase):          # Check if DDNS service can be configured and runs          svc_path = name_path + ['cloudflare']          proto = 'cloudflare' -        web_url_good = 'https://ifconfig.me/ip' -        web_url_bad = 'http:/ifconfig.me/ip' +        web_url = 'https://ifconfig.me/ip' +        web_skip = 'Current IP Address:'          self.cli_set(svc_path + ['protocol', proto])          self.cli_set(svc_path + ['zone', zone])          self.cli_set(svc_path + ['password', password])          self.cli_set(svc_path + ['host-name', hostname]) -        self.cli_set(svc_path + ['web-options', 'url', web_url_good]) -        # web-options is supported only with web service based address lookup -        # exception is raised for interface based address lookup +        # not specifying either 'interface' or 'web' will raise an exception          with self.assertRaises(ConfigSessionError): -            self.cli_set(svc_path + ['address', interface])              self.cli_commit()          self.cli_set(svc_path + ['address', 'web']) -        # commit changes +        # specifying both 'interface' and 'web' will raise an exception as well +        with self.assertRaises(ConfigSessionError): +            self.cli_set(svc_path + ['address', 'interface', interface]) +            self.cli_commit() +        self.cli_delete(svc_path + ['address', 'interface'])          self.cli_commit() -        # web-options must be a valid URL +        # web option 'skip' is useless without the option 'url'          with self.assertRaises(ConfigSessionError): -            self.cli_set(svc_path + ['web-options', 'url', web_url_bad]) +            self.cli_set(svc_path + ['address', 'web', 'skip', web_skip])              self.cli_commit() -        self.cli_set(svc_path + ['web-options', 'url', web_url_good]) - -        # commit changes +        self.cli_set(svc_path + ['address', 'web', 'url', web_url])          self.cli_commit()          # Check the generating config parameters          ddclient_conf = cmd(f'sudo cat {DDCLIENT_CONF}')          self.assertIn(f'usev4=webv4', ddclient_conf) -        self.assertIn(f'webv4={web_url_good}', ddclient_conf) +        self.assertIn(f'webv4={web_url}', ddclient_conf) +        self.assertIn(f'webv4-skip=\'{web_skip}\'', ddclient_conf)          self.assertIn(f'protocol={proto}', ddclient_conf)          self.assertIn(f'zone={zone}', ddclient_conf)          self.assertIn(f'password=\'{password}\'', ddclient_conf) @@ -294,7 +294,7 @@ class TestServiceDDNS(VyOSUnitTestSHIM.TestCase):          proto = 'namecheap'          dyn_interface = 'pppoe587' -        self.cli_set(svc_path + ['address', dyn_interface]) +        self.cli_set(svc_path + ['address', 'interface', dyn_interface])          self.cli_set(svc_path + ['protocol', proto])          self.cli_set(svc_path + ['server', server])          self.cli_set(svc_path + ['username', username]) @@ -327,7 +327,7 @@ class TestServiceDDNS(VyOSUnitTestSHIM.TestCase):          self.cli_set(['vrf', 'name', vrf_name, 'table', vrf_table])          self.cli_set(base_path + ['vrf', vrf_name]) -        self.cli_set(svc_path + ['address', interface]) +        self.cli_set(svc_path + ['address', 'interface', interface])          self.cli_set(svc_path + ['protocol', proto])          self.cli_set(svc_path + ['host-name', hostname])          self.cli_set(svc_path + ['zone', zone]) diff --git a/src/conf_mode/service_dns_dynamic.py b/src/conf_mode/service_dns_dynamic.py index 845aaa1b5..a551a9891 100755 --- a/src/conf_mode/service_dns_dynamic.py +++ b/src/conf_mode/service_dns_dynamic.py @@ -1,6 +1,6 @@  #!/usr/bin/env python3  # -# Copyright (C) 2018-2023 VyOS maintainers and contributors +# Copyright (C) 2018-2024 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 @@ -87,31 +87,36 @@ def verify(dyndns):              if field not in config:                  raise ConfigError(f'"{field.replace("_", "-")}" {error_msg_req}') -        # If dyndns address is an interface, ensure -        # that the interface exists (or just warn if dynamic interface) -        # and that web-options are not set -        if config['address'] != 'web': +        if not any(x in config['address'] for x in ['interface', 'web']): +            raise ConfigError(f'Either "interface" or "web" {error_msg_req} ' +                              f'with protocol "{config["protocol"]}"') +        if all(x in config['address'] for x in ['interface', 'web']): +            raise ConfigError(f'Both "interface" and "web" at the same time {error_msg_uns} ' +                              f'with protocol "{config["protocol"]}"') + +        # If dyndns address is an interface, ensure that the interface exists +        # and warn if a non-active dynamic interface is used +        if 'interface' in config['address']:              tmp = re.compile(dynamic_interface_pattern)              # exclude check interface for dynamic interfaces -            if tmp.match(config["address"]): -                if not interface_exists(config["address"]): -                    Warning(f'Interface "{config["address"]}" does not exist yet and cannot ' -                            f'be used for Dynamic DNS service "{service}" until it is up!') +            if tmp.match(config['address']['interface']): +                if not interface_exists(config['address']['interface']): +                    Warning(f'Interface "{config["address"]["interface"]}" does not exist yet and ' +                            f'cannot be used for Dynamic DNS service "{service}" until it is up!')              else: -                verify_interface_exists(config['address']) -            if 'web_options' in config: -                raise ConfigError(f'"web-options" is applicable only when using HTTP(S) ' -                                  f'web request to obtain the IP address') - -        # Warn if using checkip.dyndns.org, as it does not support HTTPS -        # See: https://github.com/ddclient/ddclient/issues/597 -        if 'web_options' in config: -            if 'url' not in config['web_options']: -                raise ConfigError(f'"url" in "web-options" {error_msg_req} ' +                verify_interface_exists(config['address']['interface']) + +        if 'web' in config['address']: +            # If 'skip' is specified, 'url' is required as well +            if 'skip' in config['address']['web'] and 'url' not in config['address']['web']: +                raise ConfigError(f'"url" along with "skip" {error_msg_req} '                                    f'with protocol "{config["protocol"]}"') -            elif re.search("^(https?://)?checkip\.dyndns\.org", config['web_options']['url']): -                Warning(f'"checkip.dyndns.org" does not support HTTPS requests for IP address ' -                        f'lookup. Please use a different IP address lookup service.') +            if 'url' in config['address']['web']: +                # Warn if using checkip.dyndns.org, as it does not support HTTPS +                # See: https://github.com/ddclient/ddclient/issues/597 +                if re.search("^(https?://)?checkip\.dyndns\.org", config['address']['web']['url']): +                    Warning(f'"checkip.dyndns.org" does not support HTTPS requests for IP address ' +                            f'lookup. Please use a different IP address lookup service.')          # RFC2136 uses 'key' instead of 'password'          if config['protocol'] != 'nsupdate' and 'password' not in config: diff --git a/src/migration-scripts/dns-dynamic/3-to-4 b/src/migration-scripts/dns-dynamic/3-to-4 new file mode 100755 index 000000000..b888a3b6b --- /dev/null +++ b/src/migration-scripts/dns-dynamic/3-to-4 @@ -0,0 +1,76 @@ +#!/usr/bin/env python3 + +# Copyright (C) 2024 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/>. + +# T5966: +# - migrate "service dns dynamic name <service> address <interface>" +#        to "service dns dynamic name <service> address interface <interface>" +#      when <interface> != 'web' +# - migrate "service dns dynamic name <service> web-options ..." +#        to "service dns dynamic name <service> address web ..." +#      when <interface> == 'web' + +import sys +from vyos.configtree import ConfigTree + +if len(sys.argv) < 2: +    print("Must specify file name!") +    sys.exit(1) + +file_name = sys.argv[1] + +with open(file_name, 'r') as f: +    config_file = f.read() + +config = ConfigTree(config_file) + +base_path = ['service', 'dns', 'dynamic', 'name'] + +if not config.exists(base_path): +    # Nothing to do +    sys.exit(0) + +for service in config.list_nodes(base_path): + +    service_path = base_path + [service] + +    if config.exists(service_path + ['address']): +        address = config.return_value(service_path + ['address']) +        # 'address' is not a leaf node anymore, delete it first +        config.delete(service_path + ['address']) + +        # When address is an interface (not 'web'), move it to 'address interface' +        if address != 'web': +            config.set(service_path + ['address', 'interface'], address) + +        else: # address == 'web' +            # Relocate optional 'web-options' directly under 'address web' +            if config.exists(service_path + ['web-options']): +                # config.copy does not recursively create a path, so initialize it +                config.set(service_path + ['address']) +                config.copy(service_path + ['web-options'], +                            service_path + ['address', 'web']) +                config.delete(service_path + ['web-options']) + +            # ensure that valueless 'address web' still exists even if there are no 'web-options' +            if not config.exists(service_path + ['address', 'web']): +                config.set(service_path + ['address', 'web']) + +try: +    with open(file_name, 'w') as f: +        f.write(config.to_string()) +except OSError as e: +    print("Failed to save the modified config: {}".format(e)) +    sys.exit(1) diff --git a/src/op_mode/dns.py b/src/op_mode/dns.py index 309bef3b9..16c462f23 100755 --- a/src/op_mode/dns.py +++ b/src/op_mode/dns.py @@ -15,14 +15,33 @@  # along with this program.  If not, see <http://www.gnu.org/licenses/>. -import typing +import os  import sys +import time +import typing  import vyos.opmode  from tabulate import tabulate  from vyos.configquery import ConfigTreeQuery  from vyos.utils.process import cmd, rc_cmd +from vyos.template import is_ipv4, is_ipv6 + +_dynamic_cache_file = r'/run/ddclient/ddclient.cache' + +_dynamic_status_columns = { +    'host':        'Hostname', +    'ipv4':        'IPv4 address', +    'status-ipv4': 'IPv4 status', +    'ipv6':        'IPv6 address', +    'status-ipv6': 'IPv6 status', +    'mtime':       'Last update', +} +_forwarding_statistics_columns = { +    'cache-entries':     'Cache entries', +    'max-cache-entries': 'Max cache entries', +    'cache-size':        'Cache size', +}  def _forwarding_data_to_dict(data, sep="\t") -> dict:      """ @@ -50,37 +69,106 @@ def _forwarding_data_to_dict(data, sep="\t") -> dict:              dictionary[key] = value      return dictionary +def _get_dynamic_host_records_raw() -> dict: + +    data = [] + +    if os.path.isfile(_dynamic_cache_file): # A ddclient status file might not always exist +        with open(_dynamic_cache_file, 'r') as f: +            for line in f: +                if line.startswith('#'): +                    continue + +                props = {} +                # ddclient cache rows have properties in 'key=value' format separated by comma +                # we pick up the ones we are interested in +                for kvraw in line.split(' ')[0].split(','): +                    k, v = kvraw.split('=') +                    if k in list(_dynamic_status_columns.keys()) + ['ip', 'status']:  # ip and status are legacy keys +                        props[k] = v + +                # Extract IPv4 and IPv6 address and status from legacy keys +                # Dual-stack isn't supported in legacy format, 'ip' and 'status' are for one of IPv4 or IPv6 +                if 'ip' in props: +                    if is_ipv4(props['ip']): +                        props['ipv4'] = props['ip'] +                        props['status-ipv4'] = props['status'] +                    elif is_ipv6(props['ip']): +                        props['ipv6'] = props['ip'] +                        props['status-ipv6'] = props['status'] +                    del props['ip'] + +                # Convert mtime to human readable format +                if 'mtime' in props: +                    props['mtime'] = time.strftime( +                        "%Y-%m-%d %H:%M:%S", time.localtime(int(props['mtime'], base=10))) + +                data.append(props) + +    return data + +def _get_dynamic_host_records_formatted(data): +    data_entries = [] +    for entry in data: +        data_entries.append([entry.get(key) for key in _dynamic_status_columns.keys()]) +    header = _dynamic_status_columns.values() +    output = tabulate(data_entries, header, numalign='left') +    return output  def _get_forwarding_statistics_raw() -> dict:      command = cmd('rec_control get-all')      data = _forwarding_data_to_dict(command) -    data['cache-size'] = "{0:.2f}".format( int( +    data['cache-size'] = "{0:.2f} kbytes".format( int(          cmd('rec_control get cache-bytes')) / 1024 )      return data -  def _get_forwarding_statistics_formatted(data): -    cache_entries = data.get('cache-entries') -    max_cache_entries = data.get('max-cache-entries') -    cache_size = data.get('cache-size') -    data_entries = [[cache_entries, max_cache_entries, f'{cache_size} kbytes']] -    headers = ["Cache entries", "Max cache entries" , "Cache size"] -    output = tabulate(data_entries, headers, numalign="left") +    data_entries = [] +    data_entries.append([data.get(key) for key in _forwarding_statistics_columns.keys()]) +    header = _forwarding_statistics_columns.values() +    output = tabulate(data_entries, header, numalign='left')      return output -def _verify_forwarding(func): -    """Decorator checks if DNS Forwarding config exists""" +def _verify(target): +    """Decorator checks if config for DNS related service exists"""      from functools import wraps -    @wraps(func) -    def _wrapper(*args, **kwargs): -        config = ConfigTreeQuery() -        if not config.exists('service dns forwarding'): -            raise vyos.opmode.UnconfiguredSubsystem('DNS Forwarding is not configured') -        return func(*args, **kwargs) -    return _wrapper +    if target not in ['dynamic', 'forwarding']: +        raise ValueError('Invalid target') + +    def _verify_target(func): +        @wraps(func) +        def _wrapper(*args, **kwargs): +            config = ConfigTreeQuery() +            if not config.exists(f'service dns {target}'): +                _prefix = f'Dynamic DNS' if target == 'dynamic' else 'DNS Forwarding' +                raise vyos.opmode.UnconfiguredSubsystem(f'{_prefix} is not configured') +            return func(*args, **kwargs) +        return _wrapper +    return _verify_target + +@_verify('dynamic') +def show_dynamic_status(raw: bool): +    host_data = _get_dynamic_host_records_raw() +    if raw: +        return host_data +    else: +        return _get_dynamic_host_records_formatted(host_data) -@_verify_forwarding +@_verify('dynamic') +def reset_dynamic(): +    """ +    Reset Dynamic DNS cache +    """ +    if os.path.exists(_dynamic_cache_file): +        os.remove(_dynamic_cache_file) +    rc, output = rc_cmd('systemctl restart ddclient.service') +    if rc != 0: +        print(output) +        return None +    print(f'Dynamic DNS state reset!') + +@_verify('forwarding')  def show_forwarding_statistics(raw: bool):      dns_data = _get_forwarding_statistics_raw()      if raw: @@ -88,7 +176,7 @@ def show_forwarding_statistics(raw: bool):      else:          return _get_forwarding_statistics_formatted(dns_data) -@_verify_forwarding +@_verify('forwarding')  def reset_forwarding(all: bool, domain: typing.Optional[str]):      """      Reset DNS Forwarding cache diff --git a/src/op_mode/dns_dynamic.py b/src/op_mode/dns_dynamic.py deleted file mode 100755 index 12aa5494a..000000000 --- a/src/op_mode/dns_dynamic.py +++ /dev/null @@ -1,113 +0,0 @@ -#!/usr/bin/env python3 -# -# Copyright (C) 2018-2023 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 -import argparse -import sys -import time -from tabulate import tabulate - -from vyos.config import Config -from vyos.template import is_ipv4, is_ipv6 -from vyos.utils.process import call - -cache_file = r'/run/ddclient/ddclient.cache' - -columns = { -    'host':        'Hostname', -    'ipv4':        'IPv4 address', -    'status-ipv4': 'IPv4 status', -    'ipv6':        'IPv6 address', -    'status-ipv6': 'IPv6 status', -    'mtime':       'Last update', -} - - -def _get_formatted_host_records(host_data): -    data_entries = [] -    for entry in host_data: -        data_entries.append([entry.get(key) for key in columns.keys()]) - -    header = columns.values() -    output = tabulate(data_entries, header, numalign='left') -    return output - - -def show_status(): -    # A ddclient status file might not always exist -    if not os.path.exists(cache_file): -        sys.exit(0) - -    data = [] - -    with open(cache_file, 'r') as f: -        for line in f: -            if line.startswith('#'): -                continue - -            props = {} -            # ddclient cache rows have properties in 'key=value' format separated by comma -            # we pick up the ones we are interested in -            for kvraw in line.split(' ')[0].split(','): -                k, v = kvraw.split('=') -                if k in list(columns.keys()) + ['ip', 'status']:  # ip and status are legacy keys -                    props[k] = v - -            # Extract IPv4 and IPv6 address and status from legacy keys -            # Dual-stack isn't supported in legacy format, 'ip' and 'status' are for one of IPv4 or IPv6 -            if 'ip' in props: -                if is_ipv4(props['ip']): -                    props['ipv4'] = props['ip'] -                    props['status-ipv4'] = props['status'] -                elif is_ipv6(props['ip']): -                    props['ipv6'] = props['ip'] -                    props['status-ipv6'] = props['status'] -                del props['ip'] - -            # Convert mtime to human readable format -            if 'mtime' in props: -                props['mtime'] = time.strftime( -                    "%Y-%m-%d %H:%M:%S", time.localtime(int(props['mtime'], base=10))) - -            data.append(props) - -    print(_get_formatted_host_records(data)) - - -def update_ddns(): -    call('systemctl stop ddclient.service') -    if os.path.exists(cache_file): -        os.remove(cache_file) -    call('systemctl start ddclient.service') - - -if __name__ == '__main__': -    parser = argparse.ArgumentParser() -    group = parser.add_mutually_exclusive_group() -    group.add_argument("--status", help="Show DDNS status", action="store_true") -    group.add_argument("--update", help="Update DDNS on a given interface", action="store_true") -    args = parser.parse_args() - -    # Do nothing if service is not configured -    c = Config() -    if not c.exists_effective('service dns dynamic'): -        print("Dynamic DNS not configured") -        sys.exit(1) - -    if args.status: -        show_status() -    elif args.update: -        update_ddns() | 
