-
Notifications
You must be signed in to change notification settings - Fork 160
/
Copy pathgenerate_dicts_from_data_json.py
582 lines (482 loc) · 23.2 KB
/
generate_dicts_from_data_json.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
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
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
"""
This script does the following:
- Loop over all abilities, checking what unit they create and if it requires a placement position
- Loop over all units, checking what abilities they have and which of those create units, and what tech requirements they have
- Loop over all all upgrades and get their creation ability, which unit can research it and what building requirements there are
- Loop over all units and get their unit and tech aliases
data.json origin:
https://github.com/BurnySc2/sc2-techtree/tree/develop/data
"""
from __future__ import annotations
import json
import lzma
import pickle
from collections import OrderedDict
from pathlib import Path
from loguru import logger
from sc2.game_data import GameData
from sc2.ids.ability_id import AbilityId
from sc2.ids.unit_typeid import UnitTypeId
from sc2.ids.upgrade_id import UpgradeId
def get_map_file_path() -> Path:
return Path(__file__).parent / "test" / "pickle_data" / "DeathAuraLE.xz"
# Custom repr function so that the output is always the same and only changes when there were changes in the data.json tech tree file
# The output just needs to be ordered (sorted by enum name), but it does not matter anymore if the bot then imports an unordered dict and set
class OrderedDict2(OrderedDict):
def __repr__(self):
if not self:
return "{}"
return (
"{"
+ ", ".join(f"{repr(key)}: {repr(value)}" for key, value in sorted(self.items(), key=lambda u: u[0].name))
+ "}"
)
class OrderedSet2(set):
def __repr__(self):
if not self:
return "set()"
return "{" + ", ".join(repr(item) for item in sorted(self, key=lambda u: u.name)) + "}"
def dump_dict_to_file(
my_dict: OrderedDict2, file_path: Path, dict_name: str, file_header: str = "", dict_type_annotation: str = ""
):
with file_path.open("w") as f:
f.write(file_header)
f.write("\n")
f.write(f"{dict_name}{dict_type_annotation} = ")
assert isinstance(my_dict, OrderedDict2)
logger.info(my_dict)
f.write(repr(my_dict))
def generate_init_file(dict_file_paths: list[Path], file_path: Path, file_header: str):
base_file_names = sorted(path.stem for path in dict_file_paths)
with file_path.open("w") as f:
f.write(file_header)
f.write("\n")
all_line = f"__all__ = {base_file_names}"
logger.info(all_line)
f.write(all_line)
def get_unit_train_build_abilities(data):
ability_data = data["Ability"]
unit_data = data["Unit"]
_upgrade_data = data["Upgrade"]
# From which abilities can a unit be trained
train_abilities: dict[UnitTypeId, set[AbilityId]] = OrderedDict2()
# If the ability requires a placement position
ability_requires_placement: set[AbilityId] = set()
# Map ability to unittypeid
ability_to_unittypeid_dict: dict[AbilityId, UnitTypeId] = OrderedDict2()
# From which abilities can a unit be morphed
# unit_morph_abilities: dict[UnitTypeId, set[AbilityId]] = {}
entry: dict
for entry in ability_data:
"""
"target": "PointOrUnit"
"""
if isinstance(entry.get("target", {}), str):
continue
ability_id: AbilityId = AbilityId(entry["id"])
created_unit_type_id: UnitTypeId
# Check if it is a unit train ability
requires_placement = False
train_unit_type_id_value: int = entry.get("target", {}).get("Train", {}).get("produces", 0)
train_place_unit_type_id_value: int = entry.get("target", {}).get("TrainPlace", {}).get("produces", 0)
morph_unit_type_id_value: int = entry.get("target", {}).get("Morph", {}).get("produces", 0)
build_unit_type_id_value: int = entry.get("target", {}).get("Build", {}).get("produces", 0)
build_on_unit_unit_type_id_value: int = entry.get("target", {}).get("BuildOnUnit", {}).get("produces", 0)
if not train_unit_type_id_value and train_place_unit_type_id_value:
train_unit_type_id_value = train_place_unit_type_id_value
requires_placement = True
# Collect larva morph abilities, and one way morphs (exclude burrow, hellbat morph, siege tank siege)
# Also doesnt include building addons
if not train_unit_type_id_value and (
"LARVATRAIN_" in ability_id.name
or ability_id
in {
AbilityId.MORPHTOBROODLORD_BROODLORD,
AbilityId.MORPHZERGLINGTOBANELING_BANELING,
AbilityId.MORPHTORAVAGER_RAVAGER,
AbilityId.MORPHTOBANELING_BANELING,
AbilityId.MORPH_LURKER,
AbilityId.UPGRADETOLAIR_LAIR,
AbilityId.UPGRADETOHIVE_HIVE,
AbilityId.UPGRADETOGREATERSPIRE_GREATERSPIRE,
AbilityId.UPGRADETOORBITAL_ORBITALCOMMAND,
AbilityId.UPGRADETOPLANETARYFORTRESS_PLANETARYFORTRESS,
AbilityId.MORPH_OVERLORDTRANSPORT,
AbilityId.MORPH_OVERSEER,
}
):
# If all morph units are used, unit_trained_from.py will be "wrong" because it will list that a siege tank can be trained from siegetanksieged and similar:
# UnitTypeId.SIEGETANK: {UnitTypeId.SIEGETANKSIEGED, UnitTypeId.FACTORY},
# if not train_unit_type_id_value and morph_unit_type_id_value:
train_unit_type_id_value = morph_unit_type_id_value
# Add all build abilities, like construct buildings and train queen (exception)
if not train_unit_type_id_value and build_unit_type_id_value:
train_unit_type_id_value = build_unit_type_id_value
if "BUILD_" in entry["name"]:
requires_placement = True
# Add build gas building (refinery, assimilator, extractor)
# TODO: target needs to be a unit, not a position, but i dont want to store an extra line just for this - needs to be an exception in bot_ai.py
if not train_unit_type_id_value and build_on_unit_unit_type_id_value:
train_unit_type_id_value = build_on_unit_unit_type_id_value
if train_unit_type_id_value:
created_unit_type_id = UnitTypeId(train_unit_type_id_value)
if created_unit_type_id not in train_abilities:
train_abilities[created_unit_type_id] = {ability_id}
else:
train_abilities[created_unit_type_id].add(ability_id)
if requires_placement:
ability_requires_placement.add(ability_id)
ability_to_unittypeid_dict[ability_id] = created_unit_type_id
"""
unit_train_abilities = {
UnitTypeId.GATEWAY: {
UnitTypeId.ADEPT: {
"ability": AbilityId.TRAIN_ADEPT,
"requires_techlab": False,
"required_building": UnitTypeId.CYBERNETICSCORE, # Or None
"requires_placement_position": False, # True for warp gate
"requires_power": True, # If a pylon nearby is required
},
UnitTypeId.Zealot: {
"ability": AbilityId.GATEWAYTRAIN_ZEALOT,
...
}
}
}
"""
unit_train_abilities: dict[UnitTypeId, dict[str, AbilityId | bool | UnitTypeId]] = OrderedDict2()
for entry in unit_data:
unit_abilities = entry.get("abilities", [])
unit_type = UnitTypeId(entry["id"])
current_unit_train_abilities = OrderedDict2()
for ability_info in unit_abilities:
ability_id_value: int = ability_info.get("ability", 0)
if ability_id_value:
ability_id: AbilityId = AbilityId(ability_id_value)
# Ability is not a train ability
if ability_id not in ability_to_unittypeid_dict:
continue
requires_techlab: bool = False
required_building: UnitTypeId | None = None
requires_placement_position: bool = False
requires_power: bool = False
"""
requirements = [
{
"addon": 5
},
{
"building": 29
}
]
"""
requirements: list[dict[str, int]] = ability_info.get("requirements", [])
if requirements:
# Assume train abilities only have one tech building requirement; thors requiring armory and techlab is seperatedly counted
assert (
len([req for req in requirements if req.get("building", 0)]) <= 1
), f"Error: Building {unit_type} has more than one tech requirements with train ability {ability_id}"
# UnitTypeId 5 == Techlab
requires_techlab: bool = any(req for req in requirements if req.get("addon", 0) == 5)
requires_tech_builing_id_value: int = next(
(req["building"] for req in requirements if req.get("building", 0)), 0
)
if requires_tech_builing_id_value:
required_building = UnitTypeId(requires_tech_builing_id_value)
if ability_id in ability_requires_placement:
requires_placement_position = True
requires_power = entry.get("needs_power", False)
resulting_unit = ability_to_unittypeid_dict[ability_id]
ability_dict = {"ability": ability_id}
# Only add boolean values and tech requirement if they actually exist, to make the resulting dict file smaller
if requires_techlab:
ability_dict["requires_techlab"] = requires_techlab
if required_building:
ability_dict["required_building"] = required_building
if requires_placement_position:
ability_dict["requires_placement_position"] = requires_placement_position
if requires_power:
ability_dict["requires_power"] = requires_power
current_unit_train_abilities[resulting_unit] = ability_dict
if current_unit_train_abilities:
unit_train_abilities[unit_type] = current_unit_train_abilities
return unit_train_abilities
def get_upgrade_abilities(data):
ability_data = data["Ability"]
unit_data = data["Unit"]
_upgrade_data = data["Upgrade"]
ability_to_upgrade_dict: dict[AbilityId, UpgradeId] = OrderedDict2()
"""
We want to be able to research an upgrade by doing
await self.can_research(UpgradeId, return_idle_structures=True) -> returns list of idle structures that can research it
So we need to assign each upgrade id one building type, and its research ability and requirements (e.g. armory for infantry level 2)
"""
# Collect all upgrades and their corresponding abilities
entry: dict
for entry in ability_data:
if isinstance(entry.get("target", {}), str):
continue
ability_id: AbilityId = AbilityId(entry["id"])
upgrade_id_value: int = entry.get("target", {}).get("Research", {}).get("upgrade", 0)
if upgrade_id_value:
upgrade_id: UpgradeId = UpgradeId(upgrade_id_value)
ability_to_upgrade_dict[ability_id] = upgrade_id
"""
unit_research_abilities = {
UnitTypeId.ENGINEERINGBAY: {
UpgradeId.TERRANINFANTRYWEAPONSLEVEL1:
{
"ability": AbilityId.ENGINEERINGBAYRESEARCH_TERRANINFANTRYWEAPONSLEVEL1,
"required_building": None,
"requires_power": False, # If a pylon nearby is required
},
UpgradeId.TERRANINFANTRYWEAPONSLEVEL2: {
"ability": AbilityId.ENGINEERINGBAYRESEARCH_TERRANINFANTRYWEAPONSLEVEL2,
"required_building": UnitTypeId.ARMORY,
"requires_power": False, # If a pylon nearby is required
},
}
}
"""
unit_research_abilities = OrderedDict2()
for entry in unit_data:
unit_abilities = entry.get("abilities", [])
unit_type = UnitTypeId(entry["id"])
if unit_type == UnitTypeId.TECHLAB:
continue
current_unit_research_abilities = OrderedDict2()
for ability_info in unit_abilities:
ability_id_value: int = ability_info.get("ability", 0)
if ability_id_value:
ability_id: AbilityId = AbilityId(ability_id_value)
# Upgrade is not a known upgrade ability
if ability_id not in ability_to_upgrade_dict:
continue
required_building = None
required_upgrade = None
requirements = ability_info.get("requirements", [])
if requirements:
req_building_id_value = next(
(req["building"] for req in requirements if req.get("building", 0)), None
)
if req_building_id_value:
req_building_id = UnitTypeId(req_building_id_value)
required_building = req_building_id
req_upgrade_id_value = next((req["upgrade"] for req in requirements if req.get("upgrade", 0)), None)
if req_upgrade_id_value:
req_upgrade_id = UpgradeId(req_upgrade_id_value)
required_upgrade = req_upgrade_id
requires_power = entry.get("needs_power", False)
resulting_upgrade = ability_to_upgrade_dict[ability_id]
research_info = {"ability": ability_id}
if required_building:
research_info["required_building"] = required_building
if required_upgrade:
research_info["required_upgrade"] = required_upgrade
if requires_power:
research_info["requires_power"] = requires_power
current_unit_research_abilities[resulting_upgrade] = research_info
if current_unit_research_abilities:
unit_research_abilities[unit_type] = current_unit_research_abilities
return unit_research_abilities
def get_unit_created_from(unit_train_abilities: dict):
unit_created_from = OrderedDict2()
for creator_unit, create_abilities in unit_train_abilities.items():
for created_unit, create_info in create_abilities.items():
if created_unit not in unit_created_from:
unit_created_from[created_unit] = OrderedSet2()
unit_created_from[created_unit].add(creator_unit)
return unit_created_from
def get_upgrade_researched_from(unit_research_abilities: dict):
upgrade_researched_from = OrderedDict2()
for researcher_unit, research_abilities in unit_research_abilities.items():
for upgrade, research_info in research_abilities.items():
# This if statement is to prevent LAIR and HIVE overriding "UpgradeId.OVERLORDSPEED" as well as greater spire overriding upgrade abilities
if upgrade not in upgrade_researched_from:
upgrade_researched_from[upgrade] = researcher_unit
return upgrade_researched_from
def get_unit_abilities(data: dict):
_ability_data = data["Ability"]
unit_data = data["Unit"]
_upgrade_data = data["Upgrade"]
all_unit_abilities: dict[UnitTypeId, set[AbilityId]] = OrderedDict2()
entry: dict
for entry in unit_data:
entry_unit_abilities = entry.get("abilities", [])
unit_type = UnitTypeId(entry["id"])
current_collected_unit_abilities: set[AbilityId] = OrderedSet2()
for ability_info in entry_unit_abilities:
ability_id_value: int = ability_info.get("ability", 0)
if ability_id_value:
ability_id: AbilityId = AbilityId(ability_id_value)
current_collected_unit_abilities.add(ability_id)
# logger.info(unit_type, current_unit_abilities)
if current_collected_unit_abilities:
all_unit_abilities[unit_type] = current_collected_unit_abilities
return all_unit_abilities
def generate_unit_alias_dict(data: dict):
_ability_data = data["Ability"]
unit_data = data["Unit"]
_upgrade_data = data["Upgrade"]
# Load pickled game data files from one of the test files
pickled_file_path = get_map_file_path()
assert pickled_file_path.is_file(), f"Could not find pickled data file {pickled_file_path}"
logger.info(f"Loading pickled game data file {pickled_file_path}")
with lzma.open(pickled_file_path.absolute(), "rb") as f:
raw_game_data, raw_game_info, raw_observation = pickle.load(f)
game_data = GameData(raw_game_data.data)
all_unit_aliases: dict[UnitTypeId, UnitTypeId] = OrderedDict2()
all_tech_aliases: dict[UnitTypeId, set[UnitTypeId]] = OrderedDict2()
entry: dict
for entry in unit_data:
unit_type_value = entry["id"]
unit_type = UnitTypeId(entry["id"])
current_unit_tech_aliases: set[UnitTypeId] = OrderedSet2()
assert (
unit_type_value in game_data.units
), f"Unit {unit_type} not listed in game_data.units - perhaps pickled file {pickled_file_path} is outdated?"
unit_alias: int = game_data.units[unit_type_value]._proto.unit_alias
if unit_alias:
# Might be 0 if it has no alias
unit_alias_unit_type_id = UnitTypeId(unit_alias)
all_unit_aliases[unit_type] = unit_alias_unit_type_id
tech_aliases: list[int] = game_data.units[unit_type_value]._proto.tech_alias
for tech_alias in tech_aliases:
# Might be 0 if it has no alias
unit_alias_unit_type_id = UnitTypeId(tech_alias)
current_unit_tech_aliases.add(unit_alias_unit_type_id)
if current_unit_tech_aliases:
all_tech_aliases[unit_type] = current_unit_tech_aliases
return all_unit_aliases, all_tech_aliases
def generate_redirect_abilities_dict(data: dict):
ability_data = data["Ability"]
_unit_data = data["Unit"]
_upgrade_data = data["Upgrade"]
all_redirect_abilities: dict[AbilityId, AbilityId] = OrderedDict2()
entry: dict
for entry in ability_data:
ability_id_value: int = entry["id"]
try:
ability_id: AbilityId = AbilityId(ability_id_value)
except Exception:
logger.info(f"Error with ability id value {ability_id_value}")
continue
generic_redirect_ability_value = entry.get("remaps_to_ability_id", 0)
if generic_redirect_ability_value == 0:
# No generic ability available
continue
all_redirect_abilities[ability_id] = AbilityId(generic_redirect_ability_value)
return all_redirect_abilities
def main():
path = Path(__file__).parent
data_path = path / "data" / "data.json"
with data_path.open() as f:
data = json.load(f)
dicts_path = path / "sc2" / "dicts"
Path(dicts_path).mkdir(parents=True, exist_ok=True)
# All unit train and build abilities
unit_train_abilities = get_unit_train_build_abilities(data=data)
unit_creation_dict_path = dicts_path / "unit_train_build_abilities.py"
# All upgrades and which building can research which upgrade
unit_research_abilities = get_upgrade_abilities(data=data)
unit_research_abilities_dict_path = dicts_path / "unit_research_abilities.py"
# All train abilities (where a unit can be trained from)
unit_trained_from = get_unit_created_from(unit_train_abilities=unit_train_abilities)
unit_trained_from_dict_path = dicts_path / "unit_trained_from.py"
# All research abilities (where an upgrade can be researched from)
upgrade_researched_from = get_upgrade_researched_from(unit_research_abilities=unit_research_abilities)
upgrade_researched_from_dict_path = dicts_path / "upgrade_researched_from.py"
# All unit abilities without requirements
unit_abilities = get_unit_abilities(data=data)
unit_abilities_dict_path = dicts_path / "unit_abilities.py"
# All unit_alias and tech_alias of a unit type
unit_unit_alias, unit_tech_alias = generate_unit_alias_dict(data=data)
unit_unit_alias_dict_path = dicts_path / "unit_unit_alias.py"
unit_tech_alias_dict_path = dicts_path / "unit_tech_alias.py"
# All redirect (generic) abilities of abilities
all_redirect_abilities = generate_redirect_abilities_dict(data=data)
all_redirect_abilities_path = dicts_path / "generic_redirect_abilities.py"
file_name = Path(__file__).name
file_header = f"""
# THIS FILE WAS AUTOMATICALLY GENERATED BY "{file_name}" DO NOT CHANGE MANUALLY!
# ANY CHANGE WILL BE OVERWRITTEN
from sc2.ids.unit_typeid import UnitTypeId
from sc2.ids.ability_id import AbilityId
from sc2.ids.upgrade_id import UpgradeId
# from sc2.ids.buff_id import BuffId
# from sc2.ids.effect_id import EffectId
from typing import Union
"""
dict_file_paths = [
unit_creation_dict_path,
unit_research_abilities_dict_path,
unit_trained_from_dict_path,
upgrade_researched_from_dict_path,
unit_abilities_dict_path,
unit_unit_alias_dict_path,
unit_tech_alias_dict_path,
all_redirect_abilities_path,
]
init_file_path = dicts_path / "__init__.py"
init_header = f"""# DO NOT EDIT!
# This file was automatically generated by "{file_name}"
"""
generate_init_file(dict_file_paths=dict_file_paths, file_path=init_file_path, file_header=init_header)
dump_dict_to_file(
unit_train_abilities,
unit_creation_dict_path,
dict_name="TRAIN_INFO",
file_header=file_header,
dict_type_annotation=": dict[UnitTypeId, dict[UnitTypeId, dict[str, Union[AbilityId, bool, UnitTypeId]]]]",
)
dump_dict_to_file(
unit_research_abilities,
unit_research_abilities_dict_path,
dict_name="RESEARCH_INFO",
file_header=file_header,
dict_type_annotation=": dict[UnitTypeId, dict[UpgradeId, dict[str, Union[AbilityId, bool, UnitTypeId, UpgradeId]]]]",
)
dump_dict_to_file(
unit_trained_from,
unit_trained_from_dict_path,
dict_name="UNIT_TRAINED_FROM",
file_header=file_header,
dict_type_annotation=": dict[UnitTypeId, set[UnitTypeId]]",
)
dump_dict_to_file(
upgrade_researched_from,
upgrade_researched_from_dict_path,
dict_name="UPGRADE_RESEARCHED_FROM",
file_header=file_header,
dict_type_annotation=": dict[UpgradeId, UnitTypeId]",
)
dump_dict_to_file(
unit_abilities,
unit_abilities_dict_path,
dict_name="UNIT_ABILITIES",
file_header=file_header,
dict_type_annotation=": dict[UnitTypeId, set[AbilityId]]",
)
dump_dict_to_file(
unit_unit_alias,
unit_unit_alias_dict_path,
dict_name="UNIT_UNIT_ALIAS",
file_header=file_header,
dict_type_annotation=": dict[UnitTypeId, UnitTypeId]",
)
dump_dict_to_file(
unit_tech_alias,
unit_tech_alias_dict_path,
dict_name="UNIT_TECH_ALIAS",
file_header=file_header,
dict_type_annotation=": dict[UnitTypeId, set[UnitTypeId]]",
)
dump_dict_to_file(
all_redirect_abilities,
all_redirect_abilities_path,
dict_name="GENERIC_REDIRECT_ABILITIES",
file_header=file_header,
dict_type_annotation=": dict[AbilityId, AbilityId]",
)
if __name__ == "__main__":
main()