public inbox for passt-dev@passt.top
 help / color / mirror / code / Atom feed
blob be226e405784908d4adf9475ce8a883c99181cf8 4816 bytes (raw)

  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
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
 
#! /usr/bin/python3

# SPDX-License-Identifier: GPL-2.0-or-later
#
# tasst - Test A Simple Socket Transport
#         library of test helpers for passt & pasta
#
# tasst/dhcp.py - Helpers for testing DHCP
#
# Copyright Red Hat
# Author: David Gibson <david@gibson.dropbear.id.au>

import contextlib
import ipaddress
import os

from tasst import Tasst, TasstSubData
from tasst.address import IpiAllocator, TEST_NET_1
from tasst.nstool import UnshareSite
from tasst.site import Site
from tasst.typing import typecheck


class DhcpTasstInfo:
    def __init__(self, site, ifname, addr, gw, mtu):
        self.site = typecheck(site, Site)
        self.ifname = typecheck(ifname, str)
        self.addr = typecheck(addr, ipaddress.IPv4Address)
        self.gw = typecheck(gw, ipaddress.IPv4Address)
        self.mtu = typecheck(mtu, int)

        site.require_cmds('ip')


class BaseDhcpTasst(Tasst):
    """
    Test DHCP behaviour.

    :avocado: disable
    """

    DHCLIENT = '/sbin/dhclient'

    def setup_dhcp(self):
        raise NotImplementedError("{} must implement setup_dhcp() method".format(type(self).__name__))

    @contextlib.contextmanager
    def dhclient(self):
        with self.setup_dhcp() as dti:
            dti = typecheck(dti, DhcpTasstInfo)

            dti.site.require_cmds(self.DHCLIENT, 'cat', 'kill')

            pidfile = os.path.join(self.workdir, 'dhclient.pid')
            leasefile = os.path.join(self.workdir, 'dhclient.leases')

            # We need '-nc' because we may be running with
            # capabilities but not UID 0.  Without -nc dhclient drops
            # capabilities before invoking dhclient-script, so it's
            # unable to actually configure the interface
            dti.site.fg('{} -4 -v -nc -pf {} -lf {} {}'
                        .format(self.DHCLIENT, pidfile, leasefile, dti.ifname), sudo=True)

            try:
                yield dti
            finally:
                pid = int(dti.site.output('cat {}'.format(pidfile)))
                dti.site.fg('kill {}'.format(pid))

    def test_addr(self):
        with self.dhclient() as dti:
            (addr,) = dti.site.addrs(dti.ifname, family='inet', scope='global')
            self.assertEquals(addr.ip, dti.addr)

    def test_route(self):
        with self.dhclient() as dti:
            (defroute,) = dti.site.routes4(dst='default')
            self.assertEquals(ipaddress.ip_address(defroute['gateway']), dti.gw)

    def test_mtu(self):
        with self.dhclient() as dti:
            self.assertEquals(dti.site.mtu(dti.ifname), dti.mtu)


class MetaDhcpTasst(BaseDhcpTasst):
    """Ugly workaround for
    https://github.com/avocado-framework/avocado/issues/5680.
    Explicitly apply the "meta" tag to inherited tests

    :avocado: disable
    :avocado: tags=meta

    """

    def test_addr(self):
        super().test_addr()

    def test_route(self):
        super().test_route()

    def test_mtu(self):
        super().test_mtu()


class DhcpdTasst(MetaDhcpTasst):
    DHCPD = 'dhcpd'
    SUBNET = TEST_NET_1

    @contextlib.contextmanager
    def setup_dhcp(self):
        ifname = 'clientif'
        server_ifname = 'serverif'

        with UnshareSite(type(self).__name__ + '.client', '-Un') as client, \
             UnshareSite(type(self).__name__ + '.server', '-n',
                         parent=client, sudo=True) as server:

            server.require_cmds(self.DHCPD)

            client.veth(ifname, server_ifname, server)

            # Configure the DHCP server
            ipa = IpiAllocator(self.SUBNET)
            (server_ip4,) = ipa.next_ipis()
            (client_ip4,) = ipa.next_ipis()

            confpath = os.path.join(self.workdir, 'dhcpd.conf')
            open(confpath, 'w').write('''
            subnet {} netmask {} {{
                option routers {};
                range {} {};
            }}
            '''.format(self.SUBNET.network_address, self.SUBNET.netmask,
                       server_ip4.ip, client_ip4.ip, client_ip4.ip))
            pidpath = os.path.join(self.workdir, 'dhcpd.pid')
            leasepath = os.path.join(self.workdir, 'dhcpd.leases')
            open(leasepath, 'w').write('')

            server.ifup('lo')
            server.ifup(server_ifname, server_ip4)

            opts = ('-f -d -4 -cf {} -lf {} -pf {}'.format(confpath, leasepath, pidpath))
            server.fg('{} -t {}'.format(self.DHCPD, opts)) # test config
            with server.bg('{} {}'.format(self.DHCPD, opts), sudo=True) as dhcpd:
                # Configure the client
                client.ifup('lo')

                yield DhcpTasstInfo(client, ifname, client_ip4.ip, server_ip4.ip, 1500)

                pid = int(open(pidpath).read())
                server.fg('kill {}'.format(pid))
                status = dhcpd.wait()

debug log:

solving be226e4 ...
found be226e4 in https://archives.passt.top/passt-dev/20230531015849.3229596-19-david@gibson.dropbear.id.au/

applying [1/1] https://archives.passt.top/passt-dev/20230531015849.3229596-19-david@gibson.dropbear.id.au/
diff --git a/avocado/tasst/dhcp.py b/avocado/tasst/dhcp.py
new file mode 100644
index 0000000..be226e4

Checking patch avocado/tasst/dhcp.py...
Applied patch avocado/tasst/dhcp.py cleanly.

index at:
100644 be226e405784908d4adf9475ce8a883c99181cf8	avocado/tasst/dhcp.py

Code repositories for project(s) associated with this public inbox

	https://passt.top/passt

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for IMAP folder(s).