xref: /aosp_15_r20/external/openthread/tests/toranj/cli/test-502-multi-br-leader-failure-recovery.py (revision cfb92d1480a9e65faed56933e9c12405f45898b4)
1*cfb92d14SAndroid Build Coastguard Worker#!/usr/bin/env python3
2*cfb92d14SAndroid Build Coastguard Worker#
3*cfb92d14SAndroid Build Coastguard Worker#  Copyright (c) 2024, The OpenThread Authors.
4*cfb92d14SAndroid Build Coastguard Worker#  All rights reserved.
5*cfb92d14SAndroid Build Coastguard Worker#
6*cfb92d14SAndroid Build Coastguard Worker#  Redistribution and use in source and binary forms, with or without
7*cfb92d14SAndroid Build Coastguard Worker#  modification, are permitted provided that the following conditions are met:
8*cfb92d14SAndroid Build Coastguard Worker#  1. Redistributions of source code must retain the above copyright
9*cfb92d14SAndroid Build Coastguard Worker#     notice, this list of conditions and the following disclaimer.
10*cfb92d14SAndroid Build Coastguard Worker#  2. Redistributions in binary form must reproduce the above copyright
11*cfb92d14SAndroid Build Coastguard Worker#     notice, this list of conditions and the following disclaimer in the
12*cfb92d14SAndroid Build Coastguard Worker#     documentation and/or other materials provided with the distribution.
13*cfb92d14SAndroid Build Coastguard Worker#  3. Neither the name of the copyright holder nor the
14*cfb92d14SAndroid Build Coastguard Worker#     names of its contributors may be used to endorse or promote products
15*cfb92d14SAndroid Build Coastguard Worker#     derived from this software without specific prior written permission.
16*cfb92d14SAndroid Build Coastguard Worker#
17*cfb92d14SAndroid Build Coastguard Worker#  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18*cfb92d14SAndroid Build Coastguard Worker#  AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19*cfb92d14SAndroid Build Coastguard Worker#  IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20*cfb92d14SAndroid Build Coastguard Worker#  ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
21*cfb92d14SAndroid Build Coastguard Worker#  LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22*cfb92d14SAndroid Build Coastguard Worker#  CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23*cfb92d14SAndroid Build Coastguard Worker#  SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24*cfb92d14SAndroid Build Coastguard Worker#  INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25*cfb92d14SAndroid Build Coastguard Worker#  CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26*cfb92d14SAndroid Build Coastguard Worker#  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27*cfb92d14SAndroid Build Coastguard Worker#  POSSIBILITY OF SUCH DAMAGE.
28*cfb92d14SAndroid Build Coastguard Worker
29*cfb92d14SAndroid Build Coastguard Workerfrom cli import verify
30*cfb92d14SAndroid Build Coastguard Workerfrom cli import verify_within
31*cfb92d14SAndroid Build Coastguard Workerimport cli
32*cfb92d14SAndroid Build Coastguard Workerimport time
33*cfb92d14SAndroid Build Coastguard Worker
34*cfb92d14SAndroid Build Coastguard Worker# -----------------------------------------------------------------------------------------------------------------------
35*cfb92d14SAndroid Build Coastguard Worker# Test description:
36*cfb92d14SAndroid Build Coastguard Worker#
37*cfb92d14SAndroid Build Coastguard Worker# Network with two BRs, BR1 acting as leader. Removing BR1 ensuring BR2 taking over.
38*cfb92d14SAndroid Build Coastguard Worker#
39*cfb92d14SAndroid Build Coastguard Worker#      ________________
40*cfb92d14SAndroid Build Coastguard Worker#     /                \
41*cfb92d14SAndroid Build Coastguard Worker#   br1 --- router --- br2
42*cfb92d14SAndroid Build Coastguard Worker#   /        / \        \
43*cfb92d14SAndroid Build Coastguard Worker#  c1       c2  c3      c4
44*cfb92d14SAndroid Build Coastguard Worker#
45*cfb92d14SAndroid Build Coastguard Worker
46*cfb92d14SAndroid Build Coastguard Workertest_name = __file__[:-3] if __file__.endswith('.py') else __file__
47*cfb92d14SAndroid Build Coastguard Workerprint('-' * 120)
48*cfb92d14SAndroid Build Coastguard Workerprint('Starting \'{}\''.format(test_name))
49*cfb92d14SAndroid Build Coastguard Worker
50*cfb92d14SAndroid Build Coastguard Worker# -----------------------------------------------------------------------------------------------------------------------
51*cfb92d14SAndroid Build Coastguard Worker# Creating `cli.Nodes` instances
52*cfb92d14SAndroid Build Coastguard Worker
53*cfb92d14SAndroid Build Coastguard Workerspeedup = 60
54*cfb92d14SAndroid Build Coastguard Workercli.Node.set_time_speedup_factor(speedup)
55*cfb92d14SAndroid Build Coastguard Worker
56*cfb92d14SAndroid Build Coastguard Workerbr1 = cli.Node()
57*cfb92d14SAndroid Build Coastguard Workerbr2 = cli.Node()
58*cfb92d14SAndroid Build Coastguard Workerrouter = cli.Node()
59*cfb92d14SAndroid Build Coastguard Workerc1 = cli.Node()
60*cfb92d14SAndroid Build Coastguard Workerc2 = cli.Node()
61*cfb92d14SAndroid Build Coastguard Workerc3 = cli.Node()
62*cfb92d14SAndroid Build Coastguard Workerc4 = cli.Node()
63*cfb92d14SAndroid Build Coastguard Worker
64*cfb92d14SAndroid Build Coastguard WorkerIF_INDEX = 1
65*cfb92d14SAndroid Build Coastguard Worker
66*cfb92d14SAndroid Build Coastguard Worker# -----------------------------------------------------------------------------------------------------------------------
67*cfb92d14SAndroid Build Coastguard Worker# Form topology
68*cfb92d14SAndroid Build Coastguard Worker
69*cfb92d14SAndroid Build Coastguard Workerbr1.allowlist_node(router)
70*cfb92d14SAndroid Build Coastguard Workerbr1.allowlist_node(br2)
71*cfb92d14SAndroid Build Coastguard Workerbr1.allowlist_node(c1)
72*cfb92d14SAndroid Build Coastguard Worker
73*cfb92d14SAndroid Build Coastguard Workerbr2.allowlist_node(router)
74*cfb92d14SAndroid Build Coastguard Workerbr2.allowlist_node(br1)
75*cfb92d14SAndroid Build Coastguard Workerbr2.allowlist_node(c4)
76*cfb92d14SAndroid Build Coastguard Worker
77*cfb92d14SAndroid Build Coastguard Workerrouter.allowlist_node(br1)
78*cfb92d14SAndroid Build Coastguard Workerrouter.allowlist_node(br2)
79*cfb92d14SAndroid Build Coastguard Workerrouter.allowlist_node(c2)
80*cfb92d14SAndroid Build Coastguard Workerrouter.allowlist_node(c3)
81*cfb92d14SAndroid Build Coastguard Worker
82*cfb92d14SAndroid Build Coastguard Workerc1.allowlist_node(br1)
83*cfb92d14SAndroid Build Coastguard Worker
84*cfb92d14SAndroid Build Coastguard Workerc2.allowlist_node(router)
85*cfb92d14SAndroid Build Coastguard Workerc3.allowlist_node(router)
86*cfb92d14SAndroid Build Coastguard Worker
87*cfb92d14SAndroid Build Coastguard Workerc4.allowlist_node(br2)
88*cfb92d14SAndroid Build Coastguard Worker
89*cfb92d14SAndroid Build Coastguard Workerbr1.form("multi-br")
90*cfb92d14SAndroid Build Coastguard Workerbr2.join(br1)
91*cfb92d14SAndroid Build Coastguard Workerrouter.join(br1)
92*cfb92d14SAndroid Build Coastguard Workerc1.join(br1, cli.JOIN_TYPE_END_DEVICE)
93*cfb92d14SAndroid Build Coastguard Workerc2.join(br1, cli.JOIN_TYPE_END_DEVICE)
94*cfb92d14SAndroid Build Coastguard Workerc3.join(br1, cli.JOIN_TYPE_END_DEVICE)
95*cfb92d14SAndroid Build Coastguard Workerc4.join(br1, cli.JOIN_TYPE_END_DEVICE)
96*cfb92d14SAndroid Build Coastguard Worker
97*cfb92d14SAndroid Build Coastguard Workerverify(br1.get_state() == 'leader')
98*cfb92d14SAndroid Build Coastguard Workerverify(br2.get_state() == 'router')
99*cfb92d14SAndroid Build Coastguard Workerverify(router.get_state() == 'router')
100*cfb92d14SAndroid Build Coastguard Workerverify(c1.get_state() == 'child')
101*cfb92d14SAndroid Build Coastguard Workerverify(c2.get_state() == 'child')
102*cfb92d14SAndroid Build Coastguard Workerverify(c3.get_state() == 'child')
103*cfb92d14SAndroid Build Coastguard Workerverify(c4.get_state() == 'child')
104*cfb92d14SAndroid Build Coastguard Worker
105*cfb92d14SAndroid Build Coastguard Workernodes_non_br = [router, c1, c2, c3, c4]
106*cfb92d14SAndroid Build Coastguard Worker
107*cfb92d14SAndroid Build Coastguard Worker# -----------------------------------------------------------------------------------------------------------------------
108*cfb92d14SAndroid Build Coastguard Worker# Test implementation
109*cfb92d14SAndroid Build Coastguard Worker
110*cfb92d14SAndroid Build Coastguard Worker# Start the first BR
111*cfb92d14SAndroid Build Coastguard Worker
112*cfb92d14SAndroid Build Coastguard Workerbr1.srp_server_set_addr_mode('unicast')
113*cfb92d14SAndroid Build Coastguard Workerbr1.srp_server_auto_enable()
114*cfb92d14SAndroid Build Coastguard Worker
115*cfb92d14SAndroid Build Coastguard Workerbr1.br_init(IF_INDEX, 1)
116*cfb92d14SAndroid Build Coastguard Workerbr1.br_enable()
117*cfb92d14SAndroid Build Coastguard Worker
118*cfb92d14SAndroid Build Coastguard Workertime.sleep(1)
119*cfb92d14SAndroid Build Coastguard Workerverify(br1.br_get_state() == 'running')
120*cfb92d14SAndroid Build Coastguard Worker
121*cfb92d14SAndroid Build Coastguard Workerbr1_local_omr = br1.br_get_local_omrprefix()
122*cfb92d14SAndroid Build Coastguard Workerbr1_favored_omr = br1.br_get_favored_omrprefix().split()[0]
123*cfb92d14SAndroid Build Coastguard Workerverify(br1_local_omr == br1_favored_omr)
124*cfb92d14SAndroid Build Coastguard Worker
125*cfb92d14SAndroid Build Coastguard Workerbr1_local_onlink = br1.br_get_local_onlinkprefix()
126*cfb92d14SAndroid Build Coastguard Workerbr1_favored_onlink = br1.br_get_favored_onlinkprefix().split()[0]
127*cfb92d14SAndroid Build Coastguard Workerverify(br1_local_onlink == br1_favored_onlink)
128*cfb92d14SAndroid Build Coastguard Worker
129*cfb92d14SAndroid Build Coastguard Worker# Start the second BR
130*cfb92d14SAndroid Build Coastguard Worker
131*cfb92d14SAndroid Build Coastguard Workerbr2.br_init(IF_INDEX, 1)
132*cfb92d14SAndroid Build Coastguard Workerbr2.br_enable()
133*cfb92d14SAndroid Build Coastguard Worker
134*cfb92d14SAndroid Build Coastguard Workertime.sleep(1)
135*cfb92d14SAndroid Build Coastguard Workerverify(br2.br_get_state() == 'running')
136*cfb92d14SAndroid Build Coastguard Worker
137*cfb92d14SAndroid Build Coastguard Workerbr2_local_omr = br2.br_get_local_omrprefix()
138*cfb92d14SAndroid Build Coastguard Workerbr2_favored_omr = br2.br_get_favored_omrprefix().split()[0]
139*cfb92d14SAndroid Build Coastguard Workerverify(br2_favored_omr == br1_favored_omr)
140*cfb92d14SAndroid Build Coastguard Worker
141*cfb92d14SAndroid Build Coastguard Workerbr2_favored_onlink = br2.br_get_favored_onlinkprefix().split()[0]
142*cfb92d14SAndroid Build Coastguard Workerverify(br2_favored_onlink == br1_favored_onlink)
143*cfb92d14SAndroid Build Coastguard Worker
144*cfb92d14SAndroid Build Coastguard Workerverify(br1.srp_server_get_state() == 'running')
145*cfb92d14SAndroid Build Coastguard Workerverify(br2.srp_server_get_state() == 'disabled')
146*cfb92d14SAndroid Build Coastguard Worker
147*cfb92d14SAndroid Build Coastguard Worker# Register SRP services on all nodes
148*cfb92d14SAndroid Build Coastguard Worker
149*cfb92d14SAndroid Build Coastguard Workerfor node in nodes_non_br:
150*cfb92d14SAndroid Build Coastguard Worker    node.srp_client_enable_auto_start_mode()
151*cfb92d14SAndroid Build Coastguard Worker    verify(node.srp_client_get_auto_start_mode() == 'Enabled')
152*cfb92d14SAndroid Build Coastguard Worker    node.srp_client_set_host_name('host' + str(node.index))
153*cfb92d14SAndroid Build Coastguard Worker    node.srp_client_enable_auto_host_address()
154*cfb92d14SAndroid Build Coastguard Worker    node.srp_client_add_service('srv' + str(node.index), '_test._udp', 777, 0, 0)
155*cfb92d14SAndroid Build Coastguard Worker
156*cfb92d14SAndroid Build Coastguard Workertime.sleep(1)
157*cfb92d14SAndroid Build Coastguard Worker
158*cfb92d14SAndroid Build Coastguard Workerhosts = br1.srp_server_get_hosts()
159*cfb92d14SAndroid Build Coastguard Workerverify(len(hosts) == len(nodes_non_br))
160*cfb92d14SAndroid Build Coastguard Worker
161*cfb92d14SAndroid Build Coastguard Workerservices = br1.srp_server_get_services()
162*cfb92d14SAndroid Build Coastguard Workerverify(len(services) == len(nodes_non_br))
163*cfb92d14SAndroid Build Coastguard Worker
164*cfb92d14SAndroid Build Coastguard Worker# Ensure that all registered addresses are derived from BR1 OMR.
165*cfb92d14SAndroid Build Coastguard Worker
166*cfb92d14SAndroid Build Coastguard Workerfor host in hosts:
167*cfb92d14SAndroid Build Coastguard Worker    verify(host['addresses'][0].startswith(br1_local_omr[:-4]))
168*cfb92d14SAndroid Build Coastguard Worker
169*cfb92d14SAndroid Build Coastguard Worker# Start SRP server on BR2
170*cfb92d14SAndroid Build Coastguard Worker
171*cfb92d14SAndroid Build Coastguard Workerbr2.srp_server_set_addr_mode('unicast')
172*cfb92d14SAndroid Build Coastguard Workerbr2.srp_server_auto_enable()
173*cfb92d14SAndroid Build Coastguard Worker
174*cfb92d14SAndroid Build Coastguard Workertime.sleep(1)
175*cfb92d14SAndroid Build Coastguard Worker
176*cfb92d14SAndroid Build Coastguard Workerverify(br2.srp_server_get_state() == 'running')
177*cfb92d14SAndroid Build Coastguard Worker
178*cfb92d14SAndroid Build Coastguard Worker# De-activate BR1
179*cfb92d14SAndroid Build Coastguard Worker
180*cfb92d14SAndroid Build Coastguard Workerbr1.br_disable()
181*cfb92d14SAndroid Build Coastguard Workerbr1.thread_stop()
182*cfb92d14SAndroid Build Coastguard Workerbr1.interface_down()
183*cfb92d14SAndroid Build Coastguard Workerdel br1
184*cfb92d14SAndroid Build Coastguard Worker
185*cfb92d14SAndroid Build Coastguard Workerc1.allowlist_node(br2)
186*cfb92d14SAndroid Build Coastguard Workerbr2.allowlist_node(c1)
187*cfb92d14SAndroid Build Coastguard Worker
188*cfb92d14SAndroid Build Coastguard Worker# Wait long enough for BR2 to take over
189*cfb92d14SAndroid Build Coastguard Worker
190*cfb92d14SAndroid Build Coastguard Workertime.sleep(5)
191*cfb92d14SAndroid Build Coastguard Worker
192*cfb92d14SAndroid Build Coastguard Worker# Validate that everything is registered with BR2
193*cfb92d14SAndroid Build Coastguard Worker
194*cfb92d14SAndroid Build Coastguard Workerhosts = br2.srp_server_get_hosts()
195*cfb92d14SAndroid Build Coastguard Workerverify(len(hosts) == len(nodes_non_br))
196*cfb92d14SAndroid Build Coastguard Worker
197*cfb92d14SAndroid Build Coastguard Workerservices = br2.srp_server_get_services()
198*cfb92d14SAndroid Build Coastguard Workerverify(len(services) == len(nodes_non_br))
199*cfb92d14SAndroid Build Coastguard Worker
200*cfb92d14SAndroid Build Coastguard Worker# Ensure that all registered addresses are now derived from BR2
201*cfb92d14SAndroid Build Coastguard Worker# OMR prefix.
202*cfb92d14SAndroid Build Coastguard Worker
203*cfb92d14SAndroid Build Coastguard Workerfor host in hosts:
204*cfb92d14SAndroid Build Coastguard Worker    verify(host['addresses'][0].startswith(br2_local_omr[:-4]))
205*cfb92d14SAndroid Build Coastguard Worker
206*cfb92d14SAndroid Build Coastguard Worker# -----------------------------------------------------------------------------------------------------------------------
207*cfb92d14SAndroid Build Coastguard Worker# Test finished
208*cfb92d14SAndroid Build Coastguard Worker
209*cfb92d14SAndroid Build Coastguard Workercli.Node.finalize_all_nodes()
210*cfb92d14SAndroid Build Coastguard Worker
211*cfb92d14SAndroid Build Coastguard Workerprint('\'{}\' passed.'.format(test_name))
212