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
|
#!/usr/bin/env python3
# The goal of this script is to print out sets of mods for testing. Each line
# of output is a comma-separated list of mods. Together the lines should cover
# all in-repo mods, in as few lines as possible. Each line must contain only
# mods which are mutually compatible.
import glob
import json
mods_this_time = []
exclusions = [
# Tuple of (mod_id, mod_id) - these two mods will be incompatible
# #76175 causes CI failure between limbified wings and Magiclysm ravenfolk
('limb_wip', 'magiclysm')
]
def compatible_with(mod, existing_mods):
if mod in total_conversions and total_conversions & set(existing_mods):
return False
for entry in exclusions:
if entry[0] == mod and entry[1] in existing_mods:
return False
if entry[1] == mod and entry[0] in existing_mods:
return False
return True
def add_mods(mods):
for mod in mods:
if mod not in all_mod_dependencies:
# Either an invalid mod id, or blacklisted.
return False
for mod in mods:
if mod in mods_this_time:
continue
if not compatible_with(mod, mods_this_time):
return False
if add_mods(all_mod_dependencies[mod]):
mods_this_time.append(mod)
else:
return False
return True
def print_modlist(modlist, master_list):
print(','.join(modlist))
master_list -= set(modlist)
modlist.clear()
all_mod_dependencies = {}
total_conversions = set()
for info in glob.glob('data/mods/*/modinfo.json'):
mod_info = json.load(open(info, encoding='utf-8'))
for e in mod_info:
if(e["type"] == "MOD_INFO" and
("obsolete" not in e or not e["obsolete"])):
ident = e["id"]
all_mod_dependencies[ident] = e.get("dependencies", [])
if e["category"] == "total_conversion":
total_conversions.add(ident)
mods_remaining = set(all_mod_dependencies)
# Make sure aftershock can load by itself.
add_mods(["aftershock"])
print_modlist(mods_this_time, mods_remaining)
while mods_remaining:
for mod in mods_remaining:
if mod not in mods_this_time:
add_mods([mod])
if not mods_remaining & set(mods_this_time):
raise RuntimeError(
'mods remain ({}) but none could be added'.format(mods_remaining))
print_modlist(mods_this_time, mods_remaining)
|