| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  | import unittest | 
					
						
							| 
									
										
										
										
											2021-04-15 04:01:25 +02:00
										 |  |  | import pathlib | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import Utils | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | file_path = pathlib.Path(__file__).parent.parent | 
					
						
							|  |  |  | Utils.local_path.cached_path = file_path | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  | from BaseClasses import MultiWorld, CollectionState, ItemClassification | 
					
						
							| 
									
										
										
										
											2020-10-24 05:38:56 +02:00
										 |  |  | from worlds.alttp.Items import ItemFactory | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-06 06:09:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  | class TestBase(unittest.TestCase): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |     multiworld: MultiWorld | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  |     _state_cache = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_state(self, items): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         if (self.multiworld, tuple(items)) in self._state_cache: | 
					
						
							|  |  |  |             return self._state_cache[self.multiworld, tuple(items)] | 
					
						
							|  |  |  |         state = CollectionState(self.multiworld) | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  |         for item in items: | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  |             item.classification = ItemClassification.progression | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  |             state.collect(item) | 
					
						
							|  |  |  |         state.sweep_for_events() | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         self._state_cache[self.multiworld, tuple(items)] = state | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  |         return state | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |     def get_path(self, state, region): | 
					
						
							|  |  |  |         def flist_to_iter(node): | 
					
						
							|  |  |  |             while node: | 
					
						
							|  |  |  |                 value, node = node | 
					
						
							|  |  |  |                 yield value | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         from itertools import zip_longest | 
					
						
							|  |  |  |         reversed_path_as_flist = state.path.get(region, (region, None)) | 
					
						
							|  |  |  |         string_path_flat = reversed(list(map(str, flist_to_iter(reversed_path_as_flist)))) | 
					
						
							|  |  |  |         # Now we combine the flat string list into (region, exit) pairs | 
					
						
							|  |  |  |         pathsiter = iter(string_path_flat) | 
					
						
							|  |  |  |         pathpairs = zip_longest(pathsiter, pathsiter) | 
					
						
							|  |  |  |         return list(pathpairs) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  |     def run_location_tests(self, access_pool): | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |         for i, (location, access, *item_pool) in enumerate(access_pool): | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  |             items = item_pool[0] | 
					
						
							|  |  |  |             all_except = item_pool[1] if len(item_pool) > 1 else None | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |             state = self._get_items(item_pool, all_except) | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             path = self.get_path(state, self.multiworld.get_location(location, 1).parent_region) | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |             with self.subTest(msg="Reach Location", location=location, access=access, items=items, | 
					
						
							|  |  |  |                               all_except=all_except, path=path, entry=i): | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 self.assertEqual(self.multiworld.get_location(location, 1).can_reach(state), access) | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |             # check for partial solution | 
					
						
							|  |  |  |             if not all_except and access:  # we are not supposed to be able to reach location with partial inventory | 
					
						
							| 
									
										
										
										
											2020-12-19 21:13:35 +01:00
										 |  |  |                 for missing_item in item_pool[0]: | 
					
						
							|  |  |  |                     with self.subTest(msg="Location reachable without required item", location=location, | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |                                       items=item_pool[0], missing_item=missing_item, entry=i): | 
					
						
							|  |  |  |                         state = self._get_items_partial(item_pool, missing_item) | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                         self.assertEqual(self.multiworld.get_location(location, 1).can_reach(state), False) | 
					
						
							| 
									
										
										
										
											2020-12-05 14:59:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  |     def run_entrance_tests(self, access_pool): | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |         for i, (entrance, access, *item_pool) in enumerate(access_pool): | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  |             items = item_pool[0] | 
					
						
							|  |  |  |             all_except = item_pool[1] if len(item_pool) > 1 else None | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |             state = self._get_items(item_pool, all_except) | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             path = self.get_path(state, self.multiworld.get_entrance(entrance, 1).parent_region) | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |             with self.subTest(msg="Reach Entrance", entrance=entrance, access=access, items=items, | 
					
						
							|  |  |  |                               all_except=all_except, path=path, entry=i): | 
					
						
							| 
									
										
										
										
											2020-04-29 01:34:30 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 self.assertEqual(self.multiworld.get_entrance(entrance, 1).can_reach(state), access) | 
					
						
							| 
									
										
										
										
											2020-12-05 14:59:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |             # check for partial solution | 
					
						
							|  |  |  |             if not all_except and access:  # we are not supposed to be able to reach location with partial inventory | 
					
						
							| 
									
										
										
										
											2020-12-19 21:13:35 +01:00
										 |  |  |                 for missing_item in item_pool[0]: | 
					
						
							|  |  |  |                     with self.subTest(msg="Entrance reachable without required item", entrance=entrance, | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |                                       items=item_pool[0], missing_item=missing_item, entry=i): | 
					
						
							|  |  |  |                         state = self._get_items_partial(item_pool, missing_item) | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                         self.assertEqual(self.multiworld.get_entrance(entrance, 1).can_reach(state), False) | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _get_items(self, item_pool, all_except): | 
					
						
							|  |  |  |         if all_except and len(all_except) > 0: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             items = self.multiworld.itempool[:] | 
					
						
							| 
									
										
										
										
											2021-02-02 11:24:17 +01:00
										 |  |  |             items = [item for item in items if | 
					
						
							|  |  |  |                      item.name not in all_except and not ("Bottle" in item.name and "AnyBottle" in all_except)] | 
					
						
							|  |  |  |             items.extend(ItemFactory(item_pool[0], 1)) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             items = ItemFactory(item_pool[0], 1) | 
					
						
							|  |  |  |         return self.get_state(items) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _get_items_partial(self, item_pool, missing_item): | 
					
						
							|  |  |  |         new_items = item_pool[0].copy() | 
					
						
							|  |  |  |         new_items.remove(missing_item) | 
					
						
							|  |  |  |         items = ItemFactory(new_items, 1) | 
					
						
							|  |  |  |         return self.get_state(items) |