diff options
author | Star Rauchenberger <fefferburbia@gmail.com> | 2024-03-15 04:26:00 -0400 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-03-15 09:26:00 +0100 |
commit | e016228d2b76fa367889c9d98d6eb9e783f98cb4 (patch) | |
tree | 5c4fd5427ee1c89a832b3327eef6d39d4ce82d82 | |
parent | fcb3b36f44a7d475aa49c5c44971a2b7cabc4ca8 (diff) | |
download | lingo-apworld-e016228d2b76fa367889c9d98d6eb9e783f98cb4.tar.gz lingo-apworld-e016228d2b76fa367889c9d98d6eb9e783f98cb4.tar.bz2 lingo-apworld-e016228d2b76fa367889c9d98d6eb9e783f98cb4.zip |
Lingo: Pre-compile datafile to improve loading time (#2829)
-rw-r--r-- | __init__.py | 8 | ||||
-rw-r--r-- | data/README.md | 5 | ||||
-rw-r--r-- | data/generated.dat | bin | 0 -> 129731 bytes | |||
-rw-r--r-- | datatypes.py | 67 | ||||
-rw-r--r-- | items.py | 12 | ||||
-rw-r--r-- | locations.py | 11 | ||||
-rw-r--r-- | player_logic.py | 11 | ||||
-rw-r--r-- | regions.py | 3 | ||||
-rw-r--r-- | rules.py | 3 | ||||
-rw-r--r-- | static_logic.py | 530 | ||||
-rw-r--r-- | test/TestDatafile.py | 16 | ||||
-rw-r--r-- | utils/__init__.py | 0 | ||||
-rw-r--r-- | utils/pickle_static_data.py | 475 |
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 | ||
6 | from BaseClasses import Item, ItemClassification, Tutorial | 6 | from BaseClasses import Item, ItemClassification, Tutorial |
7 | from worlds.AutoWorld import WebWorld, World | 7 | from worlds.AutoWorld import WebWorld, World |
8 | from .datatypes import Room, RoomEntrance | ||
8 | from .items import ALL_ITEM_TABLE, LingoItem | 9 | from .items import ALL_ITEM_TABLE, LingoItem |
9 | from .locations import ALL_LOCATION_TABLE | 10 | from .locations import ALL_LOCATION_TABLE |
10 | from .options import LingoOptions | 11 | from .options import LingoOptions |
11 | from .player_logic import LingoPlayerLogic | 12 | from .player_logic import LingoPlayerLogic |
12 | from .regions import create_regions | 13 | from .regions import create_regions |
13 | from .static_logic import Room, RoomEntrance | ||
14 | 14 | ||
15 | 15 | ||
16 | class LingoWebWorld(WebWorld): | 16 | class 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 | |||
3 | The 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 | |||
5 | The 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 @@ | |||
1 | from typing import List, NamedTuple, Optional | ||
2 | |||
3 | |||
4 | class RoomAndDoor(NamedTuple): | ||
5 | room: Optional[str] | ||
6 | door: str | ||
7 | |||
8 | |||
9 | class RoomAndPanel(NamedTuple): | ||
10 | room: Optional[str] | ||
11 | panel: str | ||
12 | |||
13 | |||
14 | class RoomEntrance(NamedTuple): | ||
15 | room: str # source room | ||
16 | door: Optional[RoomAndDoor] | ||
17 | painting: bool | ||
18 | |||
19 | |||
20 | class Room(NamedTuple): | ||
21 | name: str | ||
22 | entrances: List[RoomEntrance] | ||
23 | |||
24 | |||
25 | class 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 | |||
40 | class 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 | |||
52 | class 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 | |||
65 | class 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 | |||
2 | from typing import Dict, List, NamedTuple | 2 | from typing import Dict, List, NamedTuple |
3 | 3 | ||
4 | from BaseClasses import Location | 4 | from BaseClasses import Location |
5 | from .static_logic import DOORS_BY_ROOM, PANELS_BY_ROOM, RoomAndPanel, get_door_location_id, get_panel_location_id | 5 | from .datatypes import RoomAndPanel |
6 | from .static_logic import DOORS_BY_ROOM, PANELS_BY_ROOM, get_door_location_id, get_panel_location_id | ||
6 | 7 | ||
7 | 8 | ||
8 | class LocationClassification(Flag): | 9 | class 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 | ||
32 | class LingoLocation(Location): | 25 | class 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 @@ | |||
1 | from enum import Enum | 1 | from enum import Enum |
2 | from typing import Dict, List, NamedTuple, Optional, Set, Tuple, TYPE_CHECKING | 2 | from typing import Dict, List, NamedTuple, Optional, Set, Tuple, TYPE_CHECKING |
3 | 3 | ||
4 | from .datatypes import Door, RoomAndDoor, RoomAndPanel | ||
4 | from .items import ALL_ITEM_TABLE | 5 | from .items import ALL_ITEM_TABLE |
5 | from .locations import ALL_LOCATION_TABLE, LocationClassification | 6 | from .locations import ALL_LOCATION_TABLE, LocationClassification |
6 | from .options import LocationChecks, ShuffleDoors, VictoryCondition | 7 | from .options import LocationChecks, ShuffleDoors, VictoryCondition |
7 | from .static_logic import DOORS_BY_ROOM, Door, PAINTINGS, PAINTINGS_BY_ROOM, PAINTING_ENTRANCES, PAINTING_EXITS, \ | 8 | from .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 | ||
11 | if TYPE_CHECKING: | 11 | if 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 @@ | |||
1 | from typing import Dict, Optional, TYPE_CHECKING | 1 | from typing import Dict, Optional, TYPE_CHECKING |
2 | 2 | ||
3 | from BaseClasses import Entrance, ItemClassification, Region | 3 | from BaseClasses import Entrance, ItemClassification, Region |
4 | from .datatypes import Room, RoomAndDoor | ||
4 | from .items import LingoItem | 5 | from .items import LingoItem |
5 | from .locations import LingoLocation | 6 | from .locations import LingoLocation |
6 | from .player_logic import LingoPlayerLogic | 7 | from .player_logic import LingoPlayerLogic |
7 | from .rules import lingo_can_use_entrance, make_location_lambda | 8 | from .rules import lingo_can_use_entrance, make_location_lambda |
8 | from .static_logic import ALL_ROOMS, PAINTINGS, Room, RoomAndDoor | 9 | from .static_logic import ALL_ROOMS, PAINTINGS |
9 | 10 | ||
10 | if TYPE_CHECKING: | 11 | if 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 @@ | |||
1 | from typing import TYPE_CHECKING | 1 | from typing import TYPE_CHECKING |
2 | 2 | ||
3 | from BaseClasses import CollectionState | 3 | from BaseClasses import CollectionState |
4 | from .datatypes import RoomAndDoor | ||
4 | from .player_logic import AccessRequirements, LingoPlayerLogic, PlayerLocation | 5 | from .player_logic import AccessRequirements, LingoPlayerLogic, PlayerLocation |
5 | from .static_logic import PROGRESSION_BY_ROOM, PROGRESSIVE_ITEMS, RoomAndDoor | 6 | from .static_logic import PROGRESSION_BY_ROOM, PROGRESSIVE_ITEMS |
6 | 7 | ||
7 | if TYPE_CHECKING: | 8 | if 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 @@ | |||
1 | from typing import Dict, List, NamedTuple, Optional, Set | 1 | import os |
2 | import pkgutil | ||
3 | from io import BytesIO | ||
4 | from typing import Dict, List, Set | ||
2 | 5 | ||
3 | import Utils | 6 | import pickle |
4 | 7 | ||
5 | 8 | from .datatypes import Door, Painting, Panel, Progression, Room | |
6 | class RoomAndDoor(NamedTuple): | ||
7 | room: Optional[str] | ||
8 | door: str | ||
9 | |||
10 | |||
11 | class RoomAndPanel(NamedTuple): | ||
12 | room: Optional[str] | ||
13 | panel: str | ||
14 | |||
15 | |||
16 | class RoomEntrance(NamedTuple): | ||
17 | room: str # source room | ||
18 | door: Optional[RoomAndDoor] | ||
19 | painting: bool | ||
20 | |||
21 | |||
22 | class Room(NamedTuple): | ||
23 | name: str | ||
24 | entrances: List[RoomEntrance] | ||
25 | |||
26 | |||
27 | class 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 | |||
42 | class 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 | |||
55 | class 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 | |||
70 | class Progression(NamedTuple): | ||
71 | item_name: str | ||
72 | index: int | ||
73 | |||
74 | |||
75 | ROOMS: Dict[str, Room] = {} | ||
76 | PANELS: Dict[str, Panel] = {} | ||
77 | DOORS: Dict[str, Door] = {} | ||
78 | PAINTINGS: Dict[str, Painting] = {} | ||
79 | 9 | ||
80 | ALL_ROOMS: List[Room] = [] | 10 | ALL_ROOMS: List[Room] = [] |
81 | DOORS_BY_ROOM: Dict[str, Dict[str, Door]] = {} | 11 | DOORS_BY_ROOM: Dict[str, Dict[str, Door]] = {} |
82 | PANELS_BY_ROOM: Dict[str, Dict[str, Panel]] = {} | 12 | PANELS_BY_ROOM: Dict[str, Dict[str, Panel]] = {} |
83 | PAINTINGS_BY_ROOM: Dict[str, List[Painting]] = {} | 13 | PAINTINGS: Dict[str, Painting] = {} |
84 | 14 | ||
85 | PROGRESSIVE_ITEMS: List[str] = [] | 15 | PROGRESSIVE_ITEMS: List[str] = [] |
86 | PROGRESSION_BY_ROOM: Dict[str, Dict[str, Progression]] = {} | 16 | PROGRESSION_BY_ROOM: Dict[str, Dict[str, Progression]] = {} |
@@ -98,61 +28,7 @@ DOOR_ITEM_IDS: Dict[str, Dict[str, int]] = {} | |||
98 | DOOR_GROUP_ITEM_IDS: Dict[str, int] = {} | 28 | DOOR_GROUP_ITEM_IDS: Dict[str, int] = {} |
99 | PROGRESSIVE_ITEM_IDS: Dict[str, int] = {} | 29 | PROGRESSIVE_ITEM_IDS: Dict[str, int] = {} |
100 | 30 | ||
101 | 31 | HASHES: Dict[str, str] = {} | |
102 | def 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 | ||
158 | def get_special_item_id(name: str): | 34 | def 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 | ||
200 | def process_entrance(source_room, doors, room_obj): | 76 | def 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 | |||
237 | def 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 | |||
333 | def 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 | |||
433 | def 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 | |||
503 | def 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 | |||
523 | def 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. |
559 | load_static_data() | 111 | load_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 @@ | |||
1 | import os | ||
2 | import unittest | ||
3 | |||
4 | from worlds.lingo.static_logic import HASHES | ||
5 | from worlds.lingo.utils.pickle_static_data import hash_file | ||
6 | |||
7 | |||
8 | class 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 @@ | |||
1 | from typing import Dict, List, Set | ||
2 | |||
3 | import os | ||
4 | import sys | ||
5 | |||
6 | sys.path.append(os.path.join("worlds", "lingo")) | ||
7 | sys.path.append(".") | ||
8 | sys.path.append("..") | ||
9 | from datatypes import Door, Painting, Panel, Progression, Room, RoomAndDoor, RoomAndPanel, RoomEntrance | ||
10 | |||
11 | import hashlib | ||
12 | import pickle | ||
13 | import sys | ||
14 | import Utils | ||
15 | |||
16 | |||
17 | ALL_ROOMS: List[Room] = [] | ||
18 | DOORS_BY_ROOM: Dict[str, Dict[str, Door]] = {} | ||
19 | PANELS_BY_ROOM: Dict[str, Dict[str, Panel]] = {} | ||
20 | PAINTINGS: Dict[str, Painting] = {} | ||
21 | |||
22 | PROGRESSIVE_ITEMS: List[str] = [] | ||
23 | PROGRESSION_BY_ROOM: Dict[str, Dict[str, Progression]] = {} | ||
24 | |||
25 | PAINTING_ENTRANCES: int = 0 | ||
26 | PAINTING_EXIT_ROOMS: Set[str] = set() | ||
27 | PAINTING_EXITS: int = 0 | ||
28 | REQUIRED_PAINTING_ROOMS: List[str] = [] | ||
29 | REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = [] | ||
30 | |||
31 | SPECIAL_ITEM_IDS: Dict[str, int] = {} | ||
32 | PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {} | ||
33 | DOOR_LOCATION_IDS: Dict[str, Dict[str, int]] = {} | ||
34 | DOOR_ITEM_IDS: Dict[str, Dict[str, int]] = {} | ||
35 | DOOR_GROUP_ITEM_IDS: Dict[str, int] = {} | ||
36 | PROGRESSIVE_ITEM_IDS: Dict[str, int] = {} | ||
37 | |||
38 | |||
39 | def 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 | |||
50 | def 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 | |||
99 | def 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 | |||
136 | def 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 | |||
223 | def 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 | |||
316 | def 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 | |||
375 | def 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 | |||
395 | def 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 | |||
427 | if __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) | ||