252 lines
7.7 KiB
Python
252 lines
7.7 KiB
Python
|
import os
|
||
|
import signal
|
||
|
from string import Template
|
||
|
import subprocess
|
||
|
import time
|
||
|
from multiprocessing import Pool
|
||
|
from functools import cached_property
|
||
|
from TdcPlugin import TdcPlugin
|
||
|
|
||
|
from tdc_config import *
|
||
|
|
||
|
try:
|
||
|
from pyroute2 import netns
|
||
|
from pyroute2 import IPRoute
|
||
|
netlink = True
|
||
|
except ImportError:
|
||
|
netlink = False
|
||
|
print("!!! Consider installing pyroute2 !!!")
|
||
|
|
||
|
class SubPlugin(TdcPlugin):
|
||
|
def __init__(self):
|
||
|
self.sub_class = 'ns/SubPlugin'
|
||
|
super().__init__()
|
||
|
|
||
|
def pre_suite(self, testcount, testlist):
|
||
|
super().pre_suite(testcount, testlist)
|
||
|
|
||
|
def prepare_test(self, test):
|
||
|
if 'skip' in test and test['skip'] == 'yes':
|
||
|
return
|
||
|
|
||
|
if 'nsPlugin' not in test['plugins']:
|
||
|
return
|
||
|
|
||
|
if netlink == True:
|
||
|
self._nl_ns_create()
|
||
|
else:
|
||
|
self._ipr2_ns_create()
|
||
|
|
||
|
# Make sure the netns is visible in the fs
|
||
|
ticks = 20
|
||
|
while True:
|
||
|
if ticks == 0:
|
||
|
raise TimeoutError
|
||
|
self._proc_check()
|
||
|
try:
|
||
|
ns = self.args.NAMES['NS']
|
||
|
f = open('/run/netns/{}'.format(ns))
|
||
|
f.close()
|
||
|
break
|
||
|
except:
|
||
|
time.sleep(0.1)
|
||
|
ticks -= 1
|
||
|
continue
|
||
|
|
||
|
def pre_case(self, test, test_skip):
|
||
|
if self.args.verbose:
|
||
|
print('{}.pre_case'.format(self.sub_class))
|
||
|
|
||
|
if test_skip:
|
||
|
return
|
||
|
|
||
|
self.prepare_test(test)
|
||
|
|
||
|
def post_case(self):
|
||
|
if self.args.verbose:
|
||
|
print('{}.post_case'.format(self.sub_class))
|
||
|
|
||
|
if netlink == True:
|
||
|
self._nl_ns_destroy()
|
||
|
else:
|
||
|
self._ipr2_ns_destroy()
|
||
|
|
||
|
def post_suite(self, index):
|
||
|
if self.args.verbose:
|
||
|
print('{}.post_suite'.format(self.sub_class))
|
||
|
|
||
|
# Make sure we don't leak resources
|
||
|
cmd = self._replace_keywords("$IP -a netns del")
|
||
|
|
||
|
if self.args.verbose > 3:
|
||
|
print('_exec_cmd: command "{}"'.format(cmd))
|
||
|
|
||
|
subprocess.run(cmd, shell=True, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
|
||
|
|
||
|
def adjust_command(self, stage, command):
|
||
|
super().adjust_command(stage, command)
|
||
|
cmdform = 'list'
|
||
|
cmdlist = list()
|
||
|
|
||
|
if self.args.verbose:
|
||
|
print('{}.adjust_command'.format(self.sub_class))
|
||
|
|
||
|
if not isinstance(command, list):
|
||
|
cmdform = 'str'
|
||
|
cmdlist = command.split()
|
||
|
else:
|
||
|
cmdlist = command
|
||
|
if stage == 'setup' or stage == 'execute' or stage == 'verify' or stage == 'teardown':
|
||
|
if self.args.verbose:
|
||
|
print('adjust_command: stage is {}; inserting netns stuff in command [{}] list [{}]'.format(stage, command, cmdlist))
|
||
|
cmdlist.insert(0, self.args.NAMES['NS'])
|
||
|
cmdlist.insert(0, 'exec')
|
||
|
cmdlist.insert(0, 'netns')
|
||
|
cmdlist.insert(0, self.args.NAMES['IP'])
|
||
|
else:
|
||
|
pass
|
||
|
|
||
|
if cmdform == 'str':
|
||
|
command = ' '.join(cmdlist)
|
||
|
else:
|
||
|
command = cmdlist
|
||
|
|
||
|
if self.args.verbose:
|
||
|
print('adjust_command: return command [{}]'.format(command))
|
||
|
return command
|
||
|
|
||
|
def _nl_ns_create(self):
|
||
|
ns = self.args.NAMES["NS"];
|
||
|
dev0 = self.args.NAMES["DEV0"];
|
||
|
dev1 = self.args.NAMES["DEV1"];
|
||
|
dummy = self.args.NAMES["DUMMY"];
|
||
|
|
||
|
if self.args.verbose:
|
||
|
print('{}._nl_ns_create'.format(self.sub_class))
|
||
|
|
||
|
netns.create(ns)
|
||
|
netns.pushns(newns=ns)
|
||
|
with IPRoute() as ip:
|
||
|
ip.link('add', ifname=dev1, kind='veth', peer={'ifname': dev0, 'net_ns_fd':'/proc/1/ns/net'})
|
||
|
ip.link('add', ifname=dummy, kind='dummy')
|
||
|
ticks = 20
|
||
|
while True:
|
||
|
if ticks == 0:
|
||
|
raise TimeoutError
|
||
|
try:
|
||
|
dev1_idx = ip.link_lookup(ifname=dev1)[0]
|
||
|
dummy_idx = ip.link_lookup(ifname=dummy)[0]
|
||
|
ip.link('set', index=dev1_idx, state='up')
|
||
|
ip.link('set', index=dummy_idx, state='up')
|
||
|
break
|
||
|
except:
|
||
|
time.sleep(0.1)
|
||
|
ticks -= 1
|
||
|
continue
|
||
|
netns.popns()
|
||
|
|
||
|
with IPRoute() as ip:
|
||
|
ticks = 20
|
||
|
while True:
|
||
|
if ticks == 0:
|
||
|
raise TimeoutError
|
||
|
try:
|
||
|
dev0_idx = ip.link_lookup(ifname=dev0)[0]
|
||
|
ip.link('set', index=dev0_idx, state='up')
|
||
|
break
|
||
|
except:
|
||
|
time.sleep(0.1)
|
||
|
ticks -= 1
|
||
|
continue
|
||
|
|
||
|
def _ipr2_ns_create_cmds(self):
|
||
|
cmds = []
|
||
|
|
||
|
ns = self.args.NAMES['NS']
|
||
|
|
||
|
cmds.append(self._replace_keywords('netns add {}'.format(ns)))
|
||
|
cmds.append(self._replace_keywords('link add $DEV1 type veth peer name $DEV0'))
|
||
|
cmds.append(self._replace_keywords('link set $DEV1 netns {}'.format(ns)))
|
||
|
cmds.append(self._replace_keywords('link add $DUMMY type dummy'.format(ns)))
|
||
|
cmds.append(self._replace_keywords('link set $DUMMY netns {}'.format(ns)))
|
||
|
cmds.append(self._replace_keywords('netns exec {} $IP link set $DEV1 up'.format(ns)))
|
||
|
cmds.append(self._replace_keywords('netns exec {} $IP link set $DUMMY up'.format(ns)))
|
||
|
cmds.append(self._replace_keywords('link set $DEV0 up'.format(ns)))
|
||
|
|
||
|
if self.args.device:
|
||
|
cmds.append(self._replace_keywords('link set $DEV2 netns {}'.format(ns)))
|
||
|
cmds.append(self._replace_keywords('netns exec {} $IP link set $DEV2 up'.format(ns)))
|
||
|
|
||
|
return cmds
|
||
|
|
||
|
def _ipr2_ns_create(self):
|
||
|
'''
|
||
|
Create the network namespace in which the tests will be run and set up
|
||
|
the required network devices for it.
|
||
|
'''
|
||
|
self._exec_cmd_batched('pre', self._ipr2_ns_create_cmds())
|
||
|
|
||
|
def _nl_ns_destroy(self):
|
||
|
ns = self.args.NAMES['NS']
|
||
|
netns.remove(ns)
|
||
|
|
||
|
def _ipr2_ns_destroy_cmd(self):
|
||
|
return self._replace_keywords('netns delete {}'.format(self.args.NAMES['NS']))
|
||
|
|
||
|
def _ipr2_ns_destroy(self):
|
||
|
'''
|
||
|
Destroy the network namespace for testing (and any associated network
|
||
|
devices as well)
|
||
|
'''
|
||
|
self._exec_cmd('post', self._ipr2_ns_destroy_cmd())
|
||
|
|
||
|
@cached_property
|
||
|
def _proc(self):
|
||
|
ip = self._replace_keywords("$IP -b -")
|
||
|
proc = subprocess.Popen(ip,
|
||
|
shell=True,
|
||
|
stdin=subprocess.PIPE,
|
||
|
env=ENVIR)
|
||
|
|
||
|
return proc
|
||
|
|
||
|
def _proc_check(self):
|
||
|
proc = self._proc
|
||
|
|
||
|
proc.poll()
|
||
|
|
||
|
if proc.returncode is not None and proc.returncode != 0:
|
||
|
raise RuntimeError("iproute2 exited with an error code")
|
||
|
|
||
|
def _exec_cmd(self, stage, command):
|
||
|
'''
|
||
|
Perform any required modifications on an executable command, then run
|
||
|
it in a subprocess and return the results.
|
||
|
'''
|
||
|
|
||
|
if self.args.verbose > 3:
|
||
|
print('_exec_cmd: command "{}"'.format(command))
|
||
|
|
||
|
proc = self._proc
|
||
|
|
||
|
proc.stdin.write((command + '\n').encode())
|
||
|
proc.stdin.flush()
|
||
|
|
||
|
if self.args.verbose > 3:
|
||
|
print('_exec_cmd proc: {}'.format(proc))
|
||
|
|
||
|
self._proc_check()
|
||
|
|
||
|
def _exec_cmd_batched(self, stage, commands):
|
||
|
for cmd in commands:
|
||
|
self._exec_cmd(stage, cmd)
|
||
|
|
||
|
def _replace_keywords(self, cmd):
|
||
|
"""
|
||
|
For a given executable command, substitute any known
|
||
|
variables contained within NAMES with the correct values
|
||
|
"""
|
||
|
tcmd = Template(cmd)
|
||
|
subcmd = tcmd.safe_substitute(self.args.NAMES)
|
||
|
return subcmd
|