Adds tests according to the CI
Adds tests according to the CI justification Adds tests according to the CI wiki specifically the standard test interface in the spec. The playbook includes Tier1 level test cases that have been tested in the following contexts and is passing reliably: Classic. Test logs are stored in the artifacts directory. The following steps are used to execute the tests using the standard test interface: Test enveronment Make sure you have installed packages from the spec ``` ansible-2.4.1.0-2.fc28.noarch python2-dnf-2.7.5-1.fc28.noarch libselinux-python-2.7-2.fc28.x86_64 standard-test-roles-2.5-1.fc28.noarch Run tests for Classic Snip of the example test run for Classic tests: ``` net-snmp-5.7.3-38.fc29.x86_64 :: [ 17:12:37 ] :: [ PASS ] :: Checking for the presence of net-snmp rpm :: [ 17:12:37 ] :: [ PASS ] :: Checking for the presence of net-snmp rpm :: [ 17:12:37 ] :: [ LOG ] :: Package versions: :: [ 17:12:37 ] :: [ LOG ] :: Package versions: :: [ 17:12:37 ] :: [ LOG ] :: net-snmp-5.7.3-38.fc29.x86_64 :: [ 17:12:37 ] :: [ LOG ] :: net-snmp-5.7.3-38.fc29.x86_64 net-snmp-utils-5.7.3-38.fc29.x86_64 :: [ 17:12:37 ] :: [ PASS ] :: Checking for the presence of net-snmp-utils rpm :: [ 17:12:37 ] :: [ PASS ] :: Checking for the presence of net-snmp-utils rpm :: [ 17:12:37 ] :: [ LOG ] :: Package versions: :: [ 17:12:37 ] :: [ LOG ] :: Package versions: :: [ 17:12:37 ] :: [ LOG ] :: net-snmp-utils-5.7.3-38.fc29.x86_64 :: [ 17:12:37 ] :: [ LOG ] :: net-snmp-utils-5.7.3-38.fc29.x86_64 :: [ 17:12:37 ] :: [ BEGIN ] :: Running 'systemctl stop firewalld' :: [ 17:12:37 ] :: [ PASS ] :: Command 'systemctl stop firewalld' (Expected 0,5, got 0) :: [ 17:12:37 ] :: [ PASS ] :: Command 'systemctl stop firewalld' (Expected 0,5, got 0) :: [ 17:12:37 ] :: [ BEGIN ] :: Running 'setenforce 0' :: [ 17:12:37 ] :: [ PASS ] :: Command 'setenforce 0' (Expected 0,1, got 0) :: [ 17:12:37 ] :: [ PASS ] :: Command 'setenforce 0' (Expected 0,1, got 0) :: [ 17:12:37 ] :: [ BEGIN ] :: Running '[ -e /sys/class/net/veth-test ] && ip link del veth-test' :: [ 17:12:37 ] :: [ PASS ] :: Command '[ -e /sys/class/net/veth-test ] && ip link del veth-test' (Expected 0,1, got 1) :: [ 17:12:37 ] :: [ PASS ] :: Command '[ -e /sys/class/net/veth-test ] && ip link del veth-test' (Expected 0,1, got 1) :: [ 17:12:37 ] :: [ BEGIN ] :: Running 'cp net-snmp-tests.py /usr/bin/' :: [ 17:12:37 ] :: [ PASS ] :: Command 'cp net-snmp-tests.py /usr/bin/' (Expected 0, got 0) :: [ 17:12:37 ] :: [ PASS ] :: Command 'cp net-snmp-tests.py /usr/bin/' (Expected 0, got 0) :: [ 17:12:37 ] :: [ INFO ] :: using '/var/tmp/beakerlib-FznqAiL/backup' as backup destination :: [ 17:12:37 ] :: [ INFO ] :: using '/var/tmp/beakerlib-FznqAiL/backup' as backup destination :: [ 17:12:38 ] :: [ BEGIN ] :: Running 'cp snmpd.conf /etc/snmp/snmpd.conf' : [ 17:12:38 ] :: [ BEGIN ] :: Running 'cp snmpd.conf /etc/snmp/snmpd.conf' :: [ 17:12:38 ] :: [ PASS ] :: Command 'cp snmpd.conf /etc/snmp/snmpd.conf' (Expected 0, got 0) :: [ 17:12:38 ] :: [ PASS ] :: Command 'cp snmpd.conf /etc/snmp/snmpd.conf' (Expected 0, got 0) :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: :: Duration: 1s :: Duration: 1s :: Assertions: 7 good, 0 bad :: Assertions: 7 good, 0 bad :: RESULT: PASS :: RESULT: PASS :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: :: Test :: Test :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: :: [ 17:12:38 ] :: [ LOG ] :: Starting net-snmp tests ... :: [ 17:12:38 ] :: [ LOG ] :: Starting net-snmp tests ... :: [ 17:12:38 ] :: [ BEGIN ] :: Running '/usr/bin/python3 /usr/bin/net-snmp-tests.py' test_SNMP_IF_MIB_network_interface (__main__.SnmpdTests) verify network interface (1.3.6.1.2.1.2.2.1) SNMP variables ... ok test_SNMP_MIB_2_System (__main__.SnmpdTests) verify RFC 1213 System (1.3.6.1.2.1.1) SNMP variables ... ok test_SNMP_hrSWRunPath (__main__.SnmpdTests) process id ... ok test_UCD_SNMP_MIB_memory (__main__.SnmpdTests) UCD-SNMP-MIB::memory ... ok test_basic_snmpwalk (__main__.SnmpdTests) verify snmpwalk getting success snmpwalk -v2c -c public localhost ... ok ---------------------------------------------------------------------- Ran 5 tests in 7.450s OK :: [ 17:12:45 ] :: [ PASS ] :: Command '/usr/bin/python3 /usr/bin/net-snmp-tests.py' (Expected 0, got 0) :: [ 17:12:45 ] :: [ PASS ] :: Command '/usr/bin/python3 /usr/bin/net-snmp-tests.py' (Expected 0, got 0) ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
This commit is contained in:
parent
4b1b37eb16
commit
d3febaaaa0
36
tests/integration-tests/Makefile
Normal file
36
tests/integration-tests/Makefile
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
# SPDX-License-Identifier: LGPL-2.1+
|
||||||
|
# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
#
|
||||||
|
# Makefile of /CoreOS/net-snmp
|
||||||
|
# Description: Test if net-snmp working ok
|
||||||
|
# Author: Susant Sahani<susant@redhat.com>
|
||||||
|
#
|
||||||
|
# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
export TEST=/CoreOS/net-snmp
|
||||||
|
export TESTVERSION=1.0
|
||||||
|
BUILT_FILES=
|
||||||
|
FILES=$(METADATA) runtest.sh Makefile PURPOSE
|
||||||
|
.PHONY: all install download clean
|
||||||
|
run: $(FILES) build
|
||||||
|
./runtest.sh
|
||||||
|
build: $(BUILT_FILES)
|
||||||
|
test -x runtest.sh || chmod a+x runtest.sh
|
||||||
|
clean:
|
||||||
|
rm -f *~ $(BUILT_FILES)
|
||||||
|
include /usr/share/rhts/lib/rhts-make.include
|
||||||
|
$(METADATA): Makefile
|
||||||
|
@echo "Owner: Susant Sahani<susant@redhat.com>" > $(METADATA)
|
||||||
|
@echo "Name: $(TEST)" >> $(METADATA)
|
||||||
|
@echo "TestVersion: $(TESTVERSION)" >> $(METADATA)
|
||||||
|
@echo "Path: $(TEST_DIR)" >> $(METADATA)
|
||||||
|
@echo "Description: Test snmpd" >> $(METADATA)
|
||||||
|
@echo "Type: Sanity" >> $(METADATA)
|
||||||
|
@echo "TestTime: 5m" >> $(METADATA)
|
||||||
|
@echo "RunFor: net-snmp" >> $(METADATA)
|
||||||
|
@echo "Requires: net=snmp" >> $(METADATA)
|
||||||
|
@echo "Priority: Normal" >> $(METADATA)
|
||||||
|
@echo "License: GPLv2" >> $(METADATA)
|
||||||
|
@echo "Confidential: no" >> $(METADATA)
|
||||||
|
@echo "Destructive: no" >> $(METADATA)
|
||||||
|
@echo "Releases: -Fedora 28" >> $(METADATA)
|
||||||
|
rhts-lint $(METADATA)
|
3
tests/integration-tests/PURPOSE
Normal file
3
tests/integration-tests/PURPOSE
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
PURPOSE of /CoreOS/net-snmp
|
||||||
|
Description: tests for net-snmp
|
||||||
|
Author: Susant Sahani<susant@redhat.com>
|
175
tests/integration-tests/net-snmp-tests.py
Executable file
175
tests/integration-tests/net-snmp-tests.py
Executable file
@ -0,0 +1,175 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
# SPDX-License-Identifier: LGPL-2.1+
|
||||||
|
# ~~~
|
||||||
|
# Description: Tests for snmpd
|
||||||
|
#
|
||||||
|
# Author: Susant Sahani <susant@redhat.com>
|
||||||
|
# Copyright (c) 2018 Red Hat, Inc.
|
||||||
|
# ~~~
|
||||||
|
|
||||||
|
import errno
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import time
|
||||||
|
import unittest
|
||||||
|
import subprocess
|
||||||
|
import signal
|
||||||
|
import shutil
|
||||||
|
import psutil
|
||||||
|
import socket
|
||||||
|
import platform
|
||||||
|
import re
|
||||||
|
from pyroute2 import IPRoute
|
||||||
|
from psutil import virtual_memory
|
||||||
|
from collections import OrderedDict
|
||||||
|
|
||||||
|
HOST='192.168.111.50'
|
||||||
|
|
||||||
|
def setUpModule():
|
||||||
|
"""Initialize the environment, and perform sanity checks on it."""
|
||||||
|
|
||||||
|
if shutil.which('snmpd') is None:
|
||||||
|
raise OSError(errno.ENOENT, 'snmpd not found')
|
||||||
|
|
||||||
|
if shutil.which('snmpwalk') is None:
|
||||||
|
raise OSError(errno.ENOENT, 'snmpwalk not found')
|
||||||
|
|
||||||
|
def tearDownModule():
|
||||||
|
pass
|
||||||
|
|
||||||
|
class GenericUtilities():
|
||||||
|
"""Provide a set of utility functions start stop daemons. write config files etc """
|
||||||
|
|
||||||
|
def StartSnmpd(self):
|
||||||
|
"""Start snmpd"""
|
||||||
|
subprocess.check_output(['systemctl', 'start', 'snmpd'])
|
||||||
|
|
||||||
|
def StopSnmpd(self):
|
||||||
|
"""Stop snmpd"""
|
||||||
|
subprocess.check_output(['systemctl', 'stop', 'snmpd'])
|
||||||
|
|
||||||
|
def SetupVethInterface(self):
|
||||||
|
"""Setup veth interface"""
|
||||||
|
|
||||||
|
ip = IPRoute()
|
||||||
|
|
||||||
|
ip.link('add', ifname='veth-test', peer='veth-peer', kind='veth')
|
||||||
|
idx_veth_test = ip.link_lookup(ifname='veth-test')[0]
|
||||||
|
idx_veth_peer = ip.link_lookup(ifname='veth-peer')[0]
|
||||||
|
|
||||||
|
ip.link('set', index=idx_veth_test, address='12:11:12:13:14:18')
|
||||||
|
ip.link('set', index=idx_veth_peer, address='22:21:22:23:24:29')
|
||||||
|
ip.link('set', index=idx_veth_test, state='up')
|
||||||
|
ip.link('set', index=idx_veth_peer, state='up')
|
||||||
|
ip.addr('add', index=idx_veth_test, address='192.168.111.50')
|
||||||
|
ip.addr('add', index=idx_veth_peer, address='192.168.111.51')
|
||||||
|
|
||||||
|
ip.close()
|
||||||
|
|
||||||
|
def TearDownVethInterface(self):
|
||||||
|
ip = IPRoute()
|
||||||
|
|
||||||
|
ip.link('del', index=ip.link_lookup(ifname='veth-test')[0])
|
||||||
|
ip.close()
|
||||||
|
|
||||||
|
class SnmpdTests(unittest.TestCase, GenericUtilities):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
self.SetupVethInterface()
|
||||||
|
time.sleep(1)
|
||||||
|
self.StartSnmpd()
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
self.StopSnmpd()
|
||||||
|
self.TearDownVethInterface()
|
||||||
|
|
||||||
|
def test_UCD_SNMP_MIB_memory(self):
|
||||||
|
''' UCD-SNMP-MIB::memory '''
|
||||||
|
|
||||||
|
subprocess.check_output(['snmpwalk', '-v2c', '-c' , 'public', HOST, 'UCD-SNMP-MIB::memory'])
|
||||||
|
|
||||||
|
meminfo=OrderedDict()
|
||||||
|
with open('/proc/meminfo') as f:
|
||||||
|
for line in f:
|
||||||
|
meminfo[line.split(':')[0]] = line.split(':')[1].strip()
|
||||||
|
|
||||||
|
output=subprocess.check_output(['snmpwalk', '-v2c', '-c' , 'public', HOST, 'UCD-SNMP-MIB::memTotalReal.0']).rstrip().decode('utf-8')
|
||||||
|
self.assertRegex(output, meminfo['MemTotal'])
|
||||||
|
|
||||||
|
def test_SNMP_hrSWRunPath(self):
|
||||||
|
""" process id """
|
||||||
|
output=subprocess.check_output(['snmpwalk', '-v2c', '-c' , 'public', HOST, 'HOST-RESOURCES-MIB::hrSWRunPath.1']).rstrip().decode('utf-8')
|
||||||
|
self.assertRegex(output, 'systemd')
|
||||||
|
|
||||||
|
def test_SNMP_IF_MIB_network_interface(self):
|
||||||
|
""" verify network interface (1.3.6.1.2.1.2.2.1) SNMP variables """
|
||||||
|
|
||||||
|
ip = IPRoute()
|
||||||
|
|
||||||
|
subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.2.2.1'])
|
||||||
|
|
||||||
|
# 1.3.6.1.2.1.2.2.1.1 IF-MIB::ifIndex
|
||||||
|
output=subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.2.2.1.1']).rstrip().decode('utf-8')
|
||||||
|
self.assertRegex(output, 'IF-MIB::ifIndex.1 = INTEGER: 1')
|
||||||
|
|
||||||
|
# 1.3.6.1.2.1.2.2.1.1 IF-MIB::ifDescr
|
||||||
|
output=subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.2.2.1.2']).rstrip().decode('utf-8')
|
||||||
|
for link in ip.get_links():
|
||||||
|
self.assertRegex(output, link.get_attr('IFLA_IFNAME'))
|
||||||
|
|
||||||
|
# IP-MIB::ipAdEntAddr 1.3.6.1.2.1.4.20.1.1
|
||||||
|
output=subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.4.20.1.1']).rstrip().decode('utf-8')
|
||||||
|
for addr in ip.get_addr():
|
||||||
|
if addr.get_attr('IFA_ADDRESS'):
|
||||||
|
if addr.get_attr('IFA_ADDRESS') != '::1' and addr.get_attr('Ifamily') == 2:
|
||||||
|
self.assertRegex(output, addr.get_attr('IFA_ADDRESS'))
|
||||||
|
|
||||||
|
# IF-MIB::ifPhysAddress. 1.3.6.1.2.1.2.2.1.6
|
||||||
|
output=subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.2.2.1.6']).rstrip().decode('utf-8')
|
||||||
|
for link in ip.get_links():
|
||||||
|
if link.get_attr('IFLA_ADDRESS') and link.get_attr('IFLA_ADDRESS') != '00:00:00:00:00:00':
|
||||||
|
snmp_mac = re.sub(r'\b0+(\d)', r'\1', link.get_attr('IFLA_ADDRESS')).lstrip('0')
|
||||||
|
self.assertRegex(output, snmp_mac)
|
||||||
|
|
||||||
|
ip.close()
|
||||||
|
|
||||||
|
def test_SNMP_MIB_2_System(self):
|
||||||
|
""" verify RFC 1213 System (1.3.6.1.2.1.1) SNMP variables"""
|
||||||
|
|
||||||
|
subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.1']).rstrip().decode('utf-8')
|
||||||
|
|
||||||
|
# 1.3.6.1.2.1.1.1 - sysDescr
|
||||||
|
output=subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.1.1']).rstrip().decode('utf-8')
|
||||||
|
self.assertRegex(output, platform.machine())
|
||||||
|
self.assertRegex(output, platform.node())
|
||||||
|
self.assertRegex(output, platform.processor())
|
||||||
|
self.assertRegex(output, platform.release())
|
||||||
|
self.assertRegex(output, platform.version())
|
||||||
|
|
||||||
|
# 1.3.6.1.2.1.1.2 - sysObjectID
|
||||||
|
subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.1.2'])
|
||||||
|
|
||||||
|
# 1.3.6.1.2.1.1.3 - sysUpTime
|
||||||
|
subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.1.3'])
|
||||||
|
|
||||||
|
# 1.3.6.1.2.1.1.4 - sysContact
|
||||||
|
output=subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.1.4']).rstrip().decode('utf-8')
|
||||||
|
self.assertRegex(output, 'fedora-ci <fedoraci@fedoraproject.org>')
|
||||||
|
|
||||||
|
# 1.3.6.1.2.1.1.5 - sysName
|
||||||
|
output=subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.1.5']).rstrip().decode('utf-8')
|
||||||
|
self.assertRegex(output, socket.gethostname())
|
||||||
|
|
||||||
|
# 1.3.6.1.2.1.1.6 - sysLocation
|
||||||
|
output=subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST, '1.3.6.1.2.1.1.6']).rstrip().decode('utf-8')
|
||||||
|
self.assertRegex(output, 'Pune, IN')
|
||||||
|
|
||||||
|
def test_basic_snmpwalk(self):
|
||||||
|
""" verify snmpwalk getting success snmpwalk -v2c -c public localhost """
|
||||||
|
|
||||||
|
subprocess.check_output(['snmpwalk', '-v2c', '-c', 'public', HOST])
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
unittest.main(testRunner=unittest.TextTestRunner(stream=sys.stdout,
|
||||||
|
verbosity=3))
|
51
tests/integration-tests/runtest.sh
Executable file
51
tests/integration-tests/runtest.sh
Executable file
@ -0,0 +1,51 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# SPDX-License-Identifier: LGPL-2.1+
|
||||||
|
# ~~~
|
||||||
|
# runtest.sh of net-snmp
|
||||||
|
# Description: net-snmp tests
|
||||||
|
#
|
||||||
|
# Author: Susant Sahani <susant@redhat.com>
|
||||||
|
# Copyright (c) 2018 Red Hat, Inc.
|
||||||
|
# ~~~
|
||||||
|
|
||||||
|
# Include Beaker environment
|
||||||
|
. /usr/share/beakerlib/beakerlib.sh || exit 1
|
||||||
|
|
||||||
|
PACKAGE_NET_SNMP="net-snmp"
|
||||||
|
PACKAGE_NET_SNMP_UTILS="net-snmp-utils"
|
||||||
|
|
||||||
|
NET_SNMP_CONF_FILE="/etc/snmp/snmpd.conf"
|
||||||
|
|
||||||
|
rlJournalStart
|
||||||
|
rlPhaseStartSetup
|
||||||
|
rlAssertRpm $PACKAGE_NET_SNMP
|
||||||
|
rlAssertRpm $PACKAGE_NET_SNMP_UTILS
|
||||||
|
|
||||||
|
rlRun "systemctl stop firewalld" 0,5
|
||||||
|
rlRun "setenforce 0" 0,1
|
||||||
|
|
||||||
|
rlRun "[ -e /sys/class/net/veth-test ] && ip link del veth-test" 0,1
|
||||||
|
rlRun "cp net-snmp-tests.py /usr/bin/"
|
||||||
|
|
||||||
|
rlFileBackup "$NET_SNMP_CONF_FILE"
|
||||||
|
rlRun "cp snmpd.conf $NET_SNMP_CONF_FILE"
|
||||||
|
|
||||||
|
rlPhaseEnd
|
||||||
|
|
||||||
|
rlPhaseStartTest
|
||||||
|
rlLog "Starting net-snmp tests ..."
|
||||||
|
rlRun "/usr/bin/python3 /usr/bin/net-snmp-tests.py"
|
||||||
|
rlPhaseEnd
|
||||||
|
|
||||||
|
rlPhaseStartCleanup
|
||||||
|
rlRun "rm /usr/bin/net-snmp-tests.py $NET_SNMP_CONFIG_FILE"
|
||||||
|
rlRun "systemctl daemon-reload"
|
||||||
|
rlRun "[ -e /sys/class/net/veth-test ] && ip link del veth-test" 0,1
|
||||||
|
rlFileRestore
|
||||||
|
rlRun "setenforce 1" 0,1
|
||||||
|
rlLog "net-snmp tests done"
|
||||||
|
rlPhaseEnd
|
||||||
|
rlJournalPrintText
|
||||||
|
rlJournalEnd
|
||||||
|
|
||||||
|
rlGetTestState
|
7
tests/integration-tests/snmpd.conf
Normal file
7
tests/integration-tests/snmpd.conf
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
agentAddress udp:192.168.111.50:161
|
||||||
|
|
||||||
|
syslocation Pune, IN
|
||||||
|
syscontact fedora-ci <fedoraci@fedoraproject.org>
|
||||||
|
|
||||||
|
dontLogTCPWrappersConnects yes
|
||||||
|
rocommunity public
|
14
tests/tests.yml
Normal file
14
tests/tests.yml
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
- hosts: localhost
|
||||||
|
roles:
|
||||||
|
- role: standard-test-beakerlib
|
||||||
|
tags:
|
||||||
|
- classic
|
||||||
|
tests:
|
||||||
|
- integration-tests
|
||||||
|
required_packages:
|
||||||
|
- python3
|
||||||
|
- systemd
|
||||||
|
- iproute
|
||||||
|
- python3-pyroute2
|
||||||
|
- net-snmp
|
||||||
|
- net-snmp-utils
|
Loading…
Reference in New Issue
Block a user