forked from OoTRandomizer/OoT-Randomizer
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Playthrough.py
283 lines (233 loc) · 12.3 KB
/
Playthrough.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
import copy
from collections import defaultdict
import itertools
class Playthrough(object):
def __init__(self, state_list, initial_cache=None):
self.state_list = [state.copy() for state in state_list]
# Let the states reference this playthrough.
for state in self.state_list:
state.playthrough = self
if initial_cache:
self._cache = initial_cache
self.cached_spheres = [self._cache]
else:
root_regions = [state.world.get_region('Root') for state in self.state_list]
# The cache is a dict with 5 values:
# child_regions, adult_regions: sets of Region, all the regions in that sphere
# child_queue, adult_queue: queue of Entrance, all the exits to try next sphere
# visited_locations: set of Locations visited in or before that sphere.
self._cache = {
'child_queue': list(exit for region in root_regions for exit in region.exits),
'adult_queue': list(exit for region in root_regions for exit in region.exits),
'visited_locations': set(),
'child_regions': set(root_regions),
'adult_regions': set(root_regions),
}
self.cached_spheres = [self._cache]
self.next_sphere()
def copy(self):
# we only need to copy the top sphere since that's what we're starting with and we don't go back
new_cache = {k: copy.copy(v) for k,v in self._cache.items()}
# copy always makes a nonreversible instance
return Playthrough(self.state_list, initial_cache=new_cache)
def collect_all(self, itempool):
for item in itempool:
self.state_list[item.world.id].collect(item)
def collect(self, item):
self.state_list[item.world.id].collect(item)
@classmethod
def max_explore(cls, state_list, itempool=None):
p = cls(state_list)
if itempool:
p.collect_all(itempool)
p.collect_locations()
return p
@classmethod
def with_items(cls, state_list, itempool=None):
p = cls(state_list)
if itempool:
p.collect_all(itempool)
p.next_sphere()
return p
# Truncates the sphere cache based on which sphere a location is in, and
# drops the location from the appropriate visited set.
# Doesn't forget which sphere locations are in as an optimization, so be careful
# to only unvisit locations in descending sphere order, or locations that
# have been revisited in the most recent iteration.
# Locations never visited in this Playthrough are assumed to have been visited
# in sphere 0, so unvisiting them will discard the entire cache.
# Not safe to call during iteration.
def unvisit(self, location):
raise Exception('Unimplemented for Playthrough. Perhaps you want RewindablePlaythrough.')
# Drops the item from its respective state.
# Has no effect on cache!
def uncollect(self, item):
self.state_list[item.world.id].remove(item)
# Resets the sphere cache to the first entry only.
# Does not uncollect any items!
# Not safe to call during iteration.
def reset(self):
raise Exception('Unimplemented for Playthrough. Perhaps you want RewindablePlaythrough.')
# simplified exit.can_reach(state), bypasses can_become_age
# which we've already accounted for
def validate_child(self, exit):
return exit.can_reach_simple(self.state_list[exit.parent_region.world.id], age='child')
def validate_adult(self, exit):
return exit.can_reach_simple(self.state_list[exit.parent_region.world.id], age='adult')
# Internal to the iteration. Modifies the exit_queue, region_set.
# Returns a queue of the exits whose access rule failed,
# as a cache for the exits to try on the next iteration.
@staticmethod
def _expand_regions(exit_queue, region_set, validate):
failed = []
for exit in exit_queue:
if exit.connected_region and exit.connected_region not in region_set:
if validate(exit):
region_set.add(exit.connected_region)
exit_queue.extend(exit.connected_region.exits)
else:
failed.append(exit)
return failed
# Explores available exits, updating relevant entries in the cache in-place.
# Returns the set of regions accessible in the new sphere as child,
# the set of regions accessible as adult, and the set of visited locations.
# These are references to the new entry in the cache, so they can be modified
# directly.
def next_sphere(self):
# Use the queue to iteratively add regions to the accessed set,
# until we are stuck or out of regions.
self._cache.update({
# Replace the queues (which have been modified) with just the
# failed exits that we can retry next time.
'adult_queue': Playthrough._expand_regions(
self._cache['adult_queue'], self._cache['adult_regions'], self.validate_adult),
'child_queue': Playthrough._expand_regions(
self._cache['child_queue'], self._cache['child_regions'], self.validate_child),
})
return self._cache['child_regions'], self._cache['adult_regions'], self._cache['visited_locations']
# Yields every reachable location, by iteratively deepening explored sets of
# regions (one as child, one as adult) and invoking access rules.
# item_locations is a list of Location objects from state_list that the caller
# has prefiltered (eg. by whether they contain advancement items).
#
# Inside the loop, the caller usually wants to collect items at these
# locations to see if the game is beatable. Collection should be done
# using internal State (recommended to just call playthrough.collect).
def iter_reachable_locations(self, item_locations):
# tests reachability, skipping recursive can_reach region check
def accessible(loc):
return (loc not in visited_locations
# Check adult first; it's the most likely.
and (loc.parent_region in adult_regions
and loc.can_reach_simple(self.state_list[loc.world.id], age='adult')
or (loc.parent_region in child_regions
and loc.can_reach_simple(self.state_list[loc.world.id], age='child'))))
had_reachable_locations = True
# will loop as long as any visits were made, and at least once
while had_reachable_locations:
child_regions, adult_regions, visited_locations = self.next_sphere()
# Get all locations in accessible_regions that aren't visited,
# and check if they can be reached. Collect them.
reachable_locations = filter(accessible, item_locations)
had_reachable_locations = False
for location in reachable_locations:
had_reachable_locations = True
# Mark it visited for this algorithm
visited_locations.add(location)
yield location
# This collects all item locations available in the state list given that
# the states have collected items. The purpose is that it will search for
# all new items that become accessible with a new item set.
def collect_locations(self, item_locations=None):
item_locations = item_locations or self.progression_locations()
for location in self.iter_reachable_locations(item_locations):
# Collect the item for the state world it is for
self.collect(location.item)
# A shorthand way to iterate over locations without collecting items.
def visit_locations(self, locations=None):
locations = locations or self.progression_locations()
for _ in self.iter_reachable_locations(locations):
pass
# Retrieve all item locations in the worlds that have progression items
def progression_locations(self):
return [location for state in self.state_list for location in state.world.get_filled_locations() if location.item.advancement]
# This returns True if every state is beatable. It's important to ensure
# all states beatable since items required in one world can be in another.
# A state is beatable if it can ever collect the Triforce.
# If scan_for_items is True, constructs and modifies a copy of the underlying
# state to determine beatability; otherwise, only checks that the playthrough
# has already acquired all the Triforces.
def can_beat_game(self, scan_for_items=True):
def won(state):
return state.has('Triforce')
# Check if already beaten
if all(map(won, self.state_list)):
return True
if scan_for_items:
# collect all available items
# make a new playthrough since we might be iterating over one already
playthrough = self.copy()
playthrough.collect_locations()
# if every state got the Triforce, then return True
return all(map(won, playthrough.state_list))
else:
return False
# Use the cache in the playthrough to determine region reachability.
def can_reach(self, region, age=None):
if age == 'adult':
return region in self._cache['adult_regions']
elif age == 'child':
return region in self._cache['child_regions']
elif age == 'both':
return region in self._cache['adult_regions'] and region in self._cache['child_regions']
else:
# treat None as either
return region in self._cache['adult_regions'] or region in self._cache['child_regions']
# Use the cache in the playthrough to determine location reachability.
# Only works for locations that had progression items...
def visited(self, location):
return location in self._cache['visited_locations']
# Use the cache in the playthrough to get all reachable regions.
def reachable_regions(self, age=None):
if age == 'adult':
return self._cache['adult_regions']
elif age == 'child':
return self._cache['child_regions']
else:
return self._cache['adult_regions'] + self._cache['child_regions']
# Returns whether the given age can access the spot at this age,
# by checking whether the playthrough has reached the containing region, and evaluating the spot's access rule.
def spot_access(self, spot, age):
if age == 'both':
return (self.can_reach(spot.parent_region, age=age)
and spot.can_reach_simple(self.state_list[spot.world.id], age='adult')
and spot.can_reach_simple(self.state_list[spot.world.id], age='child'))
elif age == 'either':
return (self.can_reach(spot.parent_region, age='adult')
and spot.can_reach_simple(self.state_list[spot.world.id], age='adult')) or (
self.can_reach(spot.parent_region, age='child')
and spot.can_reach_simple(self.state_list[spot.world.id], age='child'))
else:
return (self.can_reach(spot.parent_region, age=age)
and spot.can_reach_simple(self.state_list[spot.world.id], age=age))
class RewindablePlaythrough(Playthrough):
def unvisit(self, location):
# A location being unvisited is either:
# in the top two caches (if it's the first being unvisited for a sphere)
# in the topmost cache only (otherwise)
# After we unvisit every location in a sphere, the top two caches have identical visited locations.
assert location in self.cached_spheres[-1]['visited_locations']
if location in self.cached_spheres[-2]['visited_locations']:
self.cached_spheres.pop()
self._cache = self.cached_spheres[-1]
self._cache['visited_locations'].discard(location)
def reset(self):
self._cache = self.cached_spheres[0]
self.cached_spheres[1:] = []
# Adds a new layer to the sphere cache, as a copy of the previous.
def checkpoint(self):
# Save the current data into the cache.
self.cached_spheres.append({
k: copy.copy(v) for k, v in self._cache.items()
})
self._cache = self.cached_spheres[-1]