about summary refs log tree commit diff stats
path: root/tools/validator
diff options
context:
space:
mode:
Diffstat (limited to 'tools/validator')
-rw-r--r--tools/validator/human_processor.cpp183
-rw-r--r--tools/validator/structs.h32
-rw-r--r--tools/validator/validator.cpp226
-rw-r--r--tools/validator/validator.h2
4 files changed, 423 insertions, 20 deletions
diff --git a/tools/validator/human_processor.cpp b/tools/validator/human_processor.cpp index 0f63936..ffa9765 100644 --- a/tools/validator/human_processor.cpp +++ b/tools/validator/human_processor.cpp
@@ -13,6 +13,7 @@
13#include <string> 13#include <string>
14 14
15#include "structs.h" 15#include "structs.h"
16#include "util/ids_yaml_format.h"
16 17
17namespace com::fourisland::lingo2_archipelago { 18namespace com::fourisland::lingo2_archipelago {
18namespace { 19namespace {
@@ -41,7 +42,9 @@ class HumanProcessor {
41 42
42 ProcessConnectionsFile(datadir_path / "connections.txtpb", std::nullopt); 43 ProcessConnectionsFile(datadir_path / "connections.txtpb", std::nullopt);
43 ProcessMaps(datadir_path); 44 ProcessMaps(datadir_path);
44 ProcessIdsFile(datadir_path / "ids.txtpb"); 45 ProcessProgressivesFile(datadir_path / "progressives.txtpb");
46 ProcessDoorGroupsFile(datadir_path / "door_groups.txtpb");
47 ProcessIdsFile(datadir_path / "ids.yaml");
45 } 48 }
46 49
47 private: 50 private:
@@ -74,6 +77,21 @@ class HumanProcessor {
74 for (const std::string& path : metadata.excluded_nodes()) { 77 for (const std::string& path : metadata.excluded_nodes()) {
75 map_info.game_nodes[path].uses++; 78 map_info.game_nodes[path].uses++;
76 } 79 }
80
81 for (const std::string& path : metadata.custom_nodes()) {
82 map_info.game_nodes[path].defined = true;
83 }
84
85 if (metadata.has_worldport_entrance()) {
86 auto port_identifier = GetCompletePortIdentifier(
87 metadata.worldport_entrance(), current_map_name, std::nullopt);
88 if (port_identifier) {
89 PortInfo& port_info = info_.ports[*port_identifier];
90 port_info.map_worldport_entrances.push_back(current_map_name);
91 } else {
92 map_info.malformed_worldport_entrance = metadata.worldport_entrance();
93 }
94 }
77 } 95 }
78 96
79 void ProcessRooms(std::filesystem::path path, 97 void ProcessRooms(std::filesystem::path path,
@@ -355,11 +373,6 @@ class HumanProcessor {
355 DoorInfo& other_door_info = info_.doors[complete_door_identifier]; 373 DoorInfo& other_door_info = info_.doors[complete_door_identifier];
356 other_door_info.doors_referenced_by.push_back(door_identifier); 374 other_door_info.doors_referenced_by.push_back(door_identifier);
357 } 375 }
358
359 for (const std::string& ei : h_door.endings()) {
360 EndingInfo& ending_info = info_.endings[ei];
361 ending_info.doors_referenced_by.push_back(door_identifier);
362 }
363 } 376 }
364 377
365 void ProcessConnectionsFile(std::filesystem::path path, 378 void ProcessConnectionsFile(std::filesystem::path path,
@@ -391,7 +404,9 @@ class HumanProcessor {
391 } 404 }
392 } else if (human_connection.has_from()) { 405 } else if (human_connection.has_from()) {
393 ProcessSingleConnection(human_connection, human_connection.from(), 406 ProcessSingleConnection(human_connection, human_connection.from(),
394 current_map_name); 407 current_map_name,
408 /*is_target=*/!human_connection.oneway() &&
409 !human_connection.bypass_target_door());
395 } 410 }
396 411
397 if (human_connection.has_to_room()) { 412 if (human_connection.has_to_room()) {
@@ -407,8 +422,9 @@ class HumanProcessor {
407 std::cout << "A global connection used to_room." << std::endl; 422 std::cout << "A global connection used to_room." << std::endl;
408 } 423 }
409 } else if (human_connection.has_to()) { 424 } else if (human_connection.has_to()) {
410 ProcessSingleConnection(human_connection, human_connection.to(), 425 ProcessSingleConnection(
411 current_map_name); 426 human_connection, human_connection.to(), current_map_name,
427 /*is_target=*/!human_connection.bypass_target_door());
412 } 428 }
413 429
414 if (human_connection.has_door()) { 430 if (human_connection.has_door()) {
@@ -429,7 +445,7 @@ class HumanProcessor {
429 void ProcessSingleConnection( 445 void ProcessSingleConnection(
430 const HumanConnection& human_connection, 446 const HumanConnection& human_connection,
431 const HumanConnection::Endpoint& endpoint, 447 const HumanConnection::Endpoint& endpoint,
432 const std::optional<std::string>& current_map_name) { 448 const std::optional<std::string>& current_map_name, bool is_target) {
433 if (endpoint.has_room()) { 449 if (endpoint.has_room()) {
434 auto room_identifier = 450 auto room_identifier =
435 GetCompleteRoomIdentifier(endpoint.room(), current_map_name); 451 GetCompleteRoomIdentifier(endpoint.room(), current_map_name);
@@ -448,6 +464,11 @@ class HumanProcessor {
448 if (painting_identifier) { 464 if (painting_identifier) {
449 PaintingInfo& painting_info = info_.paintings[*painting_identifier]; 465 PaintingInfo& painting_info = info_.paintings[*painting_identifier];
450 painting_info.connections_referenced_by.push_back(human_connection); 466 painting_info.connections_referenced_by.push_back(human_connection);
467
468 if (is_target) {
469 painting_info.target_connections_referenced_by.push_back(
470 human_connection);
471 }
451 } else { 472 } else {
452 // Not sure where else to store this right now. 473 // Not sure where else to store this right now.
453 std::cout 474 std::cout
@@ -460,6 +481,11 @@ class HumanProcessor {
460 if (port_identifier) { 481 if (port_identifier) {
461 PortInfo& port_info = info_.ports[*port_identifier]; 482 PortInfo& port_info = info_.ports[*port_identifier];
462 port_info.connections_referenced_by.push_back(human_connection); 483 port_info.connections_referenced_by.push_back(human_connection);
484
485 if (is_target) {
486 port_info.target_connections_referenced_by.push_back(
487 human_connection);
488 }
463 } else { 489 } else {
464 // Not sure where else to store this right now. 490 // Not sure where else to store this right now.
465 std::cout 491 std::cout
@@ -477,12 +503,147 @@ class HumanProcessor {
477 panel_info.proxies[endpoint.panel().answer()] 503 panel_info.proxies[endpoint.panel().answer()]
478 .connections_referenced_by.push_back(human_connection); 504 .connections_referenced_by.push_back(human_connection);
479 } 505 }
506
507 if (is_target) {
508 panel_info.target_connections_referenced_by.push_back(
509 human_connection);
510 }
511 }
512 }
513 }
514
515 void ProcessProgressivesFile(std::filesystem::path path) {
516 if (!std::filesystem::exists(path)) {
517 return;
518 }
519
520 auto h_progs = ReadMessageFromFile<HumanProgressives>(path.string());
521
522 for (const HumanProgressive& h_prog : h_progs.progressives()) {
523 ProcessProgressive(h_prog);
524 }
525 }
526
527 void ProcessProgressive(const HumanProgressive& h_prog) {
528 ProgressiveInfo& prog_info = info_.progressives[h_prog.name()];
529 prog_info.definitions.push_back(h_prog);
530
531 for (const DoorIdentifier& di : h_prog.doors()) {
532 if (!di.has_map()) {
533 prog_info.malformed_doors.push_back(di);
534 continue;
535 }
536
537 DoorInfo& door_info = info_.doors[di];
538 door_info.progressives_referenced_by.push_back(h_prog.name());
539 }
540 }
541
542 void ProcessDoorGroupsFile(std::filesystem::path path) {
543 if (!std::filesystem::exists(path)) {
544 return;
545 }
546
547 auto h_groups = ReadMessageFromFile<HumanDoorGroups>(path.string());
548
549 for (const HumanDoorGroup& h_group : h_groups.door_groups()) {
550 ProcessDoorGroup(h_group);
551 }
552 }
553
554 void ProcessDoorGroup(const HumanDoorGroup& h_group) {
555 DoorGroupInfo& group_info = info_.door_groups[h_group.name()];
556 group_info.definitions.push_back(h_group);
557
558 for (const DoorIdentifier& di : h_group.doors()) {
559 if (!di.has_map()) {
560 group_info.malformed_doors.push_back(di);
561 continue;
480 } 562 }
563
564 DoorInfo& door_info = info_.doors[di];
565 door_info.door_groups_referenced_by.push_back(h_group.name());
481 } 566 }
482 } 567 }
483 568
484 void ProcessIdsFile(std::filesystem::path path) { 569 void ProcessIdsFile(std::filesystem::path path) {
485 // Ignore this for now. 570 auto ids = ReadIdsFromYaml(path.string());
571
572 DoorIdentifier di;
573 PanelIdentifier pai;
574 PortIdentifier poi;
575 KeyholderIdentifier ki;
576
577 for (const auto& [map_name, map] : ids.maps()) {
578 di.set_map(map_name);
579 pai.set_map(map_name);
580 poi.set_map(map_name);
581 ki.set_map(map_name);
582
583 for (const auto& [door_name, ap_id] : map.doors()) {
584 di.set_name(door_name);
585
586 DoorInfo& door_info = info_.doors[di];
587 door_info.has_id = true;
588 }
589
590 for (const auto& [room_name, room] : map.rooms()) {
591 pai.set_room(room_name);
592 poi.set_room(room_name);
593 ki.set_room(room_name);
594
595 for (const auto& [panel_name, ap_id] : room.panels()) {
596 pai.set_name(panel_name);
597
598 PanelInfo& panel_info = info_.panels[pai];
599 panel_info.has_id = true;
600 }
601
602 for (const auto& [mastery_name, ap_id] : room.masteries()) {
603 // TODO: Mastery
604 }
605
606 for (const auto& [keyholder_name, ap_id] : room.keyholders()) {
607 ki.set_name(keyholder_name);
608
609 KeyholderInfo& keyholder_info = info_.keyholders[ki];
610 keyholder_info.has_id = true;
611 }
612
613 for (const auto& [port_name, ap_id] : room.ports()) {
614 poi.set_name(port_name);
615
616 PortInfo& port_info = info_.ports[poi];
617 port_info.has_id = true;
618 }
619 }
620 }
621
622 for (const auto& [tag, id] : ids.special()) {
623 // TODO: Specials
624 }
625
626 for (const auto& [letter_name, ap_id] : ids.letters()) {
627 LetterIdentifier li =
628 std::make_tuple(letter_name[0], letter_name[1] == '2');
629 LetterInfo& letter_info = info_.letters[li];
630 letter_info.has_id = true;
631 }
632
633 for (const auto& [ending_name, ap_id] : ids.endings()) {
634 EndingInfo& ending_info = info_.endings[ending_name];
635 ending_info.has_id = true;
636 }
637
638 for (const auto& [prog_name, ap_id] : ids.progressives()) {
639 ProgressiveInfo& prog_info = info_.progressives[prog_name];
640 prog_info.has_id = true;
641 }
642
643 for (const auto& [group_name, ap_id] : ids.door_groups()) {
644 DoorGroupInfo& group_info = info_.door_groups[group_name];
645 group_info.has_id = true;
646 }
486 } 647 }
487 648
488 std::string mapdir_; 649 std::string mapdir_;
diff --git a/tools/validator/structs.h b/tools/validator/structs.h index 0ca96fe..62974a8 100644 --- a/tools/validator/structs.h +++ b/tools/validator/structs.h
@@ -27,6 +27,8 @@ struct GameNodeInfo {
27 27
28struct MapInfo { 28struct MapInfo {
29 std::map<std::string, GameNodeInfo> game_nodes; 29 std::map<std::string, GameNodeInfo> game_nodes;
30
31 std::optional<PortIdentifier> malformed_worldport_entrance;
30}; 32};
31 33
32struct RoomInfo { 34struct RoomInfo {
@@ -39,26 +41,33 @@ struct RoomInfo {
39 41
40struct DoorInfo { 42struct DoorInfo {
41 std::vector<HumanDoor> definitions; 43 std::vector<HumanDoor> definitions;
44 bool has_id = false;
42 45
43 std::vector<HumanConnection> connections_referenced_by; 46 std::vector<HumanConnection> connections_referenced_by;
44 std::vector<DoorIdentifier> doors_referenced_by; 47 std::vector<DoorIdentifier> doors_referenced_by;
45 std::vector<PanelIdentifier> panels_referenced_by; 48 std::vector<PanelIdentifier> panels_referenced_by;
46 std::vector<PaintingIdentifier> paintings_referenced_by; 49 std::vector<PaintingIdentifier> paintings_referenced_by;
47 std::vector<PortIdentifier> ports_referenced_by; 50 std::vector<PortIdentifier> ports_referenced_by;
51 std::vector<std::string> progressives_referenced_by;
52 std::vector<std::string> door_groups_referenced_by;
48 53
49 MalformedIdentifiers malformed_identifiers; 54 MalformedIdentifiers malformed_identifiers;
50}; 55};
51 56
52struct PortInfo { 57struct PortInfo {
53 std::vector<HumanPort> definitions; 58 std::vector<HumanPort> definitions;
59 bool has_id = false;
54 60
55 std::vector<HumanConnection> connections_referenced_by; 61 std::vector<HumanConnection> connections_referenced_by;
62 std::vector<HumanConnection> target_connections_referenced_by;
63 std::vector<std::string> map_worldport_entrances;
56}; 64};
57 65
58struct PaintingInfo { 66struct PaintingInfo {
59 std::vector<HumanPainting> definitions; 67 std::vector<HumanPainting> definitions;
60 68
61 std::vector<HumanConnection> connections_referenced_by; 69 std::vector<HumanConnection> connections_referenced_by;
70 std::vector<HumanConnection> target_connections_referenced_by;
62 std::vector<DoorIdentifier> doors_referenced_by; 71 std::vector<DoorIdentifier> doors_referenced_by;
63}; 72};
64 73
@@ -71,10 +80,12 @@ struct ProxyInfo {
71 80
72struct PanelInfo { 81struct PanelInfo {
73 std::vector<HumanPanel> definitions; 82 std::vector<HumanPanel> definitions;
83 bool has_id = false;
74 84
75 std::string map_area_name; 85 std::string map_area_name;
76 86
77 std::vector<HumanConnection> connections_referenced_by; 87 std::vector<HumanConnection> connections_referenced_by;
88 std::vector<HumanConnection> target_connections_referenced_by;
78 std::vector<DoorIdentifier> doors_referenced_by; 89 std::vector<DoorIdentifier> doors_referenced_by;
79 90
80 std::map<std::string, ProxyInfo> proxies; 91 std::map<std::string, ProxyInfo> proxies;
@@ -82,6 +93,7 @@ struct PanelInfo {
82 93
83struct KeyholderInfo { 94struct KeyholderInfo {
84 std::vector<HumanKeyholder> definitions; 95 std::vector<HumanKeyholder> definitions;
96 bool has_id = false;
85 97
86 std::vector<DoorIdentifier> doors_referenced_by; 98 std::vector<DoorIdentifier> doors_referenced_by;
87}; 99};
@@ -90,18 +102,32 @@ using LetterIdentifier = std::tuple<char, bool>;
90 102
91struct LetterInfo { 103struct LetterInfo {
92 std::vector<RoomIdentifier> defined_in; 104 std::vector<RoomIdentifier> defined_in;
105 bool has_id = false;
93}; 106};
94 107
95struct EndingInfo { 108struct EndingInfo {
96 std::vector<RoomIdentifier> defined_in; 109 std::vector<RoomIdentifier> defined_in;
97 110 bool has_id = false;
98 std::vector<DoorIdentifier> doors_referenced_by;
99}; 111};
100 112
101struct PanelNameInfo { 113struct PanelNameInfo {
102 std::vector<PanelIdentifier> panels_used_by; 114 std::vector<PanelIdentifier> panels_used_by;
103}; 115};
104 116
117struct ProgressiveInfo {
118 std::vector<HumanProgressive> definitions;
119 bool has_id = false;
120
121 std::vector<DoorIdentifier> malformed_doors;
122};
123
124struct DoorGroupInfo {
125 std::vector<HumanDoorGroup> definitions;
126 bool has_id = false;
127
128 std::vector<DoorIdentifier> malformed_doors;
129};
130
105struct CollectedInfo { 131struct CollectedInfo {
106 std::map<std::string, MapInfo> maps; 132 std::map<std::string, MapInfo> maps;
107 std::map<RoomIdentifier, RoomInfo, RoomIdentifierLess> rooms; 133 std::map<RoomIdentifier, RoomInfo, RoomIdentifierLess> rooms;
@@ -114,6 +140,8 @@ struct CollectedInfo {
114 std::map<LetterIdentifier, LetterInfo> letters; 140 std::map<LetterIdentifier, LetterInfo> letters;
115 std::map<std::string, EndingInfo> endings; 141 std::map<std::string, EndingInfo> endings;
116 std::map<std::string, PanelNameInfo> panel_names; 142 std::map<std::string, PanelNameInfo> panel_names;
143 std::map<std::string, ProgressiveInfo> progressives;
144 std::map<std::string, DoorGroupInfo> door_groups;
117}; 145};
118 146
119} // namespace com::fourisland::lingo2_archipelago 147} // namespace com::fourisland::lingo2_archipelago
diff --git a/tools/validator/validator.cpp b/tools/validator/validator.cpp index f5524c3..fe36be7 100644 --- a/tools/validator/validator.cpp +++ b/tools/validator/validator.cpp
@@ -45,6 +45,12 @@ class Validator {
45 for (const auto& [panel_name, panel_info] : info_.panel_names) { 45 for (const auto& [panel_name, panel_info] : info_.panel_names) {
46 ValidatePanelName(panel_name, panel_info); 46 ValidatePanelName(panel_name, panel_info);
47 } 47 }
48 for (const auto& [prog_name, prog_info] : info_.progressives) {
49 ValidateProgressive(prog_name, prog_info);
50 }
51 for (const auto& [group_name, group_info] : info_.door_groups) {
52 ValidateDoorGroup(group_name, group_info);
53 }
48 } 54 }
49 55
50 private: 56 private:
@@ -63,6 +69,11 @@ class Validator {
63 << " is not defined in the game file." << std::endl; 69 << " is not defined in the game file." << std::endl;
64 } 70 }
65 } 71 }
72
73 if (map_info.malformed_worldport_entrance) {
74 std::cout << "The worldport entrance for map " << map_name
75 << " is malformed." << std::endl;
76 }
66 } 77 }
67 78
68 void ValidateRoom(const RoomIdentifier& room_identifier, 79 void ValidateRoom(const RoomIdentifier& room_identifier,
@@ -100,7 +111,8 @@ class Validator {
100 return false; 111 return false;
101 } 112 }
102 113
103 if (h_door.keyholders_size() > 0 || h_door.endings_size() > 0) { 114 if (h_door.keyholders_size() > 0 || h_door.white_ending() ||
115 h_door.complete_at() > 0) {
104 return true; 116 return true;
105 } 117 }
106 118
@@ -164,6 +176,20 @@ class Validator {
164 std::cout << " CONNECTION " << connection.ShortDebugString() 176 std::cout << " CONNECTION " << connection.ShortDebugString()
165 << std::endl; 177 << std::endl;
166 } 178 }
179
180 for (const std::string& prog_name :
181 door_info.progressives_referenced_by) {
182 std::cout << " PROGRESSIVE " << prog_name << std::endl;
183 }
184
185 for (const std::string& group_name :
186 door_info.door_groups_referenced_by) {
187 std::cout << " DOOR GROUP " << group_name << std::endl;
188 }
189
190 if (door_info.has_id) {
191 std::cout << " An AP ID is present." << std::endl;
192 }
167 } else if (door_info.definitions.size() > 1) { 193 } else if (door_info.definitions.size() > 1) {
168 std::cout << "Door " << door_identifier.ShortDebugString() 194 std::cout << "Door " << door_identifier.ShortDebugString()
169 << " was defined multiple times." << std::endl; 195 << " was defined multiple times." << std::endl;
@@ -198,6 +224,33 @@ class Validator {
198 std::cout << "Door " << door_identifier.ShortDebugString() 224 std::cout << "Door " << door_identifier.ShortDebugString()
199 << " needs an explicit location name." << std::endl; 225 << " needs an explicit location name." << std::endl;
200 } 226 }
227
228 if (h_door.type() == DoorType::STANDARD ||
229 h_door.type() == DoorType::LOCATION_ONLY ||
230 h_door.type() == DoorType::GRAVESTONE || h_door.legacy_location()) {
231 if (h_door.double_letters()) {
232 std::cout << "Door " << door_identifier.ShortDebugString()
233 << " is a location that depends on double_letters."
234 << std::endl;
235 }
236
237 if (!h_door.has_location_room()) {
238 std::cout << "Door " << door_identifier.ShortDebugString()
239 << " is missing a location_room." << std::endl;
240 }
241 }
242
243 bool needs_id = (h_door.type() != DoorType::EVENT || h_door.latch() ||
244 h_door.legacy_location());
245 if (door_info.has_id != needs_id) {
246 if (needs_id) {
247 std::cout << "Door " << door_identifier.ShortDebugString()
248 << " is missing an AP ID." << std::endl;
249 } else {
250 std::cout << "Door " << door_identifier.ShortDebugString()
251 << " should not have an AP ID." << std::endl;
252 }
253 }
201 } 254 }
202 } 255 }
203 256
@@ -212,10 +265,57 @@ class Validator {
212 std::cout << " CONNECTION " << connection.ShortDebugString() 265 std::cout << " CONNECTION " << connection.ShortDebugString()
213 << std::endl; 266 << std::endl;
214 } 267 }
268
269 for (const std::string& map_name : port_info.map_worldport_entrances) {
270 std::cout << " MAP (worldport_entrance) " << map_name << std::endl;
271 }
272
273 if (port_info.has_id) {
274 std::cout << " An AP ID is present." << std::endl;
275 }
215 } else if (port_info.definitions.size() > 1) { 276 } else if (port_info.definitions.size() > 1) {
216 std::cout << "Port " << port_identifier.ShortDebugString() 277 std::cout << "Port " << port_identifier.ShortDebugString()
217 << " was defined multiple times." << std::endl; 278 << " was defined multiple times." << std::endl;
218 } 279 }
280
281 if (!port_info.target_connections_referenced_by.empty()) {
282 for (const HumanPort& port : port_info.definitions) {
283 if (port.has_required_door()) {
284 std::cout << "Port " << port_identifier.ShortDebugString()
285 << " has a required door but is the target of a connection:"
286 << std::endl;
287
288 for (const HumanConnection& connection :
289 port_info.target_connections_referenced_by) {
290 std::cout << " CONNECTION " << connection.ShortDebugString()
291 << std::endl;
292 }
293 }
294 }
295 }
296
297 for (const HumanPort& port : port_info.definitions) {
298 if (!port.no_shuffle()) {
299 if (!port.has_destination()) {
300 std::cout << "Port " << port_identifier.ShortDebugString()
301 << " is shuffleable and missing a destination."
302 << std::endl;
303 }
304 if (!port.has_rotation()) {
305 std::cout << "Port " << port_identifier.ShortDebugString()
306 << " is shuffleable and missing a rotation." << std::endl;
307 }
308 if (!port_info.has_id) {
309 std::cout << "Port " << port_identifier.ShortDebugString()
310 << " is missing an AP ID." << std::endl;
311 }
312 } else {
313 if (port_info.has_id) {
314 std::cout << "Port " << port_identifier.ShortDebugString()
315 << " should not have an AP ID." << std::endl;
316 }
317 }
318 }
219 } 319 }
220 320
221 void ValidatePainting(const PaintingIdentifier& painting_identifier, 321 void ValidatePainting(const PaintingIdentifier& painting_identifier,
@@ -239,6 +339,22 @@ class Validator {
239 std::cout << "Painting " << painting_identifier.ShortDebugString() 339 std::cout << "Painting " << painting_identifier.ShortDebugString()
240 << " was defined multiple times." << std::endl; 340 << " was defined multiple times." << std::endl;
241 } 341 }
342
343 if (!painting_info.target_connections_referenced_by.empty()) {
344 for (const HumanPainting& painting : painting_info.definitions) {
345 if (painting.has_required_door()) {
346 std::cout << "Painting " << painting_identifier.ShortDebugString()
347 << " has a required door but is the target of a connection:"
348 << std::endl;
349
350 for (const HumanConnection& connection :
351 painting_info.target_connections_referenced_by) {
352 std::cout << " CONNECTION " << connection.ShortDebugString()
353 << std::endl;
354 }
355 }
356 }
357 }
242 } 358 }
243 359
244 void ValidatePanel(const PanelIdentifier& panel_identifier, 360 void ValidatePanel(const PanelIdentifier& panel_identifier,
@@ -263,6 +379,10 @@ class Validator {
263 std::cout << " CONNECTION " << connection.ShortDebugString() 379 std::cout << " CONNECTION " << connection.ShortDebugString()
264 << std::endl; 380 << std::endl;
265 } 381 }
382
383 if (panel_info.has_id) {
384 std::cout << " An AP ID is present." << std::endl;
385 }
266 } else if (panel_info.definitions.size() > 1) { 386 } else if (panel_info.definitions.size() > 1) {
267 std::cout << "Panel " << panel_identifier.ShortDebugString() 387 std::cout << "Panel " << panel_identifier.ShortDebugString()
268 << " was defined multiple times." << std::endl; 388 << " was defined multiple times." << std::endl;
@@ -291,6 +411,27 @@ class Validator {
291 << "\" was defined multiple times." << std::endl; 411 << "\" was defined multiple times." << std::endl;
292 } 412 }
293 } 413 }
414
415 if (!panel_info.has_id) {
416 std::cout << "Panel " << panel_identifier.ShortDebugString()
417 << " is missing an AP ID." << std::endl;
418 }
419
420 if (!panel_info.target_connections_referenced_by.empty()) {
421 for (const HumanPanel& panel : panel_info.definitions) {
422 if (panel.has_required_door()) {
423 std::cout << "Panel " << panel_identifier.ShortDebugString()
424 << " has a required door but is the target of a connection:"
425 << std::endl;
426
427 for (const HumanConnection& connection :
428 panel_info.target_connections_referenced_by) {
429 std::cout << " CONNECTION " << connection.ShortDebugString()
430 << std::endl;
431 }
432 }
433 }
434 }
294 } 435 }
295 436
296 void ValidateKeyholder(const KeyholderIdentifier& keyholder_identifier, 437 void ValidateKeyholder(const KeyholderIdentifier& keyholder_identifier,
@@ -304,10 +445,28 @@ class Validator {
304 std::cout << " DOOR " << door_identifier.ShortDebugString() 445 std::cout << " DOOR " << door_identifier.ShortDebugString()
305 << std::endl; 446 << std::endl;
306 } 447 }
448
449 if (keyholder_info.has_id) {
450 std::cout << " An AP ID is present." << std::endl;
451 }
307 } else if (keyholder_info.definitions.size() > 1) { 452 } else if (keyholder_info.definitions.size() > 1) {
308 std::cout << "Keyholder " << keyholder_identifier.ShortDebugString() 453 std::cout << "Keyholder " << keyholder_identifier.ShortDebugString()
309 << " was defined multiple times." << std::endl; 454 << " was defined multiple times." << std::endl;
310 } 455 }
456
457 for (const HumanKeyholder& h_keyholder : keyholder_info.definitions) {
458 bool needs_id = (h_keyholder.has_key());
459
460 if (needs_id != keyholder_info.has_id) {
461 if (needs_id) {
462 std::cout << "Keyholder " << keyholder_identifier.ShortDebugString()
463 << " is missing an AP ID." << std::endl;
464 } else {
465 std::cout << "Keyholder " << keyholder_identifier.ShortDebugString()
466 << " should not have an AP ID." << std::endl;
467 }
468 }
469 }
311 } 470 }
312 471
313 void ValidateLetter(const LetterIdentifier& letter_identifier, 472 void ValidateLetter(const LetterIdentifier& letter_identifier,
@@ -315,7 +474,14 @@ class Validator {
315 std::string letter_name = std::string(1, std::get<0>(letter_identifier)) + 474 std::string letter_name = std::string(1, std::get<0>(letter_identifier)) +
316 (std::get<1>(letter_identifier) ? "2" : "1"); 475 (std::get<1>(letter_identifier) ? "2" : "1");
317 476
318 if (letter_info.defined_in.size() > 1) { 477 if (letter_info.defined_in.empty()) {
478 std::cout << "Letter " << letter_name
479 << " has no definition, but was referenced:" << std::endl;
480
481 if (letter_info.has_id) {
482 std::cout << " An AP ID is present." << std::endl;
483 }
484 } else if (letter_info.defined_in.size() > 1) {
319 std::cout << "Letter " << letter_name 485 std::cout << "Letter " << letter_name
320 << " was defined in multiple places:" << std::endl; 486 << " was defined in multiple places:" << std::endl;
321 487
@@ -323,6 +489,11 @@ class Validator {
323 std::cout << " " << room_identifier.ShortDebugString() << std::endl; 489 std::cout << " " << room_identifier.ShortDebugString() << std::endl;
324 } 490 }
325 } 491 }
492
493 if (!letter_info.has_id) {
494 std::cout << "Letter " << letter_name << " is missing an AP ID."
495 << std::endl;
496 }
326 } 497 }
327 498
328 void ValidateEnding(const std::string& ending_name, 499 void ValidateEnding(const std::string& ending_name,
@@ -331,10 +502,8 @@ class Validator {
331 std::cout << "Ending " << ending_name 502 std::cout << "Ending " << ending_name
332 << " has no definition, but was referenced:" << std::endl; 503 << " has no definition, but was referenced:" << std::endl;
333 504
334 for (const DoorIdentifier& door_identifier : 505 if (ending_info.has_id) {
335 ending_info.doors_referenced_by) { 506 std::cout << " An AP ID is present." << std::endl;
336 std::cout << " DOOR " << door_identifier.ShortDebugString()
337 << std::endl;
338 } 507 }
339 } else if (ending_info.defined_in.size() > 1) { 508 } else if (ending_info.defined_in.size() > 1) {
340 std::cout << "Ending " << ending_name 509 std::cout << "Ending " << ending_name
@@ -344,6 +513,11 @@ class Validator {
344 std::cout << " " << room_identifier.ShortDebugString() << std::endl; 513 std::cout << " " << room_identifier.ShortDebugString() << std::endl;
345 } 514 }
346 } 515 }
516
517 if (!ending_info.has_id) {
518 std::cout << "Ending " << ending_name << " is missing an AP ID."
519 << std::endl;
520 }
347 } 521 }
348 522
349 void ValidatePanelName(const std::string& panel_name, 523 void ValidatePanelName(const std::string& panel_name,
@@ -360,6 +534,46 @@ class Validator {
360 } 534 }
361 } 535 }
362 536
537 void ValidateProgressive(const std::string& prog_name,
538 const ProgressiveInfo& prog_info) const {
539 if (prog_info.definitions.empty()) {
540 std::cout << "Progressive \"" << prog_name
541 << "\" has no definition, but was referenced:" << std::endl;
542
543 if (prog_info.has_id) {
544 std::cout << " An AP ID is present." << std::endl;
545 }
546 } else if (prog_info.definitions.size() > 1) {
547 std::cout << "Progressive \"" << prog_name
548 << "\" has multiple definitions." << std::endl;
549 }
550
551 if (!prog_info.has_id) {
552 std::cout << "Progressive \"" << prog_name << "\" is missing an AP ID."
553 << std::endl;
554 }
555 }
556
557 void ValidateDoorGroup(const std::string& group_name,
558 const DoorGroupInfo& group_info) const {
559 if (group_info.definitions.empty()) {
560 std::cout << "Door group \"" << group_name
561 << "\" has no definition, but was referenced:" << std::endl;
562
563 if (group_info.has_id) {
564 std::cout << " An AP ID is present." << std::endl;
565 }
566 } else if (group_info.definitions.size() > 1) {
567 std::cout << "Door group \"" << group_name
568 << "\" has multiple definitions." << std::endl;
569 }
570
571 if (!group_info.has_id) {
572 std::cout << "Door group \"" << group_name << "\" is missing an AP ID."
573 << std::endl;
574 }
575 }
576
363 const CollectedInfo& info_; 577 const CollectedInfo& info_;
364}; 578};
365 579
diff --git a/tools/validator/validator.h b/tools/validator/validator.h index b710429..33bc7c9 100644 --- a/tools/validator/validator.h +++ b/tools/validator/validator.h
@@ -1,4 +1,4 @@
1#ifndef TOOLS_VALIDATOR_VALIDATOR_H_ 1#ifndef TOOLS_VALIDATOR_VALIDATOR_H
2#define TOOLS_VALIDATOR_VALIDATOR_H 2#define TOOLS_VALIDATOR_VALIDATOR_H
3 3
4namespace com::fourisland::lingo2_archipelago { 4namespace com::fourisland::lingo2_archipelago {