Archipelago/worlds/stardew_valley/stardew_rule/indirect_connection.py

44 lines
1.1 KiB
Python

from functools import singledispatch
from typing import Set
from . import StardewRule, Reach, Count, AggregatingStardewRule, Has
def look_for_indirect_connection(rule: StardewRule) -> Set[str]:
required_regions = set()
_find(rule, required_regions, depth=0)
return required_regions
@singledispatch
def _find(rule: StardewRule, regions: Set[str], depth: int):
...
@_find.register
def _(rule: AggregatingStardewRule, regions: Set[str], depth: int):
assert depth < 50, "Recursion depth exceeded"
for r in rule.original_rules:
_find(r, regions, depth + 1)
@_find.register
def _(rule: Count, regions: Set[str], depth: int):
assert depth < 50, "Recursion depth exceeded"
for r in rule.rules:
_find(r, regions, depth + 1)
@_find.register
def _(rule: Has, regions: Set[str], depth: int):
assert depth < 50, f"Recursion depth exceeded on {rule.item}"
r = rule.other_rules[rule.item]
_find(r, regions, depth + 1)
@_find.register
def _(rule: Reach, regions: Set[str], depth: int):
assert depth < 50, "Recursion depth exceeded"
if rule.resolution_hint == "Region":
regions.add(rule.spot)