Skip to content

Commit

Permalink
ovn-tester: add the capability to run multiple independent clusters
Browse files Browse the repository at this point in the history
This is a preliminary patch to add ovn-ic support to ovn-tester.

Signed-off-by: Lorenzo Bianconi <lorenzo.bianconi@redhat.com>
  • Loading branch information
LorenzoBianconi committed Jun 30, 2023
1 parent 2b179a0 commit f100e67
Show file tree
Hide file tree
Showing 6 changed files with 183 additions and 98 deletions.
2 changes: 1 addition & 1 deletion do.sh
Original file line number Diff line number Diff line change
Expand Up @@ -449,7 +449,7 @@ function run_test() {

cluster_vars=""
for var in enable_ssl clustered_db monitor_all use_ovsdb_etcd \
node_net datapath_type n_relays n_workers; do
node_net datapath_type n_relays n_workers n_az; do
cluster_vars="${cluster_vars} $(get_cluster_var ${test_file} ${var})"
done
echo "-- Cluster vars: ${cluster_vars}"
Expand Down
2 changes: 2 additions & 0 deletions ovn-fake-multinode-utils/playbooks/bringup-cluster.yml
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@
tasks:
- name: Start central containers
environment:
CENTRAL_COUNT: '{{ n_az }}'
CHASSIS_COUNT: 0
CREATE_FAKE_VMS: no
ENABLE_ETCD: '{{ use_ovsdb_etcd }}'
Expand All @@ -46,6 +47,7 @@
tasks:
- name: Start worker containers
environment:
CENTRAL_COUNT: '{{ n_az }}'
CHASSIS_COUNT: 0
CREATE_FAKE_VMS: no
ENABLE_ETCD: '{{ use_ovsdb_etcd }}'
Expand Down
32 changes: 1 addition & 31 deletions ovn-fake-multinode-utils/translate_yaml.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,28 +25,6 @@ class GlobalConfig:
run_ipv6: bool = False


def calculate_node_remotes(
node_net: str, clustered_db: bool, n_relays: int, enable_ssl: bool
) -> str:
net = netaddr.IPNetwork(node_net)

ip_gen = net.iter_hosts()
# The first IP is assigned to the tester, skip it.
next(ip_gen)
if n_relays > 0:
skip = 3 if clustered_db else 1
for _ in range(0, skip):
next(ip_gen)
ip_range = range(0, n_relays)
else:
ip_range = range(0, 3 if clustered_db else 1)
if enable_ssl:
remotes = ["ssl:" + str(next(ip_gen)) + ":6642" for _ in ip_range]
else:
remotes = ["tcp:" + str(next(ip_gen)) + ":6642" for _ in ip_range]
return ','.join(remotes)


DEFAULT_N_VIPS = 2
DEFAULT_VIP_PORT = 80

Expand Down Expand Up @@ -119,7 +97,6 @@ class ClusterConfig:
db_inactivity_probe: int = 60000
node_net: str = "192.16.0.0/16"
enable_ssl: bool = True
node_remote: str = None
node_timeout_s: int = 20
internal_net: str = "16.0.0.0/16"
internal_net6: str = "16::/64"
Expand All @@ -131,6 +108,7 @@ class ClusterConfig:
cluster_net6: str = "16::/32"
n_workers: int = 2
n_relays: int = 0
n_az: int = 1
vips: Dict = None
vips6: Dict = None
vip_subnet: str = "4.0.0.0/8"
Expand All @@ -141,14 +119,6 @@ class ClusterConfig:

def __post_init__(self, **kwargs):
# Some defaults have to be calculated
if not self.node_remote:
self.node_remote = calculate_node_remotes(
self.node_net,
self.clustered_db,
self.n_relays,
self.enable_ssl,
)

if self.vips is None:
self.vips = calculate_vips(self.vip_subnet)

Expand Down
150 changes: 102 additions & 48 deletions ovn-tester/ovn_tester.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@
from ovn_sandbox import PhysicalNode
from ovn_workload import BrExConfig, ClusterConfig
from ovn_workload import CentralNode, WorkerNode, Cluster
from ovn_utils import DualStackSubnet
from ovn_utils import DualStackSubnet, NodeConf
from ovs.stream import Stream


Expand Down Expand Up @@ -70,8 +70,8 @@ def read_config(config):
raft_election_to=cluster_args['raft_election_to'],
node_net=netaddr.IPNetwork(cluster_args['node_net']),
n_relays=cluster_args['n_relays'],
n_az=cluster_args['n_az'],
enable_ssl=cluster_args['enable_ssl'],
node_remote=cluster_args['node_remote'],
northd_probe_interval=cluster_args['northd_probe_interval'],
db_inactivity_probe=cluster_args['db_inactivity_probe'],
node_timeout_s=cluster_args['node_timeout_s'],
Expand Down Expand Up @@ -182,36 +182,77 @@ def configure_tests(yaml, central_node, worker_nodes, global_cfg):


def create_nodes(cluster_config, central, workers):
mgmt_net = cluster_config.node_net
mgmt_ip = mgmt_net.ip + 2
internal_net = cluster_config.internal_net
external_net = cluster_config.external_net
gw_net = cluster_config.gw_net
db_containers = (
['ovn-central-1', 'ovn-central-2', 'ovn-central-3']
if cluster_config.clustered_db
else ['ovn-central']
)
node_az_conf = [
NodeConf(
cluster_config.node_net,
DualStackSubnet.next(
cluster_config.internal_net,
i * (cluster_config.n_workers // cluster_config.n_az),
),
DualStackSubnet.next(
cluster_config.external_net,
i * (cluster_config.n_workers // cluster_config.n_az),
),
DualStackSubnet.next(
cluster_config.gw_net,
i * (cluster_config.n_workers // cluster_config.n_az),
),
)
for i in range(cluster_config.n_az)
]

db_containers = [
[
f'ovn-central-az{i+1}-1',
f'ovn-central-az{i+1}-2',
f'ovn-central-az{i+1}-3'
if cluster_config.clustered_db
else f'ovn-central-az{i+1}',
]
for i in range(cluster_config.n_az)
]

relay_containers = [
f'ovn-relay-{i + 1}' for i in range(cluster_config.n_relays)
[
f'ovn-relay{i*cluster_config.n_az+j+1}'
for j in range(cluster_config.n_relays)
]
for i in range(cluster_config.n_az)
]
central_node = CentralNode(
central, db_containers, relay_containers, mgmt_net, mgmt_ip
)
worker_nodes = [
WorkerNode(

central_nodes = [
CentralNode(
central,
db_containers[i],
relay_containers[i],
node_az_conf[i].getMgmtNet(),
node_az_conf[i].getMgmtIp(),
node_az_conf[i].getGwNet(),
i,
)
for i in range(cluster_config.n_az)
]

worker_nodes = [[] for _ in range(cluster_config.n_az)]
for i in range(cluster_config.n_workers):
wn = WorkerNode(
workers[i % len(workers)],
f'ovn-scale-{i}',
mgmt_net,
mgmt_ip + i,
DualStackSubnet.next(internal_net, i),
DualStackSubnet.next(external_net, i),
gw_net,
node_az_conf[i % cluster_config.n_az].getMgmtNet(),
node_az_conf[i % cluster_config.n_az].getMgmtIp(),
DualStackSubnet.next(
node_az_conf[i % cluster_config.n_az].getIntNet(),
i // cluster_config.n_az,
),
DualStackSubnet.next(
node_az_conf[i % cluster_config.n_az].getExtNet(),
i // cluster_config.n_az,
),
node_az_conf[i % cluster_config.n_az].getGwNet(),
i,
)
for i in range(cluster_config.n_workers)
]
return central_node, worker_nodes
worker_nodes[i % cluster_config.n_az].append(wn)
return central_nodes, worker_nodes


def set_ssl_keys(cluster_cfg):
Expand All @@ -220,28 +261,38 @@ def set_ssl_keys(cluster_cfg):
Stream.ssl_set_ca_cert_file(cluster_cfg.ssl_cacert)


def prepare_test(central_node, worker_nodes, cluster_cfg, brex_cfg):
def prepare_test(central_nodes, worker_nodes, cluster_cfg, brex_cfg):
clusters = []
if cluster_cfg.enable_ssl:
set_ssl_keys(cluster_cfg)
ovn = Cluster(central_node, worker_nodes, cluster_cfg, brex_cfg)
with Context(ovn, "prepare_test"):
ovn.start()
return ovn
for i in range(0, len(central_nodes)):
ovn = Cluster(central_nodes[i], worker_nodes[i], cluster_cfg, brex_cfg)
with Context(ovn, f'prepare_test for cluster{i}'):
ovn.start()
clusters.append(ovn)

return clusters


def run_base_cluster_bringup(ovn, bringup_cfg, global_cfg):
# create ovn topology
with Context(ovn, "base_cluster_bringup", len(ovn.worker_nodes)) as ctx:
ovn.create_cluster_router("lr-cluster")
ovn.create_cluster_join_switch("ls-join")
ovn.create_cluster_load_balancer("lb-cluster", global_cfg)
for i in ctx:
worker = ovn.worker_nodes[i]
worker.provision(ovn)
ports = worker.provision_ports(ovn, bringup_cfg.n_pods_per_node)
worker.provision_load_balancers(ovn, ports, global_cfg)
worker.ping_ports(ovn, ports)
ovn.provision_lb_group()
for i in range(0, len(clusters)):
ovn = clusters[i]
# create ovn topology
with Context(
ovn, "base_cluster_bringup", len(ovn.worker_nodes)
) as ctx:
ovn.create_cluster_router(f'lr-cluster{i}')
ovn.create_cluster_join_switch(f'ls-join{i}')
ovn.create_cluster_load_balancer(f'lb-cluster{i}', global_cfg)
for i in ctx:
worker = ovn.worker_nodes[i]
worker.provision(ovn)
ports = worker.provision_ports(
ovn, bringup_cfg.n_pods_per_node
)
worker.provision_load_balancers(ovn, ports, global_cfg)
worker.ping_ports(ovn, ports)
ovn.provision_lb_group(f'cluster-lb-group{i}')


if __name__ == '__main__':
Expand All @@ -260,11 +311,14 @@ def run_base_cluster_bringup(ovn, bringup_cfg, global_cfg):
raise ovn_exceptions.OvnInvalidConfigException()

central, workers = read_physical_deployment(sys.argv[1], global_cfg)
central_node, worker_nodes = create_nodes(cluster_cfg, central, workers)
tests = configure_tests(config, central_node, worker_nodes, global_cfg)
central_nodes, worker_nodes = create_nodes(cluster_cfg, central, workers)
tests = configure_tests(
config, central_nodes[0], worker_nodes[0], global_cfg
)

ovn = prepare_test(central_node, worker_nodes, cluster_cfg, brex_cfg)
run_base_cluster_bringup(ovn, bringup_cfg, global_cfg)
clusters = prepare_test(central_nodes, worker_nodes, cluster_cfg, brex_cfg)
run_base_cluster_bringup(clusters, bringup_cfg, global_cfg)
# FIXME run workloads for all clusters
for test in tests:
test.run(ovn, global_cfg)
test.run(clusters[0], global_cfg)
sys.exit(0)
23 changes: 23 additions & 0 deletions ovn-tester/ovn_utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,29 @@ def external_host_provision(self, ip, gw, netns='ext-ns'):
self.run(cmd=cmd)


class NodeConf:
def __init__(self, mgmt_net, int_net, ext_net, gw_net):
self.mgmt_net = mgmt_net
self.int_net = int_net
self.ext_net = ext_net
self.gw_net = gw_net

def getMgmtNet(self):
return self.mgmt_net

def getMgmtIp(self):
return self.mgmt_net.ip + 2

def getIntNet(self):
return self.int_net

def getExtNet(self):
return self.ext_net

def getGwNet(self):
return self.gw_net


class DualStackSubnet:
def __init__(self, n4=None, n6=None):
self.n4 = n4
Expand Down
Loading

0 comments on commit f100e67

Please sign in to comment.