blob: 2b26a697e179dfa0ba840d04dad968ed2f62f992 [file] [edit]
#!/bin/bash
# SPDX-License-Identifier: GPL-2.0
# These tests verify that teamd is able to enable and disable ports via the
# active backup runner.
#
# Topology:
#
# +-------------------------+ NS1
# | test_team1 |
# | + |
# | eth0 | eth1 |
# | +---+---+ |
# | | | |
# +-------------------------+
# | |
# +-------------------------+ NS2
# | | | |
# | +-------+ |
# | eth0 | eth1 |
# | + |
# | test_team2 |
# +-------------------------+
export ALL_TESTS="teamd_test_active_backup"
test_dir="$(dirname "$0")"
# shellcheck disable=SC1091
source "${test_dir}/../../../net/lib.sh"
# shellcheck disable=SC1091
source "${test_dir}/team_lib.sh"
NS1=""
NS2=""
export NODAD="nodad"
PREFIX_LENGTH="64"
NS1_IP="fd00::1"
NS2_IP="fd00::2"
NS1_IP4="192.168.0.1"
NS2_IP4="192.168.0.2"
NS1_TEAMD_CONF=""
NS2_TEAMD_CONF=""
NS1_TEAMD_PID=""
NS2_TEAMD_PID=""
while getopts "4" opt; do
case $opt in
4)
echo "IPv4 mode selected."
export NODAD=
PREFIX_LENGTH="24"
NS1_IP="${NS1_IP4}"
NS2_IP="${NS2_IP4}"
;;
\?)
echo "Invalid option: -${OPTARG}" >&2
exit 1
;;
esac
done
teamd_config_create()
{
local runner=$1
local dev=$2
local conf
conf=$(mktemp)
cat > "${conf}" <<-EOF
{
"device": "${dev}",
"runner": {"name": "${runner}"},
"ports": {
"eth0": {},
"eth1": {}
}
}
EOF
echo "${conf}"
}
# Create the network namespaces, veth pair, and team devices in the specified
# runner.
# Globals:
# RET - Used by test infra, set by `check_err` functions.
# Arguments:
# runner - The Teamd runner to use for the Team devices.
environment_create()
{
local runner=$1
echo "Setting up two-link aggregation for runner ${runner}"
echo "Teamd version is: $(teamd --version)"
trap environment_destroy EXIT
setup_ns ns1 ns2
NS1="${NS_LIST[0]}"
NS2="${NS_LIST[1]}"
for link in $(seq 0 1); do
ip -n "${NS1}" link add "eth${link}" type veth peer name \
"eth${link}" netns "${NS2}"
check_err $? "Failed to create veth pair"
done
NS1_TEAMD_CONF=$(teamd_config_create "${runner}" "test_team1")
NS2_TEAMD_CONF=$(teamd_config_create "${runner}" "test_team2")
echo "Conf files are ${NS1_TEAMD_CONF} and ${NS2_TEAMD_CONF}"
ip netns exec "${NS1}" teamd -d -f "${NS1_TEAMD_CONF}"
check_err $? "Failed to create team device in ${NS1}"
NS1_TEAMD_PID=$(pgrep -f "teamd -d -f ${NS1_TEAMD_CONF}")
ip netns exec "${NS2}" teamd -d -f "${NS2_TEAMD_CONF}"
check_err $? "Failed to create team device in ${NS2}"
NS2_TEAMD_PID=$(pgrep -f "teamd -d -f ${NS2_TEAMD_CONF}")
echo "Created team devices"
echo "Teamd PIDs are ${NS1_TEAMD_PID} and ${NS2_TEAMD_PID}"
ip -n "${NS1}" link set test_team1 up
check_err $? "Failed to set test_team1 up in ${NS1}"
ip -n "${NS2}" link set test_team2 up
check_err $? "Failed to set test_team2 up in ${NS2}"
ip -n "${NS1}" addr add "${NS1_IP}/${PREFIX_LENGTH}" "${NODAD}" dev \
test_team1
check_err $? "Failed to add address to team device in ${NS1}"
ip -n "${NS2}" addr add "${NS2_IP}/${PREFIX_LENGTH}" "${NODAD}" dev \
test_team2
check_err $? "Failed to add address to team device in ${NS2}"
slowwait 2 timeout 0.5 ip netns exec "${NS1}" ping -W 1 -c 1 "${NS2_IP}"
}
# Tear down the environment: kill teamd and delete network namespaces.
environment_destroy()
{
echo "Tearing down two-link aggregation"
rm "${NS1_TEAMD_CONF}"
rm "${NS2_TEAMD_CONF}"
# First, try graceful teamd teardown.
ip netns exec "${NS1}" teamd -k -t test_team1
ip netns exec "${NS2}" teamd -k -t test_team2
# If teamd can't be killed gracefully, then sigkill.
if kill -0 "${NS1_TEAMD_PID}" 2>/dev/null; then
echo "Sending sigkill to teamd for test_team1"
kill -9 "${NS1_TEAMD_PID}"
rm -f /var/run/teamd/test_team1.{pid,sock}
fi
if kill -0 "${NS2_TEAMD_PID}" 2>/dev/null; then
echo "Sending sigkill to teamd for test_team2"
kill -9 "${NS2_TEAMD_PID}"
rm -f /var/run/teamd/test_team2.{pid,sock}
fi
cleanup_all_ns
}
# Change the active port for an active-backup mode team.
# Arguments:
# namespace - The network namespace that the team is in.
# team - The name of the team.
# active_port - The port to make active.
set_active_port()
{
local namespace=$1
local team=$2
local active_port=$3
ip netns exec "${namespace}" teamdctl "${team}" state item set \
runner.active_port "${active_port}"
slowwait 2 bash -c "ip netns exec ${namespace} teamdctl ${team} state \
item get runner.active_port | grep -q ${active_port}"
}
# Wait for an interface to stop receiving traffic. If it keeps receiving traffic
# for the duration of the timeout, then return an error.
# Arguments:
# - namespace - The network namespace that the interface is in.
# - interface - The name of the interface.
wait_to_stop_receiving()
{
local namespace=$1
local interface=$2
echo "Waiting for ${interface} in ${namespace} to stop receiving"
slowwait 10 check_no_traffic "${interface}" "${NS2_IP}" \
"${namespace}"
}
# Test that active backup runner can change active ports.
# Globals:
# RET - Used by test infra, set by `check_err` functions.
teamd_test_active_backup()
{
export RET=0
start_listening_and_sending
### Scenario 1: Don't manually set active port, just make sure team
# works.
save_tcpdump_outputs "${NS2}" test_team2
did_interface_receive test_team2 "${NS2_IP}"
check_err $? "Traffic did not reach team interface in NS2."
clear_tcpdump_outputs test_team2
### Scenario 2: Choose active port.
set_active_port "${NS1}" test_team1 eth1
set_active_port "${NS2}" test_team2 eth1
wait_to_stop_receiving "${NS2}" eth0
save_tcpdump_outputs "${NS2}" eth0 eth1
did_interface_receive eth0 "${NS2_IP}"
check_fail $? "eth0 IS transmitting when inactive"
did_interface_receive eth1 "${NS2_IP}"
check_err $? "eth1 not transmitting when active"
clear_tcpdump_outputs eth0 eth1
### Scenario 3: Change active port.
set_active_port "${NS1}" test_team1 eth0
set_active_port "${NS2}" test_team2 eth0
wait_to_stop_receiving "${NS2}" eth1
save_tcpdump_outputs "${NS2}" eth0 eth1
did_interface_receive eth0 "${NS2_IP}"
check_err $? "eth0 not transmitting when active"
did_interface_receive eth1 "${NS2_IP}"
check_fail $? "eth1 IS transmitting when inactive"
clear_tcpdump_outputs eth0 eth1
log_test "teamd active backup runner test"
stop_sending_and_listening
}
require_command teamd
require_command teamdctl
require_command iperf3
require_command tcpdump
environment_create activebackup
tests_run
exit "${EXIT_STATUS}"