1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
|
# Script to validate a level config file. This checks that the names used within
# the file are consistent. It also checks that the panel and door IDs mentioned
# all exist in the map file.
#
# Usage: validate_config.rb [config file] [map file]
require 'set'
require 'yaml'
configpath = ARGV[0]
idspath = ARGV[1]
mappath = ARGV[2]
panels = Set["Countdown Panels/Panel_1234567890_wanderlust"]
doors = Set["Naps Room Doors/Door_hider_new1", "Tower Room Area Doors/Door_wanderer_entrance"]
paintings = Set[]
File.readlines(mappath).each do |line|
line.match(/node name=\"(.*)\" parent=\"Panels\/(.*)\" instance/) do |m|
panels.add(m[2] + "/" + m[1])
end
line.match(/node name=\"(.*)\" parent=\"Doors\/(.*)\" instance/) do |m|
doors.add(m[2] + "/" + m[1])
end
line.match(/node name=\"(.*)\" parent=\"Decorations\/Paintings\" instance/) do |m|
paintings.add(m[1])
end
line.match(/node name=\"(.*)\" parent=\"Decorations\/EndPanel\" instance/) do |m|
panels.add("EndPanel/" + m[1])
end
end
configured_rooms = Set["Menu"]
configured_doors = Set[]
configured_panels = Set[]
configured_panel_doors = Set[]
mentioned_rooms = Set[]
mentioned_doors = Set[]
mentioned_panels = Set[]
mentioned_panel_doors = Set[]
mentioned_sunwarp_entrances = Set[]
mentioned_sunwarp_exits = Set[]
mentioned_paintings = Set[]
door_groups = {}
panel_groups = {}
directives = Set["entrances", "panels", "doors", "panel_doors", "paintings", "sunwarps", "progression"]
panel_directives = Set["id", "required_room", "required_door", "required_panel", "colors", "check", "exclude_reduce", "tag", "link", "subtag", "achievement", "copy_to_sign", "non_counting", "hunt", "location_name"]
door_directives = Set["id", "painting_id", "panels", "item_name", "item_group", "location_name", "skip_location", "skip_item", "door_group", "include_reduce", "event", "warp_id"]
panel_door_directives = Set["panels", "item_name", "panel_group"]
painting_directives = Set["id", "display_name", "enter_only", "exit_only", "orientation", "required_door", "required", "required_when_no_doors", "move", "req_blocked", "req_blocked_when_no_doors"]
non_counting = 0
ids = YAML.load_file(idspath)
config = YAML.load_file(configpath)
config.each do |room_name, room|
configured_rooms.add(room_name)
used_directives = Set[]
room.each_key do |key|
used_directives.add(key)
end
diff_directives = used_directives - directives
unless diff_directives.empty? then
puts("#{room_name} has the following invalid top-level directives: #{diff_directives.to_s}")
end
(room["entrances"] || {}).each do |source_room, entrance|
mentioned_rooms.add(source_room)
entrances = []
if entrance.kind_of? Hash
entrances = [entrance]
elsif entrance.kind_of? Array
entrances = entrance
end
entrances.each do |e|
if e.include?("door") then
entrance_room = e.include?("room") ? e["room"] : room_name
mentioned_rooms.add(entrance_room)
mentioned_doors.add(entrance_room + " - " + e["door"])
end
end
end
(room["panels"] || {}).each do |panel_name, panel|
unless panel_name.kind_of? String then
puts "#{room_name} has an invalid panel name"
end
configured_panels.add(room_name + " - " + panel_name)
if panel.include?("id")
panel_ids = []
if panel["id"].kind_of? Array
panel_ids = panel["id"]
else
panel_ids = [panel["id"]]
end
panel_ids.each do |panel_id|
unless panels.include? panel_id then
puts "#{room_name} - #{panel_name} :::: Invalid Panel ID #{panel_id}"
end
end
else
puts "#{room_name} - #{panel_name} :::: Panel is missing an ID"
end
if panel.include?("required_room")
required_rooms = []
if panel["required_room"].kind_of? Array
required_rooms = panel["required_room"]
else
required_rooms = [panel["required_room"]]
end
required_rooms.each do |required_room|
mentioned_rooms.add(required_room)
end
end
if panel.include?("required_door")
required_doors = []
if panel["required_door"].kind_of? Array
required_doors = panel["required_door"]
else
required_doors = [panel["required_door"]]
end
required_doors.each do |required_door|
other_room = required_door.include?("room") ? required_door["room"] : room_name
mentioned_rooms.add(other_room)
mentioned_doors.add("#{other_room} - #{required_door["door"]}")
end
end
if panel.include?("required_panel")
required_panels = []
if panel["required_panel"].kind_of? Array
required_panels = panel["required_panel"]
else
required_panels = [panel["required_panel"]]
end
required_panels.each do |required_panel|
other_room = required_panel.include?("room") ? required_panel["room"] : room_name
mentioned_rooms.add(other_room)
mentioned_panels.add("#{other_room} - #{required_panel["panel"]}")
end
end
unless panel.include?("tag") then
puts "#{room_name} - #{panel_name} :::: Panel is missing a tag"
end
if panel.include?("non_counting") then
non_counting += 1
end
bad_subdirectives = []
panel.keys.each do |key|
unless panel_directives.include?(key) then
bad_subdirectives << key
end
end
unless bad_subdirectives.empty? then
puts "#{room_name} - #{panel_name} :::: Panel has the following invalid subdirectives: #{bad_subdirectives.join(", ")}"
end
unless ids.include?("panels") and ids["panels"].include?(room_name) and ids["panels"][room_name].include?(panel_name)
puts "#{room_name} - #{panel_name} :::: Panel is missing a location ID"
end
end
(room["doors"] || {}).each do |door_name, door|
configured_doors.add("#{room_name} - #{door_name}")
if door.include?("id")
door_ids = []
if door["id"].kind_of? Array
door_ids = door["id"]
else
door_ids = [door["id"]]
end
door_ids.each do |door_id|
unless doors.include? door_id then
puts "#{room_name} - #{door_name} :::: Invalid Door ID #{door_id}"
end
end
end
if door.include?("painting_id")
painting_ids = []
if door["painting_id"].kind_of? Array
painting_ids = door["painting_id"]
else
painting_ids = [door["painting_id"]]
end
painting_ids.each do |painting_id|
unless paintings.include? painting_id then
puts "#{room_name} - #{door_name} :::: Invalid Painting ID #{painting_id}"
end
end
end
if not door.include?("id") and not door.include?("painting_id") and not door.include?("warp_id") and not door["skip_item"] and not door["event"] then
puts "#{room_name} - #{door_name} :::: Should be marked skip_item or event if there are no doors, paintings, or warps"
end
if door.include?("panels")
door["panels"].each do |panel|
if panel.kind_of? Hash then
other_room = panel.include?("room") ? panel["room"] : room_name
mentioned_panels.add("#{other_room} - #{panel["panel"]}")
else
other_room = panel.include?("room") ? panel["room"] : room_name
mentioned_panels.add("#{room_name} - #{panel}")
end
end
elsif not door["skip_location"]
puts "#{room_name} - #{door_name} :::: Should be marked skip_location if there are no panels"
end
if door.include?("group")
door_groups[door["group"]] ||= 0
door_groups[door["group"]] += 1
end
bad_subdirectives = []
door.keys.each do |key|
unless door_directives.include?(key) then
bad_subdirectives << key
end
end
unless bad_subdirectives.empty? then
puts "#{room_name} - #{door_name} :::: Door has the following invalid subdirectives: #{bad_subdirectives.join(", ")}"
end
unless door["skip_item"] or door["event"]
unless ids.include?("doors") and ids["doors"].include?(room_name) and ids["doors"][room_name].include?(door_name) and ids["doors"][room_name][door_name].include?("item")
puts "#{room_name} - #{door_name} :::: Door is missing an item ID"
end
end
unless door["skip_location"] or door["event"]
unless ids.include?("doors") and ids["doors"].include?(room_name) and ids["doors"][room_name].include?(door_name) and ids["doors"][room_name][door_name].include?("location")
puts "#{room_name} - #{door_name} :::: Door is missing a location ID"
end
end
end
(room["panel_doors"] || {}).each do |panel_door_name, panel_door|
configured_panel_doors.add("#{room_name} - #{panel_door_name}")
if panel_door.include?("panels")
panel_door["panels"].each do |panel|
if panel.kind_of? Hash then
other_room = panel.include?("room") ? panel["room"] : room_name
mentioned_panels.add("#{other_room} - #{panel["panel"]}")
else
other_room = panel.include?("room") ? panel["room"] : room_name
mentioned_panels.add("#{room_name} - #{panel}")
end
end
else
puts "#{room_name} - #{panel_door_name} :::: Missing panels field"
end
if panel_door.include?("panel_group")
panel_groups[panel_door["panel_group"]] ||= 0
panel_groups[panel_door["panel_group"]] += 1
end
bad_subdirectives = []
panel_door.keys.each do |key|
unless panel_door_directives.include?(key) then
bad_subdirectives << key
end
end
unless bad_subdirectives.empty? then
puts "#{room_name} - #{panel_door_name} :::: Panel door has the following invalid subdirectives: #{bad_subdirectives.join(", ")}"
end
unless ids.include?("panel_doors") and ids["panel_doors"].include?(room_name) and ids["panel_doors"][room_name].include?(panel_door_name)
puts "#{room_name} - #{panel_door_name} :::: Panel door is missing an item ID"
end
end
(room["paintings"] || []).each do |painting|
if painting.include?("id") and painting["id"].kind_of? String then
unless paintings.include? painting["id"] then
puts "#{room_name} :::: Invalid Painting ID #{painting["id"]}"
end
if mentioned_paintings.include?(painting["id"]) then
puts "Painting #{painting["id"]} is mentioned more than once"
else
mentioned_paintings.add(painting["id"])
end
else
puts "#{room_name} :::: Painting is missing an ID"
end
if painting["disable"] then
# We're good.
next
end
unless painting.include? "display_name" then
puts "#{room_name} - #{painting["id"] || "painting"} :::: Missing display name"
end
if painting.include?("orientation") then
unless ["north", "south", "east", "west"].include? painting["orientation"] then
puts "#{room_name} - #{painting["id"] || "painting"} :::: Invalid orientation #{painting["orientation"]}"
end
else
puts "#{room_name} :::: Painting is missing an orientation"
end
if painting.include?("required_door")
other_room = painting["required_door"].include?("room") ? painting["required_door"]["room"] : room_name
mentioned_doors.add("#{other_room} - #{painting["required_door"]["door"]}")
unless painting["enter_only"] then
puts "#{room_name} - #{painting["id"] || "painting"} :::: Should be marked enter_only if there is a required_door"
end
end
bad_subdirectives = []
painting.keys.each do |key|
unless painting_directives.include?(key) then
bad_subdirectives << key
end
end
unless bad_subdirectives.empty? then
puts "#{room_name} - #{painting["id"] || "painting"} :::: Painting has the following invalid subdirectives: #{bad_subdirectives.join(", ")}"
end
end
(room["sunwarps"] || []).each do |sunwarp|
if sunwarp.include? "dots" and sunwarp.include? "direction" then
if sunwarp["dots"] < 1 or sunwarp["dots"] > 6 then
puts "#{room_name} :::: Contains a sunwarp with an invalid dots value"
end
if sunwarp["direction"] == "enter" then
if mentioned_sunwarp_entrances.include? sunwarp["dots"] then
puts "Multiple #{sunwarp["dots"]} sunwarp entrances were found"
else
mentioned_sunwarp_entrances.add(sunwarp["dots"])
end
elsif sunwarp["direction"] == "exit" then
if mentioned_sunwarp_exits.include? sunwarp["dots"] then
puts "Multiple #{sunwarp["dots"]} sunwarp exits were found"
else
mentioned_sunwarp_exits.add(sunwarp["dots"])
end
else
puts "#{room_name} :::: Contains a sunwarp with an invalid direction value"
end
else
puts "#{room_name} :::: Contains a sunwarp without a dots and direction"
end
end
(room["progression"] || {}).each do |progression_name, pdata|
if pdata.include? "doors" then
pdata["doors"].each do |door|
if door.kind_of? Hash then
mentioned_doors.add("#{door["room"]} - #{door["door"]}")
else
mentioned_doors.add("#{room_name} - #{door}")
end
end
end
if pdata.include? "panel_doors" then
pdata["panel_doors"].each do |panel_door|
if panel_door.kind_of? Hash then
mentioned_panel_doors.add("#{panel_door["room"]} - #{panel_door["panel_door"]}")
else
mentioned_panel_doors.add("#{room_name} - #{panel_door}")
end
end
end
unless ids.include?("progression") and ids["progression"].include?(progression_name)
puts "#{room_name} - #{progression_name} :::: Progression is missing an item ID"
end
end
end
errored_rooms = mentioned_rooms - configured_rooms
unless errored_rooms.empty? then
puts "The following rooms are mentioned but do not exist: " + errored_rooms.to_s
end
errored_panels = mentioned_panels - configured_panels
unless errored_panels.empty? then
puts "The following panels are mentioned but do not exist: " + errored_panels.to_s
end
errored_doors = mentioned_doors - configured_doors
unless errored_doors.empty? then
puts "The following doors are mentioned but do not exist: " + errored_doors.to_s
end
errored_panel_doors = mentioned_panel_doors - configured_panel_doors
unless errored_panel_doors.empty? then
puts "The following panel doors are mentioned but do not exist: " + errored_panel_doors.to_s
end
door_groups.each do |group,num|
if num == 1 then
puts "Door group \"#{group}\" only has one door in it"
end
unless ids.include?("door_groups") and ids["door_groups"].include?(group)
puts "#{group} :::: Door group is missing an item ID"
end
end
panel_groups.each do |group,num|
if num == 1 then
puts "Panel group \"#{group}\" only has one panel in it"
end
unless ids.include?("panel_groups") and ids["panel_groups"].include?(group)
puts "#{group} :::: Panel group is missing an item ID"
end
end
slashed_rooms = configured_rooms.select do |room|
room.include? "/"
end
unless slashed_rooms.empty? then
puts "The following rooms have slashes in their names: " + slashed_rooms.to_s
end
slashed_panels = configured_panels.select do |panel|
panel.include? "/"
end
unless slashed_panels.empty? then
puts "The following panels have slashes in their names: " + slashed_panels.to_s
end
slashed_doors = configured_doors.select do |door|
door.include? "/"
end
unless slashed_doors.empty? then
puts "The following doors have slashes in their names: " + slashed_doors.to_s
end
puts "#{configured_panels.size} panels (#{non_counting} non counting)"
|