mirror of
				https://git.kernel.org/pub/scm/network/wireless/iwd.git
				synced 2025-10-26 01:27:29 +02:00 
			
		
		
		
	 4c6173ee89
			
		
	
	
		4c6173ee89
		
	
	
	
	
		
			
			Updated testFT-SAE-roam to use the TestContext APIs as well as fixed the failure which was introduced after requiring stricter AKM logic for SAE networks. The new failure was due to the hostapd config not including the standard SAE AKM which is actually required by the spec.
		
			
				
	
	
		
			203 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			203 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #! /usr/bin/python3
 | |
| 
 | |
| import unittest
 | |
| import sys, os
 | |
| 
 | |
| sys.path.append('../util')
 | |
| import iwd
 | |
| from iwd import IWD
 | |
| from iwd import PSKAgent
 | |
| from iwd import NetworkType
 | |
| from hwsim import Hwsim
 | |
| from hostapd import HostapdCLI
 | |
| import testutil
 | |
| from config import ctx
 | |
| 
 | |
| class Test(unittest.TestCase):
 | |
|     def test_roam_success(self):
 | |
|         wd = IWD()
 | |
| 
 | |
|         hwsim = Hwsim()
 | |
| 
 | |
|         rule0 = hwsim.rules.create()
 | |
|         rule0.source = self.bss_radio[0].addresses[0]
 | |
|         rule0.bidirectional = True
 | |
| 
 | |
|         rule1 = hwsim.rules.create()
 | |
|         rule1.source = self.bss_radio[1].addresses[0]
 | |
|         rule1.bidirectional = True
 | |
| 
 | |
|         rule2 = hwsim.rules.create()
 | |
|         rule2.source = self.bss_radio[2].addresses[0]
 | |
|         rule2.bidirectional = True
 | |
| 
 | |
|         psk_agent = PSKAgent("EasilyGuessedPassword")
 | |
|         wd.register_psk_agent(psk_agent)
 | |
| 
 | |
|         device = wd.list_devices(1)[0]
 | |
| 
 | |
|         # Check that iwd selects BSS 0 first
 | |
|         rule0.signal = -2000
 | |
|         rule1.signal = -2500
 | |
|         rule2.signal = -3000
 | |
| 
 | |
|         condition = 'not obj.scanning'
 | |
|         wd.wait_for_object_condition(device, condition)
 | |
| 
 | |
|         device.scan()
 | |
| 
 | |
|         condition = 'obj.scanning'
 | |
|         wd.wait_for_object_condition(device, condition)
 | |
| 
 | |
|         condition = 'not obj.scanning'
 | |
|         wd.wait_for_object_condition(device, condition)
 | |
| 
 | |
|         ordered_network = device.get_ordered_network('TestFT')
 | |
| 
 | |
|         self.assertEqual(ordered_network.type, NetworkType.psk)
 | |
|         self.assertEqual(ordered_network.signal_strength, -2000)
 | |
| 
 | |
|         condition = 'not obj.connected'
 | |
|         wd.wait_for_object_condition(ordered_network.network_object, condition)
 | |
| 
 | |
|         self.assertFalse(self.bss_hostapd[0].list_sta())
 | |
|         self.assertFalse(self.bss_hostapd[1].list_sta())
 | |
| 
 | |
|         ordered_network.network_object.connect()
 | |
| 
 | |
|         condition = 'obj.connected'
 | |
|         wd.wait_for_object_condition(ordered_network.network_object, condition)
 | |
| 
 | |
|         self.assertTrue(self.bss_hostapd[0].list_sta())
 | |
|         self.assertFalse(self.bss_hostapd[1].list_sta())
 | |
| 
 | |
|         wd.unregister_psk_agent(psk_agent)
 | |
| 
 | |
|         testutil.test_iface_operstate(device.name)
 | |
|         testutil.test_ifaces_connected(self.bss_hostapd[0].ifname, device.name)
 | |
|         self.assertRaises(Exception, testutil.test_ifaces_connected,
 | |
|                           (self.bss_hostapd[1].ifname, device.name))
 | |
| 
 | |
|         # Check that iwd starts transition to BSS 1 in less than 10 seconds.
 | |
|         # The 10 seconds is longer than needed to scan on just two channels
 | |
|         # but short enough that a full scan on the 2.4 + 5.8 bands supported
 | |
|         # by mac80211_hwsim will not finish.  If this times out then, but
 | |
|         # device_roam_trigger_cb has happened, it probably means that
 | |
|         # Neighbor Reports are broken.
 | |
|         rule0.signal = -8000
 | |
| 
 | |
|         condition = 'obj.state == DeviceState.roaming'
 | |
|         wd.wait_for_object_condition(device, condition)
 | |
| 
 | |
|         # Check that iwd is on BSS 1 once out of roaming state and doesn't
 | |
|         # go through 'disconnected', 'autoconnect', 'connecting' in between
 | |
|         condition = 'obj.state != DeviceState.roaming'
 | |
|         wd.wait_for_object_condition(device, condition)
 | |
| 
 | |
|         rule1.signal = -2000
 | |
| 
 | |
|         # wait for IWD's signal levels to recover
 | |
|         wd.wait(5)
 | |
| 
 | |
|         self.assertEqual(device.state, iwd.DeviceState.connected)
 | |
|         self.assertTrue(self.bss_hostapd[1].list_sta())
 | |
| 
 | |
|         testutil.test_iface_operstate(device.name)
 | |
|         testutil.test_ifaces_connected(self.bss_hostapd[1].ifname, device.name)
 | |
|         self.assertRaises(Exception, testutil.test_ifaces_connected,
 | |
|                           (self.bss_hostapd[0].ifname, device.name))
 | |
