1*cfb92d14SAndroid Build Coastguard Worker#!/usr/bin/env python3 2*cfb92d14SAndroid Build Coastguard Worker# 3*cfb92d14SAndroid Build Coastguard Worker# Copyright (c) 2022, 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 Worker 30*cfb92d14SAndroid Build Coastguard Workerimport unittest 31*cfb92d14SAndroid Build Coastguard Worker 32*cfb92d14SAndroid Build Coastguard Workerimport thread_cert 33*cfb92d14SAndroid Build Coastguard Workerimport config 34*cfb92d14SAndroid Build Coastguard Workerfrom pktverify.consts import MLE_CHILD_UPDATE_REQUEST, TIMEOUT_TLV, ADDR_REL_URI 35*cfb92d14SAndroid Build Coastguard Workerfrom pktverify.packet_verifier import PacketVerifier 36*cfb92d14SAndroid Build Coastguard Worker 37*cfb92d14SAndroid Build Coastguard Worker# Test description: 38*cfb92d14SAndroid Build Coastguard Worker# This test verifies that detaching function can send correct "goodbye" messages. 39*cfb92d14SAndroid Build Coastguard Worker# 40*cfb92d14SAndroid Build Coastguard Worker# Topology: 41*cfb92d14SAndroid Build Coastguard Worker# 42*cfb92d14SAndroid Build Coastguard Worker# CHILD_1 ----- ROUTER_1 ----- LEADER 43*cfb92d14SAndroid Build Coastguard Worker# 44*cfb92d14SAndroid Build Coastguard Worker# 45*cfb92d14SAndroid Build Coastguard Worker 46*cfb92d14SAndroid Build Coastguard WorkerLEADER = 1 47*cfb92d14SAndroid Build Coastguard WorkerROUTER_1 = 2 48*cfb92d14SAndroid Build Coastguard WorkerCHILD_1 = 3 49*cfb92d14SAndroid Build Coastguard Worker 50*cfb92d14SAndroid Build Coastguard Worker 51*cfb92d14SAndroid Build Coastguard Workerclass TestDetach(thread_cert.TestCase): 52*cfb92d14SAndroid Build Coastguard Worker USE_MESSAGE_FACTORY = False 53*cfb92d14SAndroid Build Coastguard Worker SUPPORT_NCP = False 54*cfb92d14SAndroid Build Coastguard Worker 55*cfb92d14SAndroid Build Coastguard Worker TOPOLOGY = { 56*cfb92d14SAndroid Build Coastguard Worker LEADER: { 57*cfb92d14SAndroid Build Coastguard Worker 'name': 'Leader', 58*cfb92d14SAndroid Build Coastguard Worker 'allowlist': [ROUTER_1], 59*cfb92d14SAndroid Build Coastguard Worker 'mode': 'rdn', 60*cfb92d14SAndroid Build Coastguard Worker }, 61*cfb92d14SAndroid Build Coastguard Worker ROUTER_1: { 62*cfb92d14SAndroid Build Coastguard Worker 'name': 'Router_1', 63*cfb92d14SAndroid Build Coastguard Worker 'allowlist': [LEADER, CHILD_1], 64*cfb92d14SAndroid Build Coastguard Worker 'mode': 'rdn', 65*cfb92d14SAndroid Build Coastguard Worker }, 66*cfb92d14SAndroid Build Coastguard Worker CHILD_1: { 67*cfb92d14SAndroid Build Coastguard Worker 'name': 'Child_1', 68*cfb92d14SAndroid Build Coastguard Worker 'is_mtd': True, 69*cfb92d14SAndroid Build Coastguard Worker 'allowlist': [ROUTER_1], 70*cfb92d14SAndroid Build Coastguard Worker 'mode': '-', 71*cfb92d14SAndroid Build Coastguard Worker 'timeout': 10, 72*cfb92d14SAndroid Build Coastguard Worker }, 73*cfb92d14SAndroid Build Coastguard Worker } 74*cfb92d14SAndroid Build Coastguard Worker 75*cfb92d14SAndroid Build Coastguard Worker def test(self): 76*cfb92d14SAndroid Build Coastguard Worker leader = self.nodes[LEADER] 77*cfb92d14SAndroid Build Coastguard Worker router1 = self.nodes[ROUTER_1] 78*cfb92d14SAndroid Build Coastguard Worker child1 = self.nodes[CHILD_1] 79*cfb92d14SAndroid Build Coastguard Worker 80*cfb92d14SAndroid Build Coastguard Worker leader.start() 81*cfb92d14SAndroid Build Coastguard Worker self.simulator.go(config.LEADER_STARTUP_DELAY) 82*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(leader.get_state(), 'leader') 83*cfb92d14SAndroid Build Coastguard Worker 84*cfb92d14SAndroid Build Coastguard Worker router1.start() 85*cfb92d14SAndroid Build Coastguard Worker self.simulator.go(config.ROUTER_STARTUP_DELAY) 86*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(router1.get_state(), 'router') 87*cfb92d14SAndroid Build Coastguard Worker router1_rloc16 = router1.get_addr16() 88*cfb92d14SAndroid Build Coastguard Worker self.assertTrue(list(filter(lambda x: x[1]['rloc16'] == router1_rloc16, leader.router_table().items()))) 89*cfb92d14SAndroid Build Coastguard Worker 90*cfb92d14SAndroid Build Coastguard Worker self.collect_rloc16s() 91*cfb92d14SAndroid Build Coastguard Worker 92*cfb92d14SAndroid Build Coastguard Worker child1.start() 93*cfb92d14SAndroid Build Coastguard Worker self.simulator.go(7) 94*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(child1.get_state(), 'child') 95*cfb92d14SAndroid Build Coastguard Worker child_table = router1.get_child_table() 96*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(len(child_table), 1) 97*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(child_table[1]['timeout'], 10) 98*cfb92d14SAndroid Build Coastguard Worker 99*cfb92d14SAndroid Build Coastguard Worker child1.detach() 100*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(child1.get_state(), 'disabled') 101*cfb92d14SAndroid Build Coastguard Worker self.assertFalse(router1.get_child_table()) 102*cfb92d14SAndroid Build Coastguard Worker 103*cfb92d14SAndroid Build Coastguard Worker router1.detach() 104*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(router1.get_state(), 'disabled') 105*cfb92d14SAndroid Build Coastguard Worker self.assertFalse(list(filter(lambda x: x[1]['rloc16'] == router1_rloc16, leader.router_table().items()))) 106*cfb92d14SAndroid Build Coastguard Worker 107*cfb92d14SAndroid Build Coastguard Worker router1.start() 108*cfb92d14SAndroid Build Coastguard Worker self.simulator.go(config.ROUTER_STARTUP_DELAY) 109*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(router1.get_state(), 'router') 110*cfb92d14SAndroid Build Coastguard Worker 111*cfb92d14SAndroid Build Coastguard Worker child1.start() 112*cfb92d14SAndroid Build Coastguard Worker self.simulator.go(7) 113*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(child1.get_state(), 'child') 114*cfb92d14SAndroid Build Coastguard Worker child_table = router1.get_child_table() 115*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(len(child_table), 1) 116*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(child_table[2]['timeout'], 10) 117*cfb92d14SAndroid Build Coastguard Worker 118*cfb92d14SAndroid Build Coastguard Worker router1.thread_stop() 119*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(router1.get_state(), 'disabled') 120*cfb92d14SAndroid Build Coastguard Worker child1.detach() 121*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(child1.get_state(), 'disabled') 122*cfb92d14SAndroid Build Coastguard Worker 123*cfb92d14SAndroid Build Coastguard Worker router1.start() 124*cfb92d14SAndroid Build Coastguard Worker self.simulator.go(config.ROUTER_STARTUP_DELAY) 125*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(router1.get_state(), 'router') 126*cfb92d14SAndroid Build Coastguard Worker 127*cfb92d14SAndroid Build Coastguard Worker child1.start() 128*cfb92d14SAndroid Build Coastguard Worker self.simulator.go(7) 129*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(child1.get_state(), 'child') 130*cfb92d14SAndroid Build Coastguard Worker 131*cfb92d14SAndroid Build Coastguard Worker leader.detach() 132*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(leader.get_state(), 'disabled') 133*cfb92d14SAndroid Build Coastguard Worker 134*cfb92d14SAndroid Build Coastguard Worker self.assertTrue(child1.ping(router1.get_mleid(), timeout=20)) 135*cfb92d14SAndroid Build Coastguard Worker 136*cfb92d14SAndroid Build Coastguard Worker router1.detach() 137*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(router1.get_state(), 'disabled') 138*cfb92d14SAndroid Build Coastguard Worker 139*cfb92d14SAndroid Build Coastguard Worker leader.detach() 140*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(leader.get_state(), 'disabled') 141*cfb92d14SAndroid Build Coastguard Worker 142*cfb92d14SAndroid Build Coastguard Worker leader.start() 143*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(leader.get_state(), 'detached') 144*cfb92d14SAndroid Build Coastguard Worker leader.detach() 145*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(leader.get_state(), 'disabled') 146*cfb92d14SAndroid Build Coastguard Worker 147*cfb92d14SAndroid Build Coastguard Worker leader.start() 148*cfb92d14SAndroid Build Coastguard Worker # leader didn't become leader after the last start(), so it re-syncs in a non-critical manner thus taking ROUTER_RESET_DELAY to recover 149*cfb92d14SAndroid Build Coastguard Worker self.simulator.go(config.ROUTER_RESET_DELAY / 2) 150*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(leader.get_state(), 'detached') 151*cfb92d14SAndroid Build Coastguard Worker self.simulator.go(config.ROUTER_RESET_DELAY / 2) 152*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(leader.get_state(), 'leader') 153*cfb92d14SAndroid Build Coastguard Worker router1.start() 154*cfb92d14SAndroid Build Coastguard Worker self.simulator.go(config.ROUTER_RESET_DELAY) 155*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(router1.get_state(), 'router') 156*cfb92d14SAndroid Build Coastguard Worker 157*cfb92d14SAndroid Build Coastguard Worker leader.thread_stop() 158*cfb92d14SAndroid Build Coastguard Worker router1.detach(is_async=True) 159*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(router1.get_state(), 'router') 160*cfb92d14SAndroid Build Coastguard Worker router1.thread_stop() 161*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(router1.get_state(), 'disabled') 162*cfb92d14SAndroid Build Coastguard Worker router1.detach() 163*cfb92d14SAndroid Build Coastguard Worker self.assertEqual(router1.get_state(), 'disabled') 164*cfb92d14SAndroid Build Coastguard Worker 165*cfb92d14SAndroid Build Coastguard Worker def verify(self, pv: PacketVerifier): 166*cfb92d14SAndroid Build Coastguard Worker pkts = pv.pkts 167*cfb92d14SAndroid Build Coastguard Worker pv.summary.show() 168*cfb92d14SAndroid Build Coastguard Worker 169*cfb92d14SAndroid Build Coastguard Worker leader = pv.vars['Leader'] 170*cfb92d14SAndroid Build Coastguard Worker router1 = pv.vars['Router_1'] 171*cfb92d14SAndroid Build Coastguard Worker child1 = pv.vars['Child_1'] 172*cfb92d14SAndroid Build Coastguard Worker leader_rloc16 = pv.vars['Leader_RLOC16'] 173*cfb92d14SAndroid Build Coastguard Worker 174*cfb92d14SAndroid Build Coastguard Worker pkts.filter_wpan_src64(child1).filter_mle_cmd(MLE_CHILD_UPDATE_REQUEST).filter_wpan_dst64( 175*cfb92d14SAndroid Build Coastguard Worker router1).must_next().must_verify(lambda p: TIMEOUT_TLV in set(p.mle.tlv.type) and p.mle.tlv.timeout == 0) 176*cfb92d14SAndroid Build Coastguard Worker pkts.filter_wpan_src64(router1).filter_coap_request(ADDR_REL_URI).filter_wpan_dst16(leader_rloc16).must_next() 177*cfb92d14SAndroid Build Coastguard Worker pkts.filter_wpan_src64(child1).filter_mle_cmd(MLE_CHILD_UPDATE_REQUEST).filter_wpan_dst64( 178*cfb92d14SAndroid Build Coastguard Worker router1).must_next().must_verify(lambda p: TIMEOUT_TLV in set(p.mle.tlv.type) and p.mle.tlv.timeout == 0) 179*cfb92d14SAndroid Build Coastguard Worker pkts.filter_wpan_src64(leader).filter_coap_request(ADDR_REL_URI).must_not_next() 180*cfb92d14SAndroid Build Coastguard Worker pkts.filter_wpan_src64(router1).filter_coap_request(ADDR_REL_URI).filter_wpan_dst16(leader_rloc16).must_next() 181*cfb92d14SAndroid Build Coastguard Worker 182*cfb92d14SAndroid Build Coastguard Worker 183*cfb92d14SAndroid Build Coastguard Workerif __name__ == '__main__': 184*cfb92d14SAndroid Build Coastguard Worker unittest.main() 185