| 
									
										
										
										
											2021-04-10 03:03:46 +02:00
										 |  |  | from __future__ import annotations | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | # Factorio technologies are imported from a .json document in /data | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  | from typing import Dict, Set, FrozenSet | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | import json | 
					
						
							| 
									
										
										
										
											2021-05-22 10:06:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | import Options | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | import Utils | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  | import logging | 
					
						
							| 
									
										
										
										
											2021-05-22 10:06:21 +02:00
										 |  |  | import functools | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:37:15 +02:00
										 |  |  | factorio_id = 2 ** 17 | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  | source_folder = Utils.local_path("data", "factorio") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | with open(os.path.join(source_folder, "techs.json")) as f: | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |     raw = json.load(f) | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  | with open(os.path.join(source_folder, "recipes.json")) as f: | 
					
						
							| 
									
										
										
										
											2021-04-08 19:53:24 +02:00
										 |  |  |     raw_recipes = json.load(f) | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  | with open(os.path.join(source_folder, "machines.json")) as f: | 
					
						
							|  |  |  |     raw_machines = json.load(f) | 
					
						
							| 
									
										
										
										
											2021-05-03 18:06:21 +02:00
										 |  |  | tech_table: Dict[str, int] = {} | 
					
						
							| 
									
										
										
										
											2021-04-24 01:16:49 +02:00
										 |  |  | technology_table: Dict[str, Technology] = {} | 
					
						
							| 
									
										
										
										
											2021-04-05 15:37:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-10 00:08:59 +02:00
										 |  |  | always = lambda state: True | 
					
						
							| 
									
										
										
										
											2021-04-05 15:37:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-10 03:03:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  | class FactorioElement(): | 
					
						
							|  |  |  |     name: str | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return f"{self.__class__.__name__}({self.name})" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __hash__(self): | 
					
						
							|  |  |  |         return hash(self.name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Technology(FactorioElement):  # maybe make subclass of Location? | 
					
						
							| 
									
										
										
										
											2021-04-24 01:16:49 +02:00
										 |  |  |     def __init__(self, name, ingredients, factorio_id): | 
					
						
							| 
									
										
										
										
											2021-04-05 15:37:15 +02:00
										 |  |  |         self.name = name | 
					
						
							|  |  |  |         self.factorio_id = factorio_id | 
					
						
							|  |  |  |         self.ingredients = ingredients | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:16:49 +02:00
										 |  |  |     def build_rule(self, player: int): | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  |         logging.debug(f"Building rules for {self.name}") | 
					
						
							| 
									
										
										
										
											2021-04-10 00:08:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 10:06:21 +02:00
										 |  |  |         return lambda state, technologies=technologies: all(state.has(f"Automated {ingredient}", player) | 
					
						
							|  |  |  |                                                             for ingredient in self.ingredients) | 
					
						
							| 
									
										
										
										
											2021-04-08 19:53:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  |     def get_prior_technologies(self) -> Set[Technology]: | 
					
						
							| 
									
										
										
										
											2021-04-10 03:03:46 +02:00
										 |  |  |         """Get Technologies that have to precede this one to resolve tree connections.""" | 
					
						
							|  |  |  |         technologies = set() | 
					
						
							|  |  |  |         for ingredient in self.ingredients: | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  |             technologies |= required_technologies[ingredient]  # technologies that unlock the recipes | 
					
						
							| 
									
										
										
										
											2021-04-10 03:03:46 +02:00
										 |  |  |         return technologies | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:37:15 +02:00
										 |  |  |     def __hash__(self): | 
					
						
							|  |  |  |         return self.factorio_id | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:16:49 +02:00
										 |  |  |     def get_custom(self, world, allowed_packs: Set[str], player: int) -> CustomTechnology: | 
					
						
							|  |  |  |         return CustomTechnology(self, world, allowed_packs, player) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class CustomTechnology(Technology): | 
					
						
							|  |  |  |     """A particularly configured Technology for a world.""" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, origin: Technology, world, allowed_packs: Set[str], player: int): | 
					
						
							|  |  |  |         ingredients = origin.ingredients & allowed_packs | 
					
						
							|  |  |  |         self.player = player | 
					
						
							| 
									
										
										
										
											2021-06-15 15:32:40 +02:00
										 |  |  |         if world.random_tech_ingredients[player] and origin.name not in world.worlds[player].static_nodes: | 
					
						
							| 
									
										
										
										
											2021-04-24 01:16:49 +02:00
										 |  |  |             ingredients = list(ingredients) | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  |             ingredients.sort()  # deterministic sample | 
					
						
							| 
									
										
										
										
											2021-04-24 01:16:49 +02:00
										 |  |  |             ingredients = world.random.sample(ingredients, world.random.randint(1, len(ingredients))) | 
					
						
							|  |  |  |         super(CustomTechnology, self).__init__(origin.name, ingredients, origin.factorio_id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  | class Recipe(FactorioElement): | 
					
						
							| 
									
										
										
										
											2021-04-08 19:53:24 +02:00
										 |  |  |     def __init__(self, name, category, ingredients, products): | 
					
						
							|  |  |  |         self.name = name | 
					
						
							|  |  |  |         self.category = category | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  |         self.ingredients = ingredients | 
					
						
							|  |  |  |         self.products = products | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return f"{self.__class__.__name__}({self.name})" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def crafting_machines(self) -> Set[Machine]: | 
					
						
							|  |  |  |         """crafting machines able to run this recipe""" | 
					
						
							|  |  |  |         return machines_per_category[self.category] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def unlocking_technologies(self) -> Set[Technology]: | 
					
						
							|  |  |  |         """Unlocked by any of the returned technologies. Empty set indicates a starting recipe.""" | 
					
						
							|  |  |  |         return {technology_table[tech_name] for tech_name in recipe_sources.get(self.name, ())} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  | class Machine(FactorioElement): | 
					
						
							|  |  |  |     def __init__(self, name, categories): | 
					
						
							|  |  |  |         self.name: str = name | 
					
						
							|  |  |  |         self.categories: set = categories | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | # recipes and technologies can share names in Factorio | 
					
						
							| 
									
										
										
										
											2021-04-05 15:37:15 +02:00
										 |  |  | for technology_name in sorted(raw): | 
					
						
							|  |  |  |     data = raw[technology_name] | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |     factorio_id += 1 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:37:15 +02:00
										 |  |  |     current_ingredients = set(data["ingredients"]) | 
					
						
							| 
									
										
										
										
											2021-04-24 01:16:49 +02:00
										 |  |  |     technology = Technology(technology_name, current_ingredients, factorio_id) | 
					
						
							|  |  |  |     factorio_id += 1 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:37:15 +02:00
										 |  |  |     tech_table[technology_name] = technology.factorio_id | 
					
						
							|  |  |  |     technology_table[technology_name] = technology | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  | recipe_sources: Dict[str, str] = {}  # recipe_name -> technology source | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | for technology, data in raw.items(): | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  |     for recipe_name in data["unlocks"]: | 
					
						
							|  |  |  |         recipe_sources.setdefault(recipe_name, set()).add(technology) | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-05 15:37:15 +02:00
										 |  |  | del (raw) | 
					
						
							|  |  |  | lookup_id_to_name: Dict[int, str] = {item_id: item_name for item_name, item_id in tech_table.items()} | 
					
						
							| 
									
										
										
										
											2021-06-06 21:11:58 +02:00
										 |  |  | recipes = {} | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  | all_product_sources: Dict[str, Set[Recipe]] = {"character": set()} | 
					
						
							| 
									
										
										
										
											2021-04-08 19:53:24 +02:00
										 |  |  | for recipe_name, recipe_data in raw_recipes.items(): | 
					
						
							|  |  |  |     # example: | 
					
						
							|  |  |  |     # "accumulator":{"ingredients":["iron-plate","battery"],"products":["accumulator"],"category":"crafting"} | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     recipe = Recipe(recipe_name, recipe_data["category"], set(recipe_data["ingredients"]), set(recipe_data["products"])) | 
					
						
							| 
									
										
										
										
											2021-06-06 21:11:58 +02:00
										 |  |  |     recipes[recipe_name] = Recipe | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  |     if recipe.products.isdisjoint(recipe.ingredients) and "empty-barrel" not in recipe.products:  # prevents loop recipes like uranium centrifuging | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  |         for product_name in recipe.products: | 
					
						
							| 
									
										
										
										
											2021-05-10 02:33:54 +02:00
										 |  |  |             all_product_sources.setdefault(product_name, set()).add(recipe) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  | del (raw_recipes) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | machines: Dict[str, Machine] = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | for name, categories in raw_machines.items(): | 
					
						
							|  |  |  |     machine = Machine(name, set(categories)) | 
					
						
							|  |  |  |     machines[name] = machine | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | del (raw_machines) | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | # build requirements graph for all technology ingredients | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | all_ingredient_names: Set[str] = set() | 
					
						
							|  |  |  | for technology in technology_table.values(): | 
					
						
							|  |  |  |     all_ingredient_names |= technology.ingredients | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  | def unlock_just_tech(recipe: Recipe, _done) -> Set[Technology]: | 
					
						
							|  |  |  |     current_technologies = set() | 
					
						
							|  |  |  |     current_technologies |= recipe.unlocking_technologies | 
					
						
							|  |  |  |     for ingredient_name in recipe.ingredients: | 
					
						
							|  |  |  |         current_technologies |= recursively_get_unlocking_technologies(ingredient_name, _done) | 
					
						
							|  |  |  |     return current_technologies | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def unlock(recipe: Recipe, _done) -> Set[Technology]: | 
					
						
							|  |  |  |     current_technologies = set() | 
					
						
							|  |  |  |     current_technologies |= recipe.unlocking_technologies | 
					
						
							|  |  |  |     for ingredient_name in recipe.ingredients: | 
					
						
							|  |  |  |         current_technologies |= recursively_get_unlocking_technologies(ingredient_name, _done) | 
					
						
							|  |  |  |     current_technologies |= required_category_technologies[recipe.category] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return current_technologies | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def recursively_get_unlocking_technologies(ingredient_name, _done=None, unlock_func=unlock_just_tech) -> Set[Technology]: | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  |     if _done: | 
					
						
							|  |  |  |         if ingredient_name in _done: | 
					
						
							|  |  |  |             return set() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             _done.add(ingredient_name) | 
					
						
							|  |  |  |     else: | 
					
						
							| 
									
										
										
										
											2021-04-10 00:21:56 +02:00
										 |  |  |         _done = {ingredient_name} | 
					
						
							| 
									
										
										
										
											2021-05-10 02:33:54 +02:00
										 |  |  |     recipes = all_product_sources.get(ingredient_name) | 
					
						
							|  |  |  |     if not recipes: | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  |         return set() | 
					
						
							| 
									
										
										
										
											2021-05-10 02:33:54 +02:00
										 |  |  |     current_technologies = set() | 
					
						
							|  |  |  |     for recipe in recipes: | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  |         current_technologies |= unlock_func(recipe, _done) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  |     return current_technologies | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | required_machine_technologies: Dict[str, FrozenSet[Technology]] = {} | 
					
						
							|  |  |  | for ingredient_name in machines: | 
					
						
							|  |  |  |     required_machine_technologies[ingredient_name] = frozenset(recursively_get_unlocking_technologies(ingredient_name)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | logical_machines = {} | 
					
						
							|  |  |  | for machine in machines.values(): | 
					
						
							|  |  |  |     logically_useful = True | 
					
						
							|  |  |  |     for pot_source_machine in machines.values(): | 
					
						
							|  |  |  |         if machine != pot_source_machine \ | 
					
						
							|  |  |  |                 and machine.categories.issuperset(pot_source_machine.categories) \ | 
					
						
							|  |  |  |                 and required_machine_technologies[machine.name].issuperset( | 
					
						
							|  |  |  |                 required_machine_technologies[pot_source_machine.name]): | 
					
						
							|  |  |  |             logically_useful = False | 
					
						
							|  |  |  |             break | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if logically_useful: | 
					
						
							|  |  |  |         logical_machines[machine.name] = machine | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | del(required_machine_technologies) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | machines_per_category: Dict[str: Set[Machine]] = {} | 
					
						
							|  |  |  | for machine in logical_machines.values(): | 
					
						
							|  |  |  |     for category in machine.categories: | 
					
						
							|  |  |  |         machines_per_category.setdefault(category, set()).add(machine) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # required technologies to be able to craft recipes from a certain category | 
					
						
							|  |  |  | required_category_technologies: Dict[str, FrozenSet[FrozenSet[Technology]]] = {} | 
					
						
							|  |  |  | for category_name, cat_machines in machines_per_category.items(): | 
					
						
							|  |  |  |     techs = set() | 
					
						
							|  |  |  |     for machine in cat_machines: | 
					
						
							|  |  |  |         techs |= recursively_get_unlocking_technologies(machine.name) | 
					
						
							|  |  |  |     required_category_technologies[category_name] = frozenset(techs) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  | required_technologies: Dict[str, FrozenSet[Technology]] = {} | 
					
						
							|  |  |  | for ingredient_name in all_ingredient_names: | 
					
						
							| 
									
										
										
										
											2021-05-19 06:52:53 +02:00
										 |  |  |     required_technologies[ingredient_name] = frozenset( | 
					
						
							|  |  |  |         recursively_get_unlocking_technologies(ingredient_name, unlock_func=unlock)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 22:10:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | advancement_technologies: Set[str] = set() | 
					
						
							|  |  |  | for technologies in required_technologies.values(): | 
					
						
							|  |  |  |     advancement_technologies |= {technology.name for technology in technologies} | 
					
						
							| 
									
										
										
										
											2021-05-22 10:06:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | @functools.lru_cache(10) | 
					
						
							|  |  |  | def get_rocket_requirements(ingredients: Set[str]) -> Set[str]: | 
					
						
							| 
									
										
										
										
											2021-05-22 21:13:53 +02:00
										 |  |  |     techs = recursively_get_unlocking_technologies("rocket-silo") | 
					
						
							| 
									
										
										
										
											2021-05-22 10:06:21 +02:00
										 |  |  |     for ingredient in ingredients: | 
					
						
							|  |  |  |         techs |= recursively_get_unlocking_technologies(ingredient) | 
					
						
							|  |  |  |     return {tech.name for tech in techs} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | rocket_recipes = { | 
					
						
							|  |  |  |     Options.MaxSciencePack.option_space_science_pack: | 
					
						
							|  |  |  |         {"rocket-control-unit": 10, "low-density-structure": 10, "rocket-fuel": 10}, | 
					
						
							|  |  |  |     Options.MaxSciencePack.option_utility_science_pack: | 
					
						
							|  |  |  |         {"speed-module": 10, "steel-plate": 10, "solid-fuel": 10}, | 
					
						
							|  |  |  |     Options.MaxSciencePack.option_production_science_pack: | 
					
						
							|  |  |  |         {"speed-module": 10, "steel-plate": 10, "solid-fuel": 10}, | 
					
						
							|  |  |  |     Options.MaxSciencePack.option_chemical_science_pack: | 
					
						
							|  |  |  |         {"advanced-circuit": 10, "steel-plate": 10, "solid-fuel": 10}, | 
					
						
							|  |  |  |     Options.MaxSciencePack.option_military_science_pack: | 
					
						
							|  |  |  |         {"defender-capsule": 10, "stone-wall": 10, "coal": 10}, | 
					
						
							|  |  |  |     Options.MaxSciencePack.option_logistic_science_pack: | 
					
						
							|  |  |  |         {"electronic-circuit": 10, "stone-brick": 10, "coal": 10}, | 
					
						
							|  |  |  |     Options.MaxSciencePack.option_automation_science_pack: | 
					
						
							|  |  |  |         {"copper-cable": 10, "iron-plate": 10, "wood": 10} | 
					
						
							|  |  |  | } |