[docs]deftest_uuids_conversions(self):str_uuid='a3df91c8-52a6-4afa-957b-3479a7d0897c'one_np_uuid=np.array([-411333541468446813,8973933150224022421])two_np_uuid=np.tile(one_np_uuid,[2,1])# array gives a listself.assertTrue(all(map(lambdax:x==str_uuid,np2str(two_np_uuid))))# single uuid gives a stringself.assertTrue(np2str(one_np_uuid)==str_uuid)# list uuids with some None entriesuuid_list=['bc74f49f33ec0f7545ebc03f0490bdf6','c5779e6d02ae6d1d6772df40a1a94243',None,'643371c81724378d34e04a60ef8769f4']assertnp.all(str2np(uuid_list)[2,:]==0)
[docs]deftest_uuids_intersections(self):ntotal=500nsub=17nadd=3eids=uuid2np([uuid.uuid4()for_inrange(ntotal)])np.random.seed(42)isel=np.floor(np.argsort(np.random.random(nsub))/nsub*ntotal).astype(np.int16)sids=np.r_[eids[isel,:],uuid2np([uuid.uuid4()for_inrange(nadd)])]np.random.shuffle(sids)# check the intersectionv,i0,i1=intersect2d(eids,sids)assertnp.all(eids[i0,:]==sids[i1,:])assertnp.all(np.sort(isel)==np.sort(i0))v_,i0_,i1_=np.intersect1d(eids[:,0],sids[:,0],return_indices=True)assertnp.setxor1d(v_,v[:,0]).size==0assertnp.setxor1d(i0,i0_).size==0assertnp.setxor1d(i1,i1_).size==0fora,binzip(ismember2d(sids,eids),ismember(sids[:,0],eids[:,0])):assertnp.all(a==b)# check conversion to numpy back and forthuuids=[uuid.uuid4()for_innp.arange(4)]np_uuids=uuid2np(uuids)assertnp2uuid(np_uuids)==uuids
[docs]@mock.patch('iblutil.io.params.time.sleep')deftest_file_lock_sync(self,sleep_mock):"""Test synchronous FileLock context manager."""# Check input validationself.assertRaises(ValueError,params.FileLock,self.file,timeout_action='foo')# Check behaviour when lock file doesn't exist (i.e. no other process writing to file)assertnotself.lock_file.exists()withparams.FileLock(self.file,timeout_action='raise'):self.assertTrue(self.lock_file.exists(),'Failed to create lock file')self.assertFalse(self.lock_file.exists(),'Failed to remove lock file upon exit of context manager')sleep_mock.assert_not_called()# no file present so no need to sleep# Check behaviour when lock file present and not removed by other processself.lock_file.touch()assertself.lock_file.exists()lock=params.FileLock(self.file,timeout_action='raise')withself.assertLogs('iblutil.io.params',10)aslg:self.assertRaises(TimeoutError,lock.__enter__)msg=next((x.getMessage()forxinlg.recordsifx.levelno==10),None)self.assertEqual('file lock contents: <empty>',msg)# should try 5 attempts by default; default total timeout is 10 seconds so should sleep 5x for 2 seconds eachexpected_attempts=5sleep_mock.assert_called_with(2)self.assertEqual(expected_attempts,sleep_mock.call_count)self.assertEqual(expected_attempts,len([xforxinlg.recordsifx.levelno==20]))msg=next(x.getMessage()forxinlg.recordsifx.levelno==20)self.assertRegex(msg,'file lock found, waiting 2.00 seconds')# Check delete timeout actionassertself.lock_file.exists()withself.assertLogs('iblutil.io.params',10)aslg, \
params.FileLock(self.file,timeout_action='delete'):# Should have replaced empty lock file with timestamped oneself.assertTrue(self.lock_file.exists())withopen(self.lock_file,'r')asfp:lock_info=json.load(fp)self.assertCountEqual(('datetime','hostname'),lock_info)self.assertFalse(self.lock_file.exists(),'Failed to remove lock file upon exit of context manager')self.assertRegex(lg.records[-1].getMessage(),'stale file lock found, deleting')
asyncdef_mock(self,obj):""" Add side effect to mock object that awaits a future. This is required because async lambdas are not supported. Parameters ---------- obj : unittest.mock.AsyncMock An asynchronous mock object to install side effect for. Returns ------- asyncio.Future A future awaited by input mock object. """fut=asyncio.get_event_loop().create_future()self.addCleanup(fut.cancel)asyncdefwait(_):returnawaitfutobj.side_effect=waitreturnfut
[docs]@mock.patch('iblutil.io.params.asyncio.sleep')asyncdeftest_file_lock_async(self,sleep_mock):"""Test asynchronous FileLock context manager."""# Check behaviour when lock file doesn't exist (i.e. no other process writing to file)assertnotself.lock_file.exists()asyncwithparams.FileLock(self.file,timeout_action='raise'):self.assertTrue(self.lock_file.exists(),'Failed to create lock file')self.assertFalse(self.lock_file.exists(),'Failed to remove lock file upon exit of context manager')sleep_mock.assert_not_called()# no file present so no need to sleep# Check behaviour when lock file present and not removed by other processself.lock_file.touch()assertself.lock_file.exists()lock=params.FileLock(self.file,timeout=1e-3,timeout_action='raise')# The loop that checks the lock file is too fast when async.sleep is mocked so adding a side# effect that awaits a future that's never set allows the timeout code to execute.awaitself._mock(sleep_mock)withself.assertLogs('iblutil.io.params',10)aslg,self.assertRaises(asyncio.TimeoutError):awaitlock.__aenter__()# fut = asyncio.get_running_loop().create_future()# with mock.patch.object(lock, '_lock_check_async', return_value) as m:# async with params.FileLock(self.file, timeout=1e-3, timeout_action='raise') as lock:# ...sleep_mock.assert_awaited_with(lock._async_poll_freq)msg=next((x.getMessage()forxinlg.recordsifx.levelno==10),None)self.assertEqual('file lock contents: <empty>',msg)# Check remove timeout actionassertself.lock_file.exists()awaitself._mock(sleep_mock)withself.assertLogs('iblutil.io.params',10)aslg:asyncwithparams.FileLock(self.file,timeout=1e-5,timeout_action='delete'):# Should have replaced empty lock file with timestamped oneself.assertTrue(self.lock_file.exists())withopen(self.lock_file,'r')asfp:lock_info=json.load(fp)self.assertCountEqual(('datetime','hostname','pid'),lock_info)self.assertFalse(self.lock_file.exists(),'Failed to remove lock file upon exit of context manager')