| 
 | |
|         # test FT-PSK after FT-SAE
 | |
|         rule1.signal = -8000
 | |
|         rule0.signal = -8000
 | |
|         rule2.signal = -1000
 | |
| 
 | |
|         condition = 'obj.state == DeviceState.roaming'
 | |
|         wd.wait_for_object_condition(device, condition)
 | |
| 
 | |
|         condition = 'obj.state != DeviceState.roaming'
 | |
|         wd.wait_for_object_condition(device, condition)
 | |
| 
 | |
|         self.assertEqual(device.state, iwd.DeviceState.connected)
 | |
|         self.assertTrue(self.bss_hostapd[2].list_sta())
 | |
| 
 | |
|         testutil.test_iface_operstate(device.name)
 | |
|         testutil.test_ifaces_connected(self.bss_hostapd[2].ifname, device.name)
 | |
|         self.assertRaises(Exception, testutil.test_ifaces_connected,
 | |
|                             (self.bss_hostapd[1].ifname, device.name))
 | |
| 
 | |
|     def tearDown(self):
 | |
|         os.system('ifconfig "' + self.bss_hostapd[0].ifname + '" down')
 | |
|         os.system('ifconfig "' + self.bss_hostapd[1].ifname + '" down')
 | |
|         os.system('ifconfig "' + self.bss_hostapd[2].ifname + '" down')
 | |
|         os.system('ifconfig "' + self.bss_hostapd[0].ifname + '" up')
 | |
|         os.system('ifconfig "' + self.bss_hostapd[1].ifname + '" up')
 | |
|         os.system('ifconfig "' + self.bss_hostapd[2].ifname + '" up')
 | |
| 
 | |
|         hwsim = Hwsim()
 | |
|         wd = IWD()
 | |
|         device = wd.list_devices(1)[0]
 | |
|         try:
 | |
|             device.disconnect()
 | |
|         except:
 | |
|             pass
 | |
| 
 | |
|         condition = 'obj.state == DeviceState.disconnected'
 | |
|         wd.wait_for_object_condition(device, condition)
 | |
| 
 | |
|         for rule in list(hwsim.rules.keys()):
 | |
|             del hwsim.rules[rule]
 | |
| 
 | |
|     @classmethod
 | |
|     def setUpClass(cls):
 | |
|         hwsim = Hwsim()
 | |
| 
 | |
|         cls.bss_hostapd = [ HostapdCLI(config='ft-sae-1.conf'),
 | |
|                             HostapdCLI(config='ft-sae-2.conf'),
 | |
|                             HostapdCLI(config='ft-psk-3.conf') ]
 | |
|         cls.bss_radio =  [ hwsim.get_radio('rad0'),
 | |
|                            hwsim.get_radio('rad1'),
 | |
|                            hwsim.get_radio('rad2') ]
 | |
| 
 | |
|         ctx.start_process(['ifconfig', cls.bss_hostapd[0].ifname, 'down', 'hw', \
 | |
|                                 'ether', '12:00:00:00:00:01', 'up'], wait=True)
 | |
|         ctx.start_process(['ifconfig', cls.bss_hostapd[1].ifname, 'down', 'hw', \
 | |
|                                 'ether', '12:00:00:00:00:02', 'up'], wait=True)
 | |
|         ctx.start_process(['ifconfig', cls.bss_hostapd[2].ifname, 'down', 'hw', \
 | |
|                                 'ether', '12:00:00:00:00:03', 'up'], wait=True)
 | |
| 
 | |
|         # Set interface addresses to those expected by hostapd config files
 | |
|         cls.bss_hostapd[0].reload()
 | |
|         cls.bss_hostapd[0].wait_for_event("AP-ENABLED")
 | |
|         cls.bss_hostapd[1].reload()
 | |
|         cls.bss_hostapd[1].wait_for_event("AP-ENABLED")
 | |
|         cls.bss_hostapd[2].reload()
 | |
|         cls.bss_hostapd[2].wait_for_event("AP-ENABLED")
 | |
| 
 | |
|         # Fill in the neighbor AP tables in both BSSes.  By default each
 | |
|         # instance knows only about current BSS, even inside one hostapd
 | |
|         # process.
 | |
|         # FT still works without the neighbor AP table but neighbor reports
 | |
|         # have to be disabled in the .conf files
 | |
|         cls.bss_hostapd[0].set_neighbor('12:00:00:00:00:02', 'TestFT',
 | |
|                 '1200000000028f0000005102060603000000')
 | |
|         cls.bss_hostapd[0].set_neighbor('12:00:00:00:00:03', 'TestFT',
 | |
|                 '1200000000038f0000005102060603000000')
 | |
| 
 | |
|         cls.bss_hostapd[1].set_neighbor('12:00:00:00:00:01', 'TestFT',
 | |
|                 '1200000000018f0000005101060603000000')
 | |
|         cls.bss_hostapd[1].set_neighbor('12:00:00:00:00:03', 'TestFT',
 | |
|                 '1200000000038f0000005101060603000000')
 | |
| 
 | |
|         cls.bss_hostapd[2].set_neighbor('12:00:00:00:00:01', 'TestFT',
 | |
|                 '1200000000018f0000005101060603000000')
 | |
|         cls.bss_hostapd[2].set_neighbor('12:00:00:00:00:02', 'TestFT',
 | |
|                 '1200000000028f0000005101060603000000')
 | |
| 
 | |
|     @classmethod
 | |
|     def tearDownClass(cls):
 | |
|         IWD.clear_storage()
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     unittest.main(exit=True)
 |