summaryrefslogtreecommitdiff
path: root/python/vyos/ifconfig/dhcp.py
blob: 5f99a0b7ebfab3288f949360a55bcaa877784927 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
# Copyright 2020 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
# License as published by the Free Software Foundation; either
# version 2.1 of the License, or (at your option) any later version.
#
# This library 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
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
# License along with this library.  If not, see <http://www.gnu.org/licenses/>.

import os
import jmespath

from vyos.configdict import dict_merge
from vyos.configverify import verify_dhcpv6
from vyos.ifconfig.control import Control
from vyos.template import render

class _DHCPv4 (Control):
    def __init__(self, ifname):
        super().__init__()
        config_base = r'/var/lib/dhcp/dhclient'
        self._conf_file = f'{config_base}_{ifname}.conf'
        self._options_file = f'{config_base}_{ifname}.options'
        self._pid_file = f'{config_base}_{ifname}.pid'
        self._lease_file = f'{config_base}_{ifname}.leases'
        self.options = {'ifname' : ifname}

    # replace dhcpv4/v6 with systemd.networkd?
    def set(self):
        """
        Configure interface as DHCP client. The dhclient binary is automatically
        started in background!

        Example:
        >>> from vyos.ifconfig import Interface
        >>> j = Interface('eth0')
        >>> j.dhcp.v4.set()
        """

        if jmespath.search('dhcp_options.host_name', self.options) == None:
            # read configured system hostname.
            # maybe change to vyos hostd client ???
            hostname = 'vyos'
            with open('/etc/hostname', 'r') as f:
                hostname = f.read().rstrip('\n')
                tmp = {'dhcp_options' : { 'host_name' : hostname}}
                self.options = dict_merge(tmp, self.options)

        render(self._options_file, 'dhcp-client/daemon-options.tmpl',
               self.options, trim_blocks=True)
        render(self._conf_file, 'dhcp-client/ipv4.tmpl',
               self.options, trim_blocks=True)

        return self._cmd('systemctl restart dhclient@{ifname}.service'.format(**self.options))

    def delete(self):
        """
        De-configure interface as DHCP clinet. All auto generated files like
        pid, config and lease will be removed.

        Example:
        >>> from vyos.ifconfig import Interface
        >>> j = Interface('eth0')
        >>> j.dhcp.v4.delete()
        """
        if not os.path.isfile(self._pid_file):
            self._debug_msg('No DHCP client PID found')
            return None

        self._cmd('systemctl stop dhclient@{ifname}.service'.format(**self.options))

        # cleanup old config files
        for file in [self._conf_file, self._options_file, self._pid_file, self._lease_file]:
            if os.path.isfile(file):
                os.remove(file)

class _DHCPv6 (Control):
    def __init__(self, ifname):
        super().__init__()
        self.options = {'ifname' : ifname}
        self._config = f'/run/dhcp6c/dhcp6c.{ifname}.conf'

    def set(self):
        """
        Configure interface as DHCPv6 client. The client is automatically
        started in background when address is configured as DHCP.

        Example:
        >>> from vyos.ifconfig import Interface
        >>> j = Interface('eth0')
        >>> j.dhcp.v6.set()
        """

        # better save then sorry .. should be checked in interface script but if you
        # missed it we are safe!
        verify_dhcpv6(self.options)

        render(self._config, 'dhcp-client/ipv6.tmpl',
               self.options, trim_blocks=True)
        return self._cmd('systemctl restart dhcp6c@{ifname}.service'.format(
            **self.options))

    def delete(self):
        """
        De-configure interface as DHCPv6 client. All auto generated files like
        pid, config and lease will be removed.

        Example:
        >>> from vyos.ifconfig import Interface
        >>> j = Interface('eth0')
        >>> j.dhcp.v6.delete()
        """
        self._cmd('systemctl stop dhcp6c@{ifname}.service'.format(**self.options))

        # cleanup old config files
        if os.path.isfile(self._config):
            os.remove(self._config)

class DHCP(object):
    def __init__(self, ifname):
        self.v4 = _DHCPv4(ifname)
        self.v6 = _DHCPv6(ifname)