summary refs log tree commit diff stats
diff options
context:
space:
mode:
-rw-r--r--__init__.py8
-rw-r--r--data/README.md5
-rw-r--r--data/generated.datbin0 -> 129731 bytes
-rw-r--r--datatypes.py67
-rw-r--r--items.py12
-rw-r--r--locations.py11
-rw-r--r--player_logic.py11
-rw-r--r--regions.py3
-rw-r--r--rules.py3
-rw-r--r--static_logic.py530
-rw-r--r--test/TestDatafile.py16
-rw-r--r--utils/__init__.py0
-rw-r--r--utils/pickle_static_data.py475
13 files changed, 626 insertions, 515 deletions
diff --git a/__init__.py b/__init__.py index 2f93541..e35a102 100644 --- a/__init__.py +++ b/__init__.py
@@ -5,12 +5,12 @@ from logging import warning
5 5
6from BaseClasses import Item, ItemClassification, Tutorial 6from BaseClasses import Item, ItemClassification, Tutorial
7from worlds.AutoWorld import WebWorld, World 7from worlds.AutoWorld import WebWorld, World
8from .datatypes import Room, RoomEntrance
8from .items import ALL_ITEM_TABLE, LingoItem 9from .items import ALL_ITEM_TABLE, LingoItem
9from .locations import ALL_LOCATION_TABLE 10from .locations import ALL_LOCATION_TABLE
10from .options import LingoOptions 11from .options import LingoOptions
11from .player_logic import LingoPlayerLogic 12from .player_logic import LingoPlayerLogic
12from .regions import create_regions 13from .regions import create_regions
13from .static_logic import Room, RoomEntrance
14 14
15 15
16class LingoWebWorld(WebWorld): 16class LingoWebWorld(WebWorld):
@@ -100,9 +100,9 @@ class LingoWorld(World):
100 item = ALL_ITEM_TABLE[name] 100 item = ALL_ITEM_TABLE[name]
101 101
102 classification = item.classification 102 classification = item.classification
103 if hasattr(self, "options") and self.options.shuffle_paintings and len(item.painting_ids) > 0\ 103 if hasattr(self, "options") and self.options.shuffle_paintings and len(item.painting_ids) > 0 \
104 and len(item.door_ids) == 0 and all(painting_id not in self.player_logic.painting_mapping 104 and not item.has_doors and all(painting_id not in self.player_logic.painting_mapping
105 for painting_id in item.painting_ids)\ 105 for painting_id in item.painting_ids) \
106 and "pilgrim_painting2" not in item.painting_ids: 106 and "pilgrim_painting2" not in item.painting_ids:
107 # If this is a "door" that just moves one or more paintings, and painting shuffle is on and those paintings 107 # If this is a "door" that just moves one or more paintings, and painting shuffle is on and those paintings
108 # go nowhere, then this item should not be progression. The Pilgrim Room painting is special and needs to be 108 # go nowhere, then this item should not be progression. The Pilgrim Room painting is special and needs to be
diff --git a/data/README.md b/data/README.md new file mode 100644 index 0000000..fe834ce --- /dev/null +++ b/data/README.md
@@ -0,0 +1,5 @@
1# lingo data
2
3The source of truth for the Lingo randomizer is (currently) the LL1.yaml and ids.yaml files located here. These files are used by the generator, the game client, and the tracker, in order to have logic that is consistent across them all.
4
5The generator does not actually read in the yaml files. Instead, a compiled datafile called generated.dat is also located in this directory. If you update LL1.yaml and/or ids.yaml, you must also regenerate the datafile using `python worlds/lingo/utils/pickle_static_data.py`. A unit test will fail if you don't.
diff --git a/data/generated.dat b/data/generated.dat new file mode 100644 index 0000000..49ea60d --- /dev/null +++ b/data/generated.dat
Binary files differ
diff --git a/datatypes.py b/datatypes.py new file mode 100644 index 0000000..eb5c879 --- /dev/null +++ b/datatypes.py
@@ -0,0 +1,67 @@
1from typing import List, NamedTuple, Optional
2
3
4class RoomAndDoor(NamedTuple):
5 room: Optional[str]
6 door: str
7
8
9class RoomAndPanel(NamedTuple):
10 room: Optional[str]
11 panel: str
12
13
14class RoomEntrance(NamedTuple):
15 room: str # source room
16 door: Optional[RoomAndDoor]
17 painting: bool
18
19
20class Room(NamedTuple):
21 name: str
22 entrances: List[RoomEntrance]
23
24
25class Door(NamedTuple):
26 name: str
27 item_name: str
28 location_name: Optional[str]
29 panels: Optional[List[RoomAndPanel]]
30 skip_location: bool
31 skip_item: bool
32 has_doors: bool
33 painting_ids: List[str]
34 event: bool
35 group: Optional[str]
36 include_reduce: bool
37 junk_item: bool
38
39
40class Panel(NamedTuple):
41 required_rooms: List[str]
42 required_doors: List[RoomAndDoor]
43 required_panels: List[RoomAndPanel]
44 colors: List[str]
45 check: bool
46 event: bool
47 exclude_reduce: bool
48 achievement: bool
49 non_counting: bool
50
51
52class Painting(NamedTuple):
53 id: str
54 room: str
55 enter_only: bool
56 exit_only: bool
57 required: bool
58 required_when_no_doors: bool
59 required_door: Optional[RoomAndDoor]
60 disable: bool
61 req_blocked: bool
62 req_blocked_when_no_doors: bool
63
64
65class Progression(NamedTuple):
66 item_name: str
67 index: int
diff --git a/items.py b/items.py index 9f8bf56..623cd79 100644 --- a/items.py +++ b/items.py
@@ -16,7 +16,7 @@ class ItemData(NamedTuple):
16 code: int 16 code: int
17 classification: ItemClassification 17 classification: ItemClassification
18 mode: Optional[str] 18 mode: Optional[str]
19 door_ids: List[str] 19 has_doors: bool
20 painting_ids: List[str] 20 painting_ids: List[str]
21 21
22 def should_include(self, world: "LingoWorld") -> bool: 22 def should_include(self, world: "LingoWorld") -> bool:
@@ -61,7 +61,7 @@ def load_item_data():
61 door_mode = "doors" 61 door_mode = "doors"
62 else: 62 else:
63 door_mode = "complex door" 63 door_mode = "complex door"
64 door_groups.setdefault(door.group, []).extend(door.door_ids) 64 door_groups.setdefault(door.group, [])
65 65
66 if room_name in PROGRESSION_BY_ROOM and door_name in PROGRESSION_BY_ROOM[room_name]: 66 if room_name in PROGRESSION_BY_ROOM and door_name in PROGRESSION_BY_ROOM[room_name]:
67 door_mode = "special" 67 door_mode = "special"
@@ -69,11 +69,11 @@ def load_item_data():
69 ALL_ITEM_TABLE[door.item_name] = \ 69 ALL_ITEM_TABLE[door.item_name] = \
70 ItemData(get_door_item_id(room_name, door_name), 70 ItemData(get_door_item_id(room_name, door_name),
71 ItemClassification.filler if door.junk_item else ItemClassification.progression, door_mode, 71 ItemClassification.filler if door.junk_item else ItemClassification.progression, door_mode,
72 door.door_ids, door.painting_ids) 72 door.has_doors, door.painting_ids)
73 73
74 for group, group_door_ids in door_groups.items(): 74 for group, group_door_ids in door_groups.items():
75 ALL_ITEM_TABLE[group] = ItemData(get_door_group_item_id(group), 75 ALL_ITEM_TABLE[group] = ItemData(get_door_group_item_id(group),
76 ItemClassification.progression, "door group", group_door_ids, []) 76 ItemClassification.progression, "door group", True, [])
77 77
78 special_items: Dict[str, ItemClassification] = { 78 special_items: Dict[str, ItemClassification] = {
79 ":)": ItemClassification.filler, 79 ":)": ItemClassification.filler,
@@ -88,11 +88,11 @@ def load_item_data():
88 88
89 for item_name, classification in special_items.items(): 89 for item_name, classification in special_items.items():
90 ALL_ITEM_TABLE[item_name] = ItemData(get_special_item_id(item_name), classification, 90 ALL_ITEM_TABLE[item_name] = ItemData(get_special_item_id(item_name), classification,
91 "special", [], []) 91 "special", False, [])
92 92
93 for item_name in PROGRESSIVE_ITEMS: 93 for item_name in PROGRESSIVE_ITEMS:
94 ALL_ITEM_TABLE[item_name] = ItemData(get_progressive_item_id(item_name), 94 ALL_ITEM_TABLE[item_name] = ItemData(get_progressive_item_id(item_name),
95 ItemClassification.progression, "special", [], []) 95 ItemClassification.progression, "special", False, [])
96 96
97 97
98# Initialize the item data at module scope. 98# Initialize the item data at module scope.
diff --git a/locations.py b/locations.py index 5903d60..e66ebac 100644 --- a/locations.py +++ b/locations.py
@@ -2,7 +2,8 @@ from enum import Flag, auto
2from typing import Dict, List, NamedTuple 2from typing import Dict, List, NamedTuple
3 3
4from BaseClasses import Location 4from BaseClasses import Location
5from .static_logic import DOORS_BY_ROOM, PANELS_BY_ROOM, RoomAndPanel, get_door_location_id, get_panel_location_id 5from .datatypes import RoomAndPanel
6from .static_logic import DOORS_BY_ROOM, PANELS_BY_ROOM, get_door_location_id, get_panel_location_id
6 7
7 8
8class LocationClassification(Flag): 9class LocationClassification(Flag):
@@ -20,14 +21,6 @@ class LocationData(NamedTuple):
20 panels: List[RoomAndPanel] 21 panels: List[RoomAndPanel]
21 classification: LocationClassification 22 classification: LocationClassification
22 23
23 def panel_ids(self):
24 ids = set()
25 for panel in self.panels:
26 effective_room = self.room if panel.room is None else panel.room
27 panel_data = PANELS_BY_ROOM[effective_room][panel.panel]
28 ids = ids | set(panel_data.internal_ids)
29 return ids
30
31 24
32class LingoLocation(Location): 25class LingoLocation(Location):
33 """ 26 """
diff --git a/player_logic.py b/player_logic.py index 3a6eedf..f5eb986 100644 --- a/player_logic.py +++ b/player_logic.py
@@ -1,12 +1,12 @@
1from enum import Enum 1from enum import Enum
2from typing import Dict, List, NamedTuple, Optional, Set, Tuple, TYPE_CHECKING 2from typing import Dict, List, NamedTuple, Optional, Set, Tuple, TYPE_CHECKING
3 3
4from .datatypes import Door, RoomAndDoor, RoomAndPanel
4from .items import ALL_ITEM_TABLE 5from .items import ALL_ITEM_TABLE
5from .locations import ALL_LOCATION_TABLE, LocationClassification 6from .locations import ALL_LOCATION_TABLE, LocationClassification
6from .options import LocationChecks, ShuffleDoors, VictoryCondition 7from .options import LocationChecks, ShuffleDoors, VictoryCondition
7from .static_logic import DOORS_BY_ROOM, Door, PAINTINGS, PAINTINGS_BY_ROOM, PAINTING_ENTRANCES, PAINTING_EXITS, \ 8from .static_logic import DOORS_BY_ROOM, PAINTINGS, PAINTING_ENTRANCES, PAINTING_EXITS, \
8 PANELS_BY_ROOM, PROGRESSION_BY_ROOM, REQUIRED_PAINTING_ROOMS, REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS, RoomAndDoor, \ 9 PANELS_BY_ROOM, PROGRESSION_BY_ROOM, REQUIRED_PAINTING_ROOMS, REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS
9 RoomAndPanel
10 10
11if TYPE_CHECKING: 11if TYPE_CHECKING:
12 from . import LingoWorld 12 from . import LingoWorld
@@ -279,8 +279,9 @@ class LingoPlayerLogic:
279 279
280 # When painting shuffle is off, most Starting Room paintings give color hallways access. The Wondrous's 280 # When painting shuffle is off, most Starting Room paintings give color hallways access. The Wondrous's
281 # painting does not, but it gives access to SHRINK and WELCOME BACK. 281 # painting does not, but it gives access to SHRINK and WELCOME BACK.
282 for painting_obj in PAINTINGS_BY_ROOM["Starting Room"]: 282 for painting_obj in PAINTINGS.values():
283 if not painting_obj.enter_only or painting_obj.required_door is None: 283 if not painting_obj.enter_only or painting_obj.required_door is None\
284 or painting_obj.room != "Starting Room":
284 continue 285 continue
285 286
286 # If painting shuffle is on, we only want to consider paintings that actually go somewhere. 287 # If painting shuffle is on, we only want to consider paintings that actually go somewhere.
diff --git a/regions.py b/regions.py index bdc42f4..464e9a1 100644 --- a/regions.py +++ b/regions.py
@@ -1,11 +1,12 @@
1from typing import Dict, Optional, TYPE_CHECKING 1from typing import Dict, Optional, TYPE_CHECKING
2 2
3from BaseClasses import Entrance, ItemClassification, Region 3from BaseClasses import Entrance, ItemClassification, Region
4from .datatypes import Room, RoomAndDoor
4from .items import LingoItem 5from .items import LingoItem
5from .locations import LingoLocation 6from .locations import LingoLocation
6from .player_logic import LingoPlayerLogic 7from .player_logic import LingoPlayerLogic
7from .rules import lingo_can_use_entrance, make_location_lambda 8from .rules import lingo_can_use_entrance, make_location_lambda
8from .static_logic import ALL_ROOMS, PAINTINGS, Room, RoomAndDoor 9from .static_logic import ALL_ROOMS, PAINTINGS
9 10
10if TYPE_CHECKING: 11if TYPE_CHECKING:
11 from . import LingoWorld 12 from . import LingoWorld
diff --git a/rules.py b/rules.py index 481fab1..054c330 100644 --- a/rules.py +++ b/rules.py
@@ -1,8 +1,9 @@
1from typing import TYPE_CHECKING 1from typing import TYPE_CHECKING
2 2
3from BaseClasses import CollectionState 3from BaseClasses import CollectionState
4from .datatypes import RoomAndDoor
4from .player_logic import AccessRequirements, LingoPlayerLogic, PlayerLocation 5from .player_logic import AccessRequirements, LingoPlayerLogic, PlayerLocation
5from .static_logic import PROGRESSION_BY_ROOM, PROGRESSIVE_ITEMS, RoomAndDoor 6from .static_logic import PROGRESSION_BY_ROOM, PROGRESSIVE_ITEMS
6 7
7if TYPE_CHECKING: 8if TYPE_CHECKING:
8 from . import LingoWorld 9 from . import LingoWorld
diff --git a/static_logic.py b/static_logic.py index e9f82fb..1da265d 100644 --- a/static_logic.py +++ b/static_logic.py
@@ -1,86 +1,16 @@
1from typing import Dict, List, NamedTuple, Optional, Set 1import os
2import pkgutil
3from io import BytesIO
4from typing import Dict, List, Set
2 5
3import Utils 6import pickle
4 7
5 8from .datatypes import Door, Painting, Panel, Progression, Room
6class RoomAndDoor(NamedTuple):
7 room: Optional[str]
8 door: str
9
10
11class RoomAndPanel(NamedTuple):
12 room: Optional[str]
13 panel: str
14
15
16class RoomEntrance(NamedTuple):
17 room: str # source room
18 door: Optional[RoomAndDoor]
19 painting: bool
20
21
22class Room(NamedTuple):
23 name: str
24 entrances: List[RoomEntrance]
25
26
27class Door(NamedTuple):
28 name: str
29 item_name: str
30 location_name: Optional[str]
31 panels: Optional[List[RoomAndPanel]]
32 skip_location: bool
33 skip_item: bool
34 door_ids: List[str]
35 painting_ids: List[str]
36 event: bool
37 group: Optional[str]
38 include_reduce: bool
39 junk_item: bool
40
41
42class Panel(NamedTuple):
43 required_rooms: List[str]
44 required_doors: List[RoomAndDoor]
45 required_panels: List[RoomAndPanel]
46 colors: List[str]
47 check: bool
48 event: bool
49 internal_ids: List[str]
50 exclude_reduce: bool
51 achievement: bool
52 non_counting: bool
53
54
55class Painting(NamedTuple):
56 id: str
57 room: str
58 enter_only: bool
59 exit_only: bool
60 orientation: str
61 required: bool
62 required_when_no_doors: bool
63 required_door: Optional[RoomAndDoor]
64 disable: bool
65 move: bool
66 req_blocked: bool
67 req_blocked_when_no_doors: bool
68
69
70class Progression(NamedTuple):
71 item_name: str
72 index: int
73
74
75ROOMS: Dict[str, Room] = {}
76PANELS: Dict[str, Panel] = {}
77DOORS: Dict[str, Door] = {}
78PAINTINGS: Dict[str, Painting] = {}
79 9
80ALL_ROOMS: List[Room] = [] 10ALL_ROOMS: List[Room] = []
81DOORS_BY_ROOM: Dict[str, Dict[str, Door]] = {} 11DOORS_BY_ROOM: Dict[str, Dict[str, Door]] = {}
82PANELS_BY_ROOM: Dict[str, Dict[str, Panel]] = {} 12PANELS_BY_ROOM: Dict[str, Dict[str, Panel]] = {}
83PAINTINGS_BY_ROOM: Dict[str, List[Painting]] = {} 13PAINTINGS: Dict[str, Painting] = {}
84 14
85PROGRESSIVE_ITEMS: List[str] = [] 15PROGRESSIVE_ITEMS: List[str] = []
86PROGRESSION_BY_ROOM: Dict[str, Dict[str, Progression]] = {} 16PROGRESSION_BY_ROOM: Dict[str, Dict[str, Progression]] = {}
@@ -98,61 +28,7 @@ DOOR_ITEM_IDS: Dict[str, Dict[str, int]] = {}
98DOOR_GROUP_ITEM_IDS: Dict[str, int] = {} 28DOOR_GROUP_ITEM_IDS: Dict[str, int] = {}
99PROGRESSIVE_ITEM_IDS: Dict[str, int] = {} 29PROGRESSIVE_ITEM_IDS: Dict[str, int] = {}
100 30
101 31HASHES: Dict[str, str] = {}
102def load_static_data():
103 global PAINTING_EXITS, SPECIAL_ITEM_IDS, PANEL_LOCATION_IDS, DOOR_LOCATION_IDS, DOOR_ITEM_IDS, \
104 DOOR_GROUP_ITEM_IDS, PROGRESSIVE_ITEM_IDS
105
106 try:
107 from importlib.resources import files
108 except ImportError:
109 from importlib_resources import files
110
111 from . import data
112
113 # Load in all item and location IDs. These are broken up into groups based on the type of item/location.
114 with files(data).joinpath("ids.yaml").open() as file:
115 config = Utils.parse_yaml(file)
116
117 if "special_items" in config:
118 for item_name, item_id in config["special_items"].items():
119 SPECIAL_ITEM_IDS[item_name] = item_id
120
121 if "panels" in config:
122 for room_name in config["panels"].keys():
123 PANEL_LOCATION_IDS[room_name] = {}
124
125 for panel_name, location_id in config["panels"][room_name].items():
126 PANEL_LOCATION_IDS[room_name][panel_name] = location_id
127
128 if "doors" in config:
129 for room_name in config["doors"].keys():
130 DOOR_LOCATION_IDS[room_name] = {}
131 DOOR_ITEM_IDS[room_name] = {}
132
133 for door_name, door_data in config["doors"][room_name].items():
134 if "location" in door_data:
135 DOOR_LOCATION_IDS[room_name][door_name] = door_data["location"]
136
137 if "item" in door_data:
138 DOOR_ITEM_IDS[room_name][door_name] = door_data["item"]
139
140 if "door_groups" in config:
141 for item_name, item_id in config["door_groups"].items():
142 DOOR_GROUP_ITEM_IDS[item_name] = item_id
143
144 if "progression" in config:
145 for item_name, item_id in config["progression"].items():
146 PROGRESSIVE_ITEM_IDS[item_name] = item_id
147
148 # Process the main world file.
149 with files(data).joinpath("LL1.yaml").open() as file:
150 config = Utils.parse_yaml(file)
151
152 for room_name, room_data in config.items():
153 process_room(room_name, room_data)
154
155 PAINTING_EXITS = len(PAINTING_EXIT_ROOMS)
156 32
157 33
158def get_special_item_id(name: str): 34def get_special_item_id(name: str):
@@ -197,363 +73,39 @@ def get_progressive_item_id(name: str):
197 return PROGRESSIVE_ITEM_IDS[name] 73 return PROGRESSIVE_ITEM_IDS[name]
198 74
199 75
200def process_entrance(source_room, doors, room_obj): 76def load_static_data_from_file():
201 global PAINTING_ENTRANCES, PAINTING_EXIT_ROOMS 77 global PAINTING_ENTRANCES, PAINTING_EXITS
202 78
203 # If the value of an entrance is just True, that means that the entrance is always accessible. 79 class RenameUnpickler(pickle.Unpickler):
204 if doors is True: 80 def find_class(self, module, name):
205 room_obj.entrances.append(RoomEntrance(source_room, None, False)) 81 renamed_module = module
206 elif isinstance(doors, dict): 82 if module == "datatypes":
207 # If the value of an entrance is a dictionary, that means the entrance requires a door to be accessible, is a 83 renamed_module = "worlds.lingo.datatypes"
208 # painting-based entrance, or both. 84
209 if "painting" in doors and "door" not in doors: 85 return super(RenameUnpickler, self).find_class(renamed_module, name)
210 PAINTING_EXIT_ROOMS.add(room_obj.name) 86
211 PAINTING_ENTRANCES += 1 87 file = pkgutil.get_data(__name__, os.path.join("data", "generated.dat"))
212 88 pickdata = RenameUnpickler(BytesIO(file)).load()
213 room_obj.entrances.append(RoomEntrance(source_room, None, True)) 89
214 else: 90 HASHES.update(pickdata["HASHES"])
215 if "painting" in doors and doors["painting"]: 91 PAINTINGS.update(pickdata["PAINTINGS"])
216 PAINTING_EXIT_ROOMS.add(room_obj.name) 92 ALL_ROOMS.extend(pickdata["ALL_ROOMS"])
217 PAINTING_ENTRANCES += 1 93 DOORS_BY_ROOM.update(pickdata["DOORS_BY_ROOM"])
218 94 PANELS_BY_ROOM.update(pickdata["PANELS_BY_ROOM"])
219 room_obj.entrances.append(RoomEntrance(source_room, RoomAndDoor( 95 PROGRESSIVE_ITEMS.extend(pickdata["PROGRESSIVE_ITEMS"])
220 doors["room"] if "room" in doors else None, 96 PROGRESSION_BY_ROOM.update(pickdata["PROGRESSION_BY_ROOM"])
221 doors["door"] 97 PAINTING_ENTRANCES = pickdata["PAINTING_ENTRANCES"]
222 ), doors["painting"] if "painting" in doors else False)) 98 PAINTING_EXIT_ROOMS.update(pickdata["PAINTING_EXIT_ROOMS"])
223 else: 99 PAINTING_EXITS = pickdata["PAINTING_EXITS"]
224 # If the value of an entrance is a list, then there are multiple possible doors that can give access to the 100 REQUIRED_PAINTING_ROOMS.extend(pickdata["REQUIRED_PAINTING_ROOMS"])
225 # entrance. 101 REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS.extend(pickdata["REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS"])
226 for door in doors: 102 SPECIAL_ITEM_IDS.update(pickdata["SPECIAL_ITEM_IDS"])
227 if "painting" in door and door["painting"]: 103 PANEL_LOCATION_IDS.update(pickdata["PANEL_LOCATION_IDS"])
228 PAINTING_EXIT_ROOMS.add(room_obj.name) 104 DOOR_LOCATION_IDS.update(pickdata["DOOR_LOCATION_IDS"])
229 PAINTING_ENTRANCES += 1 105 DOOR_ITEM_IDS.update(pickdata["DOOR_ITEM_IDS"])
230 106 DOOR_GROUP_ITEM_IDS.update(pickdata["DOOR_GROUP_ITEM_IDS"])
231 room_obj.entrances.append(RoomEntrance(source_room, RoomAndDoor( 107 PROGRESSIVE_ITEM_IDS.update(pickdata["PROGRESSIVE_ITEM_IDS"])
232 door["room"] if "room" in door else None,
233 door["door"]
234 ), door["painting"] if "painting" in door else False))
235
236
237def process_panel(room_name, panel_name, panel_data):
238 global PANELS, PANELS_BY_ROOM
239
240 full_name = f"{room_name} - {panel_name}"
241
242 # required_room can either be a single room or a list of rooms.
243 if "required_room" in panel_data:
244 if isinstance(panel_data["required_room"], list):
245 required_rooms = panel_data["required_room"]
246 else:
247 required_rooms = [panel_data["required_room"]]
248 else:
249 required_rooms = []
250
251 # required_door can either be a single door or a list of doors. For convenience, the room key for each door does not
252 # need to be specified if the door is in this room.
253 required_doors = list()
254 if "required_door" in panel_data:
255 if isinstance(panel_data["required_door"], dict):
256 door = panel_data["required_door"]
257 required_doors.append(RoomAndDoor(
258 door["room"] if "room" in door else None,
259 door["door"]
260 ))
261 else:
262 for door in panel_data["required_door"]:
263 required_doors.append(RoomAndDoor(
264 door["room"] if "room" in door else None,
265 door["door"]
266 ))
267
268 # required_panel can either be a single panel or a list of panels. For convenience, the room key for each panel does
269 # not need to be specified if the panel is in this room.
270 required_panels = list()
271 if "required_panel" in panel_data:
272 if isinstance(panel_data["required_panel"], dict):
273 other_panel = panel_data["required_panel"]
274 required_panels.append(RoomAndPanel(
275 other_panel["room"] if "room" in other_panel else None,
276 other_panel["panel"]
277 ))
278 else:
279 for other_panel in panel_data["required_panel"]:
280 required_panels.append(RoomAndPanel(
281 other_panel["room"] if "room" in other_panel else None,
282 other_panel["panel"]
283 ))
284
285 # colors can either be a single color or a list of colors.
286 if "colors" in panel_data:
287 if isinstance(panel_data["colors"], list):
288 colors = panel_data["colors"]
289 else:
290 colors = [panel_data["colors"]]
291 else:
292 colors = []
293
294 if "check" in panel_data:
295 check = panel_data["check"]
296 else:
297 check = False
298
299 if "event" in panel_data:
300 event = panel_data["event"]
301 else:
302 event = False
303
304 if "achievement" in panel_data:
305 achievement = True
306 else:
307 achievement = False
308
309 if "exclude_reduce" in panel_data:
310 exclude_reduce = panel_data["exclude_reduce"]
311 else:
312 exclude_reduce = False
313
314 if "non_counting" in panel_data:
315 non_counting = panel_data["non_counting"]
316 else:
317 non_counting = False
318
319 if "id" in panel_data:
320 if isinstance(panel_data["id"], list):
321 internal_ids = panel_data["id"]
322 else:
323 internal_ids = [panel_data["id"]]
324 else:
325 internal_ids = []
326
327 panel_obj = Panel(required_rooms, required_doors, required_panels, colors, check, event, internal_ids,
328 exclude_reduce, achievement, non_counting)
329 PANELS[full_name] = panel_obj
330 PANELS_BY_ROOM[room_name][panel_name] = panel_obj
331
332
333def process_door(room_name, door_name, door_data):
334 global DOORS, DOORS_BY_ROOM
335
336 # The item name associated with a door can be explicitly specified in the configuration. If it is not, it is
337 # generated from the room and door name.
338 if "item_name" in door_data:
339 item_name = door_data["item_name"]
340 else:
341 item_name = f"{room_name} - {door_name}"
342
343 if "skip_location" in door_data:
344 skip_location = door_data["skip_location"]
345 else:
346 skip_location = False
347
348 if "skip_item" in door_data:
349 skip_item = door_data["skip_item"]
350 else:
351 skip_item = False
352
353 if "event" in door_data:
354 event = door_data["event"]
355 else:
356 event = False
357
358 if "include_reduce" in door_data:
359 include_reduce = door_data["include_reduce"]
360 else:
361 include_reduce = False
362
363 if "junk_item" in door_data:
364 junk_item = door_data["junk_item"]
365 else:
366 junk_item = False
367
368 if "group" in door_data:
369 group = door_data["group"]
370 else:
371 group = None
372
373 # panels is a list of panels. Each panel can either be a simple string (the name of a panel in the current room) or
374 # a dictionary specifying a panel in a different room.
375 if "panels" in door_data:
376 panels = list()
377 for panel in door_data["panels"]:
378 if isinstance(panel, dict):
379 panels.append(RoomAndPanel(panel["room"], panel["panel"]))
380 else:
381 panels.append(RoomAndPanel(None, panel))
382 else:
383 skip_location = True
384 panels = None
385
386 # The location name associated with a door can be explicitly specified in the configuration. If it is not, then the
387 # name is generated using a combination of all of the panels that would ordinarily open the door. This can get quite
388 # messy if there are a lot of panels, especially if panels from multiple rooms are involved, so in these cases it
389 # would be better to specify a name.
390 if "location_name" in door_data:
391 location_name = door_data["location_name"]
392 elif skip_location is False:
393 panel_per_room = dict()
394 for panel in panels:
395 panel_room_name = room_name if panel.room is None else panel.room
396 panel_per_room.setdefault(panel_room_name, []).append(panel.panel)
397
398 room_strs = list()
399 for door_room_str, door_panels_str in panel_per_room.items():
400 room_strs.append(door_room_str + " - " + ", ".join(door_panels_str))
401
402 location_name = " and ".join(room_strs)
403 else:
404 location_name = None
405
406 # The id field can be a single item, or a list of door IDs, in the event that the item for this logical door should
407 # open more than one actual in-game door.
408 if "id" in door_data:
409 if isinstance(door_data["id"], list):
410 door_ids = door_data["id"]
411 else:
412 door_ids = [door_data["id"]]
413 else:
414 door_ids = []
415
416 # The painting_id field can be a single item, or a list of painting IDs, in the event that the item for this logical
417 # door should move more than one actual in-game painting.
418 if "painting_id" in door_data:
419 if isinstance(door_data["painting_id"], list):
420 painting_ids = door_data["painting_id"]
421 else:
422 painting_ids = [door_data["painting_id"]]
423 else:
424 painting_ids = []
425
426 door_obj = Door(door_name, item_name, location_name, panels, skip_location, skip_item, door_ids,
427 painting_ids, event, group, include_reduce, junk_item)
428
429 DOORS[door_obj.item_name] = door_obj
430 DOORS_BY_ROOM[room_name][door_name] = door_obj
431
432
433def process_painting(room_name, painting_data):
434 global PAINTINGS, PAINTINGS_BY_ROOM, REQUIRED_PAINTING_ROOMS, REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS
435
436 # Read in information about this painting and store it in an object.
437 painting_id = painting_data["id"]
438
439 if "orientation" in painting_data:
440 orientation = painting_data["orientation"]
441 else:
442 orientation = ""
443
444 if "disable" in painting_data:
445 disable_painting = painting_data["disable"]
446 else:
447 disable_painting = False
448
449 if "required" in painting_data:
450 required_painting = painting_data["required"]
451 if required_painting:
452 REQUIRED_PAINTING_ROOMS.append(room_name)
453 else:
454 required_painting = False
455
456 if "move" in painting_data:
457 move_painting = painting_data["move"]
458 else:
459 move_painting = False
460
461 if "required_when_no_doors" in painting_data:
462 rwnd = painting_data["required_when_no_doors"]
463 if rwnd:
464 REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS.append(room_name)
465 else:
466 rwnd = False
467
468 if "exit_only" in painting_data:
469 exit_only = painting_data["exit_only"]
470 else:
471 exit_only = False
472
473 if "enter_only" in painting_data:
474 enter_only = painting_data["enter_only"]
475 else:
476 enter_only = False
477
478 if "req_blocked" in painting_data:
479 req_blocked = painting_data["req_blocked"]
480 else:
481 req_blocked = False
482
483 if "req_blocked_when_no_doors" in painting_data:
484 req_blocked_when_no_doors = painting_data["req_blocked_when_no_doors"]
485 else:
486 req_blocked_when_no_doors = False
487
488 required_door = None
489 if "required_door" in painting_data:
490 door = painting_data["required_door"]
491 required_door = RoomAndDoor(
492 door["room"] if "room" in door else room_name,
493 door["door"]
494 )
495
496 painting_obj = Painting(painting_id, room_name, enter_only, exit_only, orientation,
497 required_painting, rwnd, required_door, disable_painting, move_painting, req_blocked,
498 req_blocked_when_no_doors)
499 PAINTINGS[painting_id] = painting_obj
500 PAINTINGS_BY_ROOM[room_name].append(painting_obj)
501
502
503def process_progression(room_name, progression_name, progression_doors):
504 global PROGRESSIVE_ITEMS, PROGRESSION_BY_ROOM
505
506 # Progressive items are configured as a list of doors.
507 PROGRESSIVE_ITEMS.append(progression_name)
508
509 progression_index = 1
510 for door in progression_doors:
511 if isinstance(door, Dict):
512 door_room = door["room"]
513 door_door = door["door"]
514 else:
515 door_room = room_name
516 door_door = door
517
518 room_progressions = PROGRESSION_BY_ROOM.setdefault(door_room, {})
519 room_progressions[door_door] = Progression(progression_name, progression_index)
520 progression_index += 1
521
522
523def process_room(room_name, room_data):
524 global ROOMS, ALL_ROOMS
525
526 room_obj = Room(room_name, [])
527
528 if "entrances" in room_data:
529 for source_room, doors in room_data["entrances"].items():
530 process_entrance(source_room, doors, room_obj)
531
532 if "panels" in room_data:
533 PANELS_BY_ROOM[room_name] = dict()
534
535 for panel_name, panel_data in room_data["panels"].items():
536 process_panel(room_name, panel_name, panel_data)
537
538 if "doors" in room_data:
539 DOORS_BY_ROOM[room_name] = dict()
540
541 for door_name, door_data in room_data["doors"].items():
542 process_door(room_name, door_name, door_data)
543
544 if "paintings" in room_data:
545 PAINTINGS_BY_ROOM[room_name] = []
546
547 for painting_data in room_data["paintings"]:
548 process_painting(room_name, painting_data)
549
550 if "progression" in room_data:
551 for progression_name, progression_doors in room_data["progression"].items():
552 process_progression(room_name, progression_name, progression_doors)
553
554 ROOMS[room_name] = room_obj
555 ALL_ROOMS.append(room_obj)
556 108
557 109
558# Initialize the static data at module scope. 110# Initialize the static data at module scope.
559load_static_data() 111load_static_data_from_file()
diff --git a/test/TestDatafile.py b/test/TestDatafile.py new file mode 100644 index 0000000..9f4e9da --- /dev/null +++ b/test/TestDatafile.py
@@ -0,0 +1,16 @@
1import os
2import unittest
3
4from worlds.lingo.static_logic import HASHES
5from worlds.lingo.utils.pickle_static_data import hash_file
6
7
8class TestDatafile(unittest.TestCase):
9 def test_check_hashes(self) -> None:
10 ll1_file_hash = hash_file(os.path.join(os.path.dirname(__file__), "..", "data", "LL1.yaml"))
11 ids_file_hash = hash_file(os.path.join(os.path.dirname(__file__), "..", "data", "ids.yaml"))
12
13 self.assertEqual(ll1_file_hash, HASHES["LL1.yaml"],
14 "LL1.yaml hash does not match generated.dat. Please regenerate using 'python worlds/lingo/utils/pickle_static_data.py'")
15 self.assertEqual(ids_file_hash, HASHES["ids.yaml"],
16 "ids.yaml hash does not match generated.dat. Please regenerate using 'python worlds/lingo/utils/pickle_static_data.py'")
diff --git a/utils/__init__.py b/utils/__init__.py new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/utils/__init__.py
diff --git a/utils/pickle_static_data.py b/utils/pickle_static_data.py new file mode 100644 index 0000000..c7a2711 --- /dev/null +++ b/utils/pickle_static_data.py
@@ -0,0 +1,475 @@
1from typing import Dict, List, Set
2
3import os
4import sys
5
6sys.path.append(os.path.join("worlds", "lingo"))
7sys.path.append(".")
8sys.path.append("..")
9from datatypes import Door, Painting, Panel, Progression, Room, RoomAndDoor, RoomAndPanel, RoomEntrance
10
11import hashlib
12import pickle
13import sys
14import Utils
15
16
17ALL_ROOMS: List[Room] = []
18DOORS_BY_ROOM: Dict[str, Dict[str, Door]] = {}
19PANELS_BY_ROOM: Dict[str, Dict[str, Panel]] = {}
20PAINTINGS: Dict[str, Painting] = {}
21
22PROGRESSIVE_ITEMS: List[str] = []
23PROGRESSION_BY_ROOM: Dict[str, Dict[str, Progression]] = {}
24
25PAINTING_ENTRANCES: int = 0
26PAINTING_EXIT_ROOMS: Set[str] = set()
27PAINTING_EXITS: int = 0
28REQUIRED_PAINTING_ROOMS: List[str] = []
29REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = []
30
31SPECIAL_ITEM_IDS: Dict[str, int] = {}
32PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {}
33DOOR_LOCATION_IDS: Dict[str, Dict[str, int]] = {}
34DOOR_ITEM_IDS: Dict[str, Dict[str, int]] = {}
35DOOR_GROUP_ITEM_IDS: Dict[str, int] = {}
36PROGRESSIVE_ITEM_IDS: Dict[str, int] = {}
37
38
39def hash_file(path):
40 md5 = hashlib.md5()
41
42 with open(path, 'rb') as f:
43 content = f.read()
44 content = content.replace(b'\r\n', b'\n')
45 md5.update(content)
46
47 return md5.hexdigest()
48
49
50def load_static_data(ll1_path, ids_path):
51 global PAINTING_EXITS, SPECIAL_ITEM_IDS, PANEL_LOCATION_IDS, DOOR_LOCATION_IDS, DOOR_ITEM_IDS, \
52 DOOR_GROUP_ITEM_IDS, PROGRESSIVE_ITEM_IDS
53
54 # Load in all item and location IDs. These are broken up into groups based on the type of item/location.
55 with open(ids_path, "r") as file:
56 config = Utils.parse_yaml(file)
57
58 if "special_items" in config:
59 for item_name, item_id in config["special_items"].items():
60 SPECIAL_ITEM_IDS[item_name] = item_id
61
62 if "panels" in config:
63 for room_name in config["panels"].keys():
64 PANEL_LOCATION_IDS[room_name] = {}
65
66 for panel_name, location_id in config["panels"][room_name].items():
67 PANEL_LOCATION_IDS[room_name][panel_name] = location_id
68
69 if "doors" in config:
70 for room_name in config["doors"].keys():
71 DOOR_LOCATION_IDS[room_name] = {}
72 DOOR_ITEM_IDS[room_name] = {}
73
74 for door_name, door_data in config["doors"][room_name].items():
75 if "location" in door_data:
76 DOOR_LOCATION_IDS[room_name][door_name] = door_data["location"]
77
78 if "item" in door_data:
79 DOOR_ITEM_IDS[room_name][door_name] = door_data["item"]
80
81 if "door_groups" in config:
82 for item_name, item_id in config["door_groups"].items():
83 DOOR_GROUP_ITEM_IDS[item_name] = item_id
84
85 if "progression" in config:
86 for item_name, item_id in config["progression"].items():
87 PROGRESSIVE_ITEM_IDS[item_name] = item_id
88
89 # Process the main world file.
90 with open(ll1_path, "r") as file:
91 config = Utils.parse_yaml(file)
92
93 for room_name, room_data in config.items():
94 process_room(room_name, room_data)
95
96 PAINTING_EXITS = len(PAINTING_EXIT_ROOMS)
97
98
99def process_entrance(source_room, doors, room_obj):
100 global PAINTING_ENTRANCES, PAINTING_EXIT_ROOMS
101
102 # If the value of an entrance is just True, that means that the entrance is always accessible.
103 if doors is True:
104 room_obj.entrances.append(RoomEntrance(source_room, None, False))
105 elif isinstance(doors, dict):
106 # If the value of an entrance is a dictionary, that means the entrance requires a door to be accessible, is a
107 # painting-based entrance, or both.
108 if "painting" in doors and "door" not in doors:
109 PAINTING_EXIT_ROOMS.add(room_obj.name)
110 PAINTING_ENTRANCES += 1
111
112 room_obj.entrances.append(RoomEntrance(source_room, None, True))
113 else:
114 if "painting" in doors and doors["painting"]:
115 PAINTING_EXIT_ROOMS.add(room_obj.name)
116 PAINTING_ENTRANCES += 1
117
118 room_obj.entrances.append(RoomEntrance(source_room, RoomAndDoor(
119 doors["room"] if "room" in doors else None,
120 doors["door"]
121 ), doors["painting"] if "painting" in doors else False))
122 else:
123 # If the value of an entrance is a list, then there are multiple possible doors that can give access to the
124 # entrance.
125 for door in doors:
126 if "painting" in door and door["painting"]:
127 PAINTING_EXIT_ROOMS.add(room_obj.name)
128 PAINTING_ENTRANCES += 1
129
130 room_obj.entrances.append(RoomEntrance(source_room, RoomAndDoor(
131 door["room"] if "room" in door else None,
132 door["door"]
133 ), door["painting"] if "painting" in door else False))
134
135
136def process_panel(room_name, panel_name, panel_data):
137 global PANELS_BY_ROOM
138
139 full_name = f"{room_name} - {panel_name}"
140
141 # required_room can either be a single room or a list of rooms.
142 if "required_room" in panel_data:
143 if isinstance(panel_data["required_room"], list):
144 required_rooms = panel_data["required_room"]
145 else:
146 required_rooms = [panel_data["required_room"]]
147 else:
148 required_rooms = []
149
150 # required_door can either be a single door or a list of doors. For convenience, the room key for each door does not
151 # need to be specified if the door is in this room.
152 required_doors = list()
153 if "required_door" in panel_data:
154 if isinstance(panel_data["required_door"], dict):
155 door = panel_data["required_door"]
156 required_doors.append(RoomAndDoor(
157 door["room"] if "room" in door else None,
158 door["door"]
159 ))
160 else:
161 for door in panel_data["required_door"]:
162 required_doors.append(RoomAndDoor(
163 door["room"] if "room" in door else None,
164 door["door"]
165 ))
166
167 # required_panel can either be a single panel or a list of panels. For convenience, the room key for each panel does
168 # not need to be specified if the panel is in this room.
169 required_panels = list()
170 if "required_panel" in panel_data:
171 if isinstance(panel_data["required_panel"], dict):
172 other_panel = panel_data["required_panel"]
173 required_panels.append(RoomAndPanel(
174 other_panel["room"] if "room" in other_panel else None,
175 other_panel["panel"]
176 ))
177 else:
178 for other_panel in panel_data["required_panel"]:
179 required_panels.append(RoomAndPanel(
180 other_panel["room"] if "room" in other_panel else None,
181 other_panel["panel"]
182 ))
183
184 # colors can either be a single color or a list of colors.
185 if "colors" in panel_data:
186 if isinstance(panel_data["colors"], list):
187 colors = panel_data["colors"]
188 else:
189 colors = [panel_data["colors"]]
190 else:
191 colors = []
192
193 if "check" in panel_data:
194 check = panel_data["check"]
195 else:
196 check = False
197
198 if "event" in panel_data:
199 event = panel_data["event"]
200 else:
201 event = False
202
203 if "achievement" in panel_data:
204 achievement = True
205 else:
206 achievement = False
207
208 if "exclude_reduce" in panel_data:
209 exclude_reduce = panel_data["exclude_reduce"]
210 else:
211 exclude_reduce = False
212
213 if "non_counting" in panel_data:
214 non_counting = panel_data["non_counting"]
215 else:
216 non_counting = False
217
218 panel_obj = Panel(required_rooms, required_doors, required_panels, colors, check, event, exclude_reduce,
219 achievement, non_counting)
220 PANELS_BY_ROOM[room_name][panel_name] = panel_obj
221
222
223def process_door(room_name, door_name, door_data):
224 global DOORS_BY_ROOM
225
226 # The item name associated with a door can be explicitly specified in the configuration. If it is not, it is
227 # generated from the room and door name.
228 if "item_name" in door_data:
229 item_name = door_data["item_name"]
230 else:
231 item_name = f"{room_name} - {door_name}"
232
233 if "skip_location" in door_data:
234 skip_location = door_data["skip_location"]
235 else:
236 skip_location = False
237
238 if "skip_item" in door_data:
239 skip_item = door_data["skip_item"]
240 else:
241 skip_item = False
242
243 if "event" in door_data:
244 event = door_data["event"]
245 else:
246 event = False
247
248 if "include_reduce" in door_data:
249 include_reduce = door_data["include_reduce"]
250 else:
251 include_reduce = False
252
253 if "junk_item" in door_data:
254 junk_item = door_data["junk_item"]
255 else:
256 junk_item = False
257
258 if "group" in door_data:
259 group = door_data["group"]
260 else:
261 group = None
262
263 # panels is a list of panels. Each panel can either be a simple string (the name of a panel in the current room) or
264 # a dictionary specifying a panel in a different room.
265 if "panels" in door_data:
266 panels = list()
267 for panel in door_data["panels"]:
268 if isinstance(panel, dict):
269 panels.append(RoomAndPanel(panel["room"], panel["panel"]))
270 else:
271 panels.append(RoomAndPanel(None, panel))
272 else:
273 skip_location = True
274 panels = None
275
276 # The location name associated with a door can be explicitly specified in the configuration. If it is not, then the
277 # name is generated using a combination of all of the panels that would ordinarily open the door. This can get quite
278 # messy if there are a lot of panels, especially if panels from multiple rooms are involved, so in these cases it
279 # would be better to specify a name.
280 if "location_name" in door_data:
281 location_name = door_data["location_name"]
282 elif skip_location is False:
283 panel_per_room = dict()
284 for panel in panels:
285 panel_room_name = room_name if panel.room is None else panel.room
286 panel_per_room.setdefault(panel_room_name, []).append(panel.panel)
287
288 room_strs = list()
289 for door_room_str, door_panels_str in panel_per_room.items():
290 room_strs.append(door_room_str + " - " + ", ".join(door_panels_str))
291
292 location_name = " and ".join(room_strs)
293 else:
294 location_name = None
295
296 # The id field can be a single item, or a list of door IDs, in the event that the item for this logical door should
297 # open more than one actual in-game door.
298 has_doors = "id" in door_data
299
300 # The painting_id field can be a single item, or a list of painting IDs, in the event that the item for this logical
301 # door should move more than one actual in-game painting.
302 if "painting_id" in door_data:
303 if isinstance(door_data["painting_id"], list):
304 painting_ids = door_data["painting_id"]
305 else:
306 painting_ids = [door_data["painting_id"]]
307 else:
308 painting_ids = []
309
310 door_obj = Door(door_name, item_name, location_name, panels, skip_location, skip_item, has_doors,
311 painting_ids, event, group, include_reduce, junk_item)
312
313 DOORS_BY_ROOM[room_name][door_name] = door_obj
314
315
316def process_painting(room_name, painting_data):
317 global PAINTINGS, REQUIRED_PAINTING_ROOMS, REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS
318
319 # Read in information about this painting and store it in an object.
320 painting_id = painting_data["id"]
321
322 if "disable" in painting_data:
323 disable_painting = painting_data["disable"]
324 else:
325 disable_painting = False
326
327 if "required" in painting_data:
328 required_painting = painting_data["required"]
329 if required_painting:
330 REQUIRED_PAINTING_ROOMS.append(room_name)
331 else:
332 required_painting = False
333
334 if "required_when_no_doors" in painting_data:
335 rwnd = painting_data["required_when_no_doors"]
336 if rwnd:
337 REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS.append(room_name)
338 else:
339 rwnd = False
340
341 if "exit_only" in painting_data:
342 exit_only = painting_data["exit_only"]
343 else:
344 exit_only = False
345
346 if "enter_only" in painting_data:
347 enter_only = painting_data["enter_only"]
348 else:
349 enter_only = False
350
351 if "req_blocked" in painting_data:
352 req_blocked = painting_data["req_blocked"]
353 else:
354 req_blocked = False
355
356 if "req_blocked_when_no_doors" in painting_data:
357 req_blocked_when_no_doors = painting_data["req_blocked_when_no_doors"]
358 else:
359 req_blocked_when_no_doors = False
360
361 required_door = None
362 if "required_door" in painting_data:
363 door = painting_data["required_door"]
364 required_door = RoomAndDoor(
365 door["room"] if "room" in door else room_name,
366 door["door"]
367 )
368
369 painting_obj = Painting(painting_id, room_name, enter_only, exit_only,
370 required_painting, rwnd, required_door, disable_painting, req_blocked,
371 req_blocked_when_no_doors)
372 PAINTINGS[painting_id] = painting_obj
373
374
375def process_progression(room_name, progression_name, progression_doors):
376 global PROGRESSIVE_ITEMS, PROGRESSION_BY_ROOM
377
378 # Progressive items are configured as a list of doors.
379 PROGRESSIVE_ITEMS.append(progression_name)
380
381 progression_index = 1
382 for door in progression_doors:
383 if isinstance(door, Dict):
384 door_room = door["room"]
385 door_door = door["door"]
386 else:
387 door_room = room_name
388 door_door = door
389
390 room_progressions = PROGRESSION_BY_ROOM.setdefault(door_room, {})
391 room_progressions[door_door] = Progression(progression_name, progression_index)
392 progression_index += 1
393
394
395def process_room(room_name, room_data):
396 global ALL_ROOMS
397
398 room_obj = Room(room_name, [])
399
400 if "entrances" in room_data:
401 for source_room, doors in room_data["entrances"].items():
402 process_entrance(source_room, doors, room_obj)
403
404 if "panels" in room_data:
405 PANELS_BY_ROOM[room_name] = dict()
406
407 for panel_name, panel_data in room_data["panels"].items():
408 process_panel(room_name, panel_name, panel_data)
409
410 if "doors" in room_data:
411 DOORS_BY_ROOM[room_name] = dict()
412
413 for door_name, door_data in room_data["doors"].items():
414 process_door(room_name, door_name, door_data)
415
416 if "paintings" in room_data:
417 for painting_data in room_data["paintings"]:
418 process_painting(room_name, painting_data)
419
420 if "progression" in room_data:
421 for progression_name, progression_doors in room_data["progression"].items():
422 process_progression(room_name, progression_name, progression_doors)
423
424 ALL_ROOMS.append(room_obj)
425
426
427if __name__ == '__main__':
428 if len(sys.argv) == 1:
429 ll1_path = os.path.join("worlds", "lingo", "data", "LL1.yaml")
430 ids_path = os.path.join("worlds", "lingo", "data", "ids.yaml")
431 output_path = os.path.join("worlds", "lingo", "data", "generated.dat")
432 elif len(sys.argv) != 4:
433 print("")
434 print("Usage: python worlds/lingo/utils/pickle_static_data.py [args]")
435 print("Arguments:")
436 print(" - Path to LL1.yaml")
437 print(" - Path to ids.yaml")
438 print(" - Path to output file")
439
440 exit()
441 else:
442 ll1_path = sys.argv[1]
443 ids_path = sys.argv[2]
444 output_path = sys.argv[3]
445
446 load_static_data(ll1_path, ids_path)
447
448 hashes = {
449 "LL1.yaml": hash_file(ll1_path),
450 "ids.yaml": hash_file(ids_path),
451 }
452
453 pickdata = {
454 "HASHES": hashes,
455 "PAINTINGS": PAINTINGS,
456 "ALL_ROOMS": ALL_ROOMS,
457 "DOORS_BY_ROOM": DOORS_BY_ROOM,
458 "PANELS_BY_ROOM": PANELS_BY_ROOM,
459 "PROGRESSIVE_ITEMS": PROGRESSIVE_ITEMS,
460 "PROGRESSION_BY_ROOM": PROGRESSION_BY_ROOM,
461 "PAINTING_ENTRANCES": PAINTING_ENTRANCES,
462 "PAINTING_EXIT_ROOMS": PAINTING_EXIT_ROOMS,
463 "PAINTING_EXITS": PAINTING_EXITS,
464 "REQUIRED_PAINTING_ROOMS": REQUIRED_PAINTING_ROOMS,
465 "REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS": REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS,
466 "SPECIAL_ITEM_IDS": SPECIAL_ITEM_IDS,
467 "PANEL_LOCATION_IDS": PANEL_LOCATION_IDS,
468 "DOOR_LOCATION_IDS": DOOR_LOCATION_IDS,
469 "DOOR_ITEM_IDS": DOOR_ITEM_IDS,
470 "DOOR_GROUP_ITEM_IDS": DOOR_GROUP_ITEM_IDS,
471 "PROGRESSIVE_ITEM_IDS": PROGRESSIVE_ITEM_IDS,
472 }
473
474 with open(output_path, "wb") as file:
475 pickle.dump(pickdata, file)