2017-12-17 00:25:46 -05:00
from collections import OrderedDict
2018-02-17 18:38:54 -05:00
import copy
2018-01-06 16:25:14 -05:00
from itertools import zip_longest
2017-12-17 00:25:46 -05:00
import json
import logging
2019-05-30 01:10:16 +02:00
import os
2017-12-17 00:25:46 -05:00
import random
import time
2020-01-04 22:08:13 +01:00
import zlib
2020-08-21 18:35:48 +02:00
import concurrent . futures
2017-12-17 00:25:46 -05:00
2018-09-22 22:51:54 -04:00
from BaseClasses import World , CollectionState , Item , Region , Location , Shop
2020-01-06 19:13:42 +01:00
from Items import ItemFactory
2020-06-20 12:22:50 +02:00
from Regions import create_regions , create_shops , mark_light_world_regions , lookup_vanilla_location_to_entrance
2019-07-27 09:13:13 -04:00
from InvertedRegions import create_inverted_regions , mark_dark_world_regions
from EntranceShuffle import link_entrances , link_inverted_entrances
2020-06-09 22:11:14 +02:00
from Rom import patch_rom , patch_race_rom , patch_enemizer , apply_rom_settings , LocalRom , get_hash_string
2017-05-15 20:28:04 +02:00
from Rules import set_rules
2017-10-15 13:52:42 -04:00
from Dungeons import create_dungeons , fill_dungeons , fill_dungeons_restrictive
2020-08-25 17:44:03 +02:00
from Fill import distribute_items_restrictive , flood_items , balance_multiworld_progression
2020-08-20 15:43:22 +02:00
from ItemPool import generate_itempool , difficulties , fill_prizes
2020-07-14 04:48:56 +02:00
from Utils import output_path , parse_player_names , get_options , __version__ , _version_tuple
2020-07-10 22:43:54 +02:00
import Patch
2017-05-15 20:28:04 +02:00
2020-06-28 00:24:45 +02:00
seeddigits = 20
def get_seed ( seed = None ) :
if seed is None :
random . seed ( None )
return random . randint ( 0 , pow ( 10 , seeddigits ) - 1 )
return seed
2019-04-29 16:11:23 -05:00
2017-05-25 15:58:35 +02:00
def main ( args , seed = None ) :
2019-12-28 17:12:27 +01:00
if args . outputpath :
2020-01-10 07:25:16 +01:00
os . makedirs ( args . outputpath , exist_ok = True )
2019-12-28 17:12:27 +01:00
output_path . cached_path = args . outputpath
2020-10-24 02:44:27 +02:00
2020-01-02 12:38:26 +11:00
start = time . perf_counter ( )
2017-05-15 20:28:04 +02:00
# initialize the world
2020-01-22 06:28:58 +01:00
world = World ( args . multi , args . shuffle , args . logic , args . mode , args . swords , args . difficulty ,
args . item_functionality , args . timer , args . progressive . copy ( ) , args . goal , args . algorithm ,
args . accessibility , args . shuffleganon , args . retro , args . custom , args . customitemarray , args . hints )
2020-07-14 07:01:51 +02:00
2017-05-20 14:07:40 +02:00
logger = logging . getLogger ( ' ' )
2020-06-28 00:24:45 +02:00
world . seed = get_seed ( seed )
2020-07-14 07:01:51 +02:00
if args . race :
world . secure ( )
else :
world . random . seed ( world . seed )
2017-05-20 14:07:40 +02:00
2020-01-18 09:50:12 +01:00
world . remote_items = args . remote_items . copy ( )
2019-12-16 21:46:47 +01:00
world . mapshuffle = args . mapshuffle . copy ( )
world . compassshuffle = args . compassshuffle . copy ( )
world . keyshuffle = args . keyshuffle . copy ( )
world . bigkeyshuffle = args . bigkeyshuffle . copy ( )
2020-07-14 07:01:51 +02:00
world . crystals_needed_for_ganon = {
player : world . random . randint ( 0 , 7 ) if args . crystals_ganon [ player ] == ' random ' else int (
args . crystals_ganon [ player ] ) for player in range ( 1 , world . players + 1 ) }
world . crystals_needed_for_gt = {
player : world . random . randint ( 0 , 7 ) if args . crystals_gt [ player ] == ' random ' else int ( args . crystals_gt [ player ] )
for player in range ( 1 , world . players + 1 ) }
2020-09-11 03:23:00 +02:00
world . open_pyramid = args . open_pyramid . copy ( )
2019-12-17 15:55:53 +01:00
world . boss_shuffle = args . shufflebosses . copy ( )
2020-08-19 23:24:17 +02:00
world . enemy_shuffle = args . enemy_shuffle . copy ( )
2019-12-17 15:55:53 +01:00
world . enemy_health = args . enemy_health . copy ( )
world . enemy_damage = args . enemy_damage . copy ( )
2020-08-19 23:24:17 +02:00
world . killable_thieves = args . killable_thieves . copy ( )
world . bush_shuffle = args . bush_shuffle . copy ( )
world . tile_shuffle = args . tile_shuffle . copy ( )
2019-12-30 03:03:53 +01:00
world . beemizer = args . beemizer . copy ( )
2020-02-02 20:10:56 -05:00
world . timer = args . timer . copy ( )
2020-01-18 12:51:10 -05:00
world . shufflepots = args . shufflepots . copy ( )
2020-01-22 06:28:58 +01:00
world . progressive = args . progressive . copy ( )
2020-04-12 15:46:32 -07:00
world . dungeon_counters = args . dungeon_counters . copy ( )
2020-04-16 11:02:16 +02:00
world . glitch_boots = args . glitch_boots . copy ( )
2020-06-17 01:02:54 -07:00
world . triforce_pieces_available = args . triforce_pieces_available . copy ( )
2020-06-07 15:22:24 +02:00
world . triforce_pieces_required = args . triforce_pieces_required . copy ( )
2020-08-23 15:03:06 +02:00
world . shop_shuffle = args . shop_shuffle . copy ( )
2020-05-18 03:54:29 +02:00
world . progression_balancing = { player : not balance for player , balance in args . skip_progression_balancing . items ( ) }
2020-09-20 04:35:45 +02:00
world . shuffle_prizes = args . shuffle_prizes . copy ( )
2020-10-06 13:22:03 -07:00
world . sprite_pool = args . sprite_pool . copy ( )
2020-10-07 19:51:46 +02:00
world . dark_room_logic = args . dark_room_logic . copy ( )
world . restrict_dungeon_item_on_boss = args . restrict_dungeon_item_on_boss . copy ( )
2019-08-11 08:55:38 -04:00
2020-07-15 23:01:29 -07:00
world . rom_seeds = { player : random . Random ( world . random . randint ( 0 , 999999999 ) ) for player in range ( 1 , world . players + 1 ) }
2019-04-18 11:23:24 +02:00
2020-04-20 14:50:49 +02:00
logger . info ( ' ALttP Berserker \' s Multiworld Version %s - Seed: %s \n ' , __version__ , world . seed )
2020-01-14 10:42:27 +01:00
parsed_names = parse_player_names ( args . names , world . players , args . teams )
world . teams = len ( parsed_names )
for i , team in enumerate ( parsed_names , 1 ) :
if world . players > 1 :
logger . info ( ' %s %s ' , ' Team %d : ' % i if world . teams > 1 else ' Players: ' , ' , ' . join ( team ) )
for player , name in enumerate ( team , 1 ) :
world . player_names [ player ] . append ( name )
2017-05-20 14:07:40 +02:00
2020-01-14 10:42:27 +01:00
logger . info ( ' ' )
2017-05-20 14:07:40 +02:00
2019-12-16 16:54:46 +01:00
for player in range ( 1 , world . players + 1 ) :
2019-12-16 17:46:21 +01:00
world . difficulty_requirements [ player ] = difficulties [ world . difficulty [ player ] ]
2018-01-04 01:06:22 -05:00
2020-01-06 19:13:42 +01:00
for tok in filter ( None , args . startinventory [ player ] . split ( ' , ' ) ) :
item = ItemFactory ( tok . strip ( ) , player )
if item :
world . push_precollected ( item )
2020-06-03 22:13:58 +02:00
world . local_items [ player ] = { item . strip ( ) for item in args . local_items [ player ] . split ( ' , ' ) }
2020-01-06 19:13:42 +01:00
2020-06-17 01:33:34 -07:00
world . triforce_pieces_available [ player ] = max ( world . triforce_pieces_available [ player ] , world . triforce_pieces_required [ player ] )
2019-12-16 16:54:46 +01:00
if world . mode [ player ] != ' inverted ' :
2019-07-27 09:13:13 -04:00
create_regions ( world , player )
2019-12-16 16:54:46 +01:00
else :
2019-07-27 09:13:13 -04:00
create_inverted_regions ( world , player )
2020-01-10 11:41:22 +01:00
create_shops ( world , player )
2019-12-16 16:54:46 +01:00
create_dungeons ( world , player )
2017-05-15 20:28:04 +02:00
2017-05-20 14:07:40 +02:00
logger . info ( ' Shuffling the World about. ' )
2019-12-16 16:54:46 +01:00
for player in range ( 1 , world . players + 1 ) :
2020-08-22 09:28:24 -04:00
if world . logic [ player ] not in [ " noglitches " , " minorglitches " ] and world . shuffle [ player ] in \
2020-07-16 04:14:44 +02:00
{ " vanilla " , " dungeonssimple " , " dungeonsfull " , " simple " , " restricted " , " full " } :
world . fix_fake_world [ player ] = False
2019-12-16 16:54:46 +01:00
if world . mode [ player ] != ' inverted ' :
2019-07-27 09:13:13 -04:00
link_entrances ( world , player )
2019-12-16 16:54:46 +01:00
mark_light_world_regions ( world , player )
else :
2019-07-27 09:13:13 -04:00
link_inverted_entrances ( world , player )
2019-12-16 16:54:46 +01:00
mark_dark_world_regions ( world , player )
2017-05-20 14:07:40 +02:00
2019-04-18 16:11:11 -05:00
logger . info ( ' Generating Item Pool. ' )
2019-04-18 11:23:24 +02:00
for player in range ( 1 , world . players + 1 ) :
generate_itempool ( world , player )
2019-04-18 16:11:11 -05:00
2017-05-20 14:07:40 +02:00
logger . info ( ' Calculating Access Rules. ' )
2019-04-18 11:23:24 +02:00
for player in range ( 1 , world . players + 1 ) :
set_rules ( world , player )
2017-05-20 14:07:40 +02:00
2019-04-18 16:11:11 -05:00
logger . info ( ' Placing Dungeon Prizes. ' )
2017-08-05 17:52:18 +02:00
2019-04-18 16:11:11 -05:00
fill_prizes ( world )
2017-08-05 17:52:18 +02:00
2017-06-23 21:32:31 +02:00
logger . info ( ' Placing Dungeon Items. ' )
2017-05-20 14:07:40 +02:00
2017-10-15 16:34:46 -04:00
shuffled_locations = None
2019-12-16 21:46:47 +01:00
if args . algorithm in [ ' balanced ' , ' vt26 ' ] or any ( list ( args . mapshuffle . values ( ) ) + list ( args . compassshuffle . values ( ) ) +
list ( args . keyshuffle . values ( ) ) + list ( args . bigkeyshuffle . values ( ) ) ) :
2020-10-07 19:51:46 +02:00
fill_dungeons_restrictive ( world )
2017-10-15 13:52:42 -04:00
else :
fill_dungeons ( world )
2017-05-20 14:07:40 +02:00
logger . info ( ' Fill the world. ' )
2017-05-25 15:58:35 +02:00
if args . algorithm == ' flood ' :
2017-05-20 14:07:40 +02:00
flood_items ( world ) # different algo, biased towards early game progress items
2017-08-01 19:07:44 +02:00
elif args . algorithm == ' vt25 ' :
2019-12-16 15:27:20 +01:00
distribute_items_restrictive ( world , False )
2017-10-15 13:52:42 -04:00
elif args . algorithm == ' vt26 ' :
2019-12-16 15:27:20 +01:00
distribute_items_restrictive ( world , True , shuffled_locations )
2017-11-11 18:05:06 -06:00
elif args . algorithm == ' balanced ' :
2019-12-16 15:27:20 +01:00
distribute_items_restrictive ( world , True )
2017-06-03 21:28:02 +02:00
2020-05-18 03:54:29 +02:00
if world . players > 1 :
2019-04-18 11:23:24 +02:00
balance_multiworld_progression ( world )
2017-05-20 14:07:40 +02:00
logger . info ( ' Patching ROM. ' )
2020-06-10 21:25:14 +02:00
outfilebase = ' BM_ %s ' % ( args . outputname if args . outputname else world . seed )
2017-05-26 18:39:32 +02:00
2020-01-04 22:08:13 +01:00
rom_names = [ ]
2017-05-20 14:07:40 +02:00
2020-03-06 23:08:46 +01:00
def _gen_rom ( team : int , player : int ) :
2020-08-19 23:24:17 +02:00
use_enemizer = ( world . boss_shuffle [ player ] != ' none ' or world . enemy_shuffle [ player ]
2020-03-06 23:08:46 +01:00
or world . enemy_health [ player ] != ' default ' or world . enemy_damage [ player ] != ' default '
2020-10-04 10:57:30 -07:00
or world . shufflepots [ player ] or world . bush_shuffle [ player ]
2020-08-19 23:24:17 +02:00
or world . killable_thieves [ player ] or world . tile_shuffle [ player ] )
2019-05-30 01:10:16 +02:00
2020-06-09 21:52:46 +02:00
rom = LocalRom ( args . rom )
2019-04-18 11:23:24 +02:00
2020-03-06 23:08:46 +01:00
patch_rom ( world , rom , player , team , use_enemizer )
2019-05-30 01:10:16 +02:00
2020-06-09 21:52:46 +02:00
if use_enemizer :
2020-10-04 10:57:30 -07:00
patch_enemizer ( world , player , rom , args . enemizercli )
2019-05-30 01:10:16 +02:00
2020-03-06 23:08:46 +01:00
if args . race :
2020-10-20 01:16:20 -07:00
patch_race_rom ( rom , world , player )
2019-04-18 11:23:24 +02:00
2020-03-06 23:08:46 +01:00
world . spoiler . hashes [ ( player , team ) ] = get_hash_string ( rom . hash )
2019-05-30 01:10:16 +02:00
2020-10-24 02:44:27 +02:00
palettes_options = { }
palettes_options [ ' dungeon ' ] = args . uw_palettes [ player ]
palettes_options [ ' overworld ' ] = args . ow_palettes [ player ]
palettes_options [ ' hud ' ] = args . hud_palettes [ player ]
palettes_options [ ' sword ' ] = args . sword_palettes [ player ]
palettes_options [ ' shield ' ] = args . shield_palettes [ player ]
palettes_options [ ' link ' ] = args . link_palettes [ player ]
2020-03-06 23:08:46 +01:00
apply_rom_settings ( rom , args . heartbeep [ player ] , args . heartcolor [ player ] , args . quickswap [ player ] ,
args . fastmenu [ player ] , args . disablemusic [ player ] , args . sprite [ player ] ,
2020-10-24 02:44:27 +02:00
palettes_options , world , player , True )
2017-06-04 13:09:47 +02:00
2020-06-09 21:52:46 +02:00
mcsb_name = ' '
if all ( [ world . mapshuffle [ player ] , world . compassshuffle [ player ] , world . keyshuffle [ player ] ,
world . bigkeyshuffle [ player ] ] ) :
mcsb_name = ' -keysanity '
elif [ world . mapshuffle [ player ] , world . compassshuffle [ player ] , world . keyshuffle [ player ] ,
world . bigkeyshuffle [ player ] ] . count ( True ) == 1 :
2020-09-02 15:06:36 -07:00
mcsb_name = ' -mapshuffle ' if world . mapshuffle [ player ] else \
' -compassshuffle ' if world . compassshuffle [ player ] else \
' -universal_keys ' if world . keyshuffle [ player ] == " universal " else \
' -keyshuffle ' if world . keyshuffle [ player ] else ' -bigkeyshuffle '
2020-06-09 21:52:46 +02:00
elif any ( [ world . mapshuffle [ player ] , world . compassshuffle [ player ] , world . keyshuffle [ player ] ,
world . bigkeyshuffle [ player ] ] ) :
mcsb_name = ' - %s %s %s %s shuffle ' % (
' M ' if world . mapshuffle [ player ] else ' ' , ' C ' if world . compassshuffle [ player ] else ' ' ,
2020-09-02 15:06:36 -07:00
' U ' if world . keyshuffle [ player ] == " universal " else ' S ' if world . keyshuffle [ player ] else ' ' ,
' B ' if world . bigkeyshuffle [ player ] else ' ' )
2020-06-09 21:52:46 +02:00
outfilepname = f ' _T { team + 1 } ' if world . teams > 1 else ' '
2020-08-02 22:11:52 +02:00
outfilepname + = f ' _P { player } '
2020-09-02 15:06:36 -07:00
outfilepname + = f " _ { world . player_names [ player ] [ team ] . replace ( ' ' , ' _ ' ) } " \
if world . player_names [ player ] [ team ] != ' Player %d ' % player else ' '
outfilestuffs = {
" logic " : world . logic [ player ] , # 0
" difficulty " : world . difficulty [ player ] , # 1
" difficulty_adjustments " : world . difficulty_adjustments [ player ] , # 2
" mode " : world . mode [ player ] , # 3
" goal " : world . goal [ player ] , # 4
" timer " : str ( world . timer [ player ] ) , # 5
" shuffle " : world . shuffle [ player ] , # 6
" algorithm " : world . algorithm , # 7
" mscb " : mcsb_name , # 8
" retro " : world . retro [ player ] , # 9
" progressive " : world . progressive , # A
" hints " : ' True ' if world . hints [ player ] else ' False ' # B
}
# 0 1 2 3 4 5 6 7 8 9 A B
outfilesuffix = ( ' _ %s _ %s - %s - %s - %s %s _ %s - %s %s %s %s %s ' % (
# 0 1 2 3 4 5 6 7 8 9 A B C
# _noglitches_normal-normal-open-ganon-ohko_simple-balanced-keysanity-retro-prog_random-nohints
# _noglitches_normal-normal-open-ganon _simple-balanced-keysanity-retro
# _noglitches_normal-normal-open-ganon _simple-balanced-keysanity -prog_random
# _noglitches_normal-normal-open-ganon _simple-balanced-keysanity -nohints
outfilestuffs [ " logic " ] , # 0
outfilestuffs [ " difficulty " ] , # 1
outfilestuffs [ " difficulty_adjustments " ] , # 2
outfilestuffs [ " mode " ] , # 3
outfilestuffs [ " goal " ] , # 4
" " if outfilestuffs [ " timer " ] in [ ' False ' , ' none ' , ' display ' ] else " - " + outfilestuffs [ " timer " ] , # 5
outfilestuffs [ " shuffle " ] , # 6
outfilestuffs [ " algorithm " ] , # 7
outfilestuffs [ " mscb " ] , # 8
" -retro " if outfilestuffs [ " retro " ] == " True " else " " , # 9
" -prog_ " + outfilestuffs [ " progressive " ] if outfilestuffs [ " progressive " ] in [ ' off ' , ' random ' ] else " " , # A
" -nohints " if not outfilestuffs [ " hints " ] == " True " else " " ) # B
) if not args . outputname else ' '
2020-06-09 21:52:46 +02:00
rompath = output_path ( f ' { outfilebase } { outfilepname } { outfilesuffix } .sfc ' )
2020-08-16 11:13:50 +02:00
rom . write_to_file ( rompath , hide_enemizer = True )
2020-06-09 21:52:46 +02:00
if args . create_diff :
Patch . create_patch_file ( rompath )
2020-07-14 04:48:56 +02:00
return player , team , bytes ( rom . name ) . decode ( )
2020-03-06 23:08:46 +01:00
2020-08-21 18:35:48 +02:00
pool = concurrent . futures . ThreadPoolExecutor ( )
2020-08-23 12:06:00 +02:00
multidata_task = None
2020-03-06 23:08:46 +01:00
if not args . suppress_rom :
2020-08-21 18:35:48 +02:00
rom_futures = [ ]
for team in range ( world . teams ) :
for player in range ( 1 , world . players + 1 ) :
rom_futures . append ( pool . submit ( _gen_rom , team , player ) )
2020-05-18 05:40:36 +02:00
def get_entrance_to_region ( region : Region ) :
for entrance in region . entrances :
if entrance . parent_region . type in ( RegionType . DarkWorld , RegionType . LightWorld ) :
return entrance
for entrance in region . entrances : # BFS might be better here, trying DFS for now.
return get_entrance_to_region ( entrance . parent_region )
# collect ER hint info
er_hint_data = { player : { } for player in range ( 1 , world . players + 1 ) if world . shuffle [ player ] != " vanilla " }
from Regions import RegionType
for region in world . regions :
if region . player in er_hint_data and region . locations :
main_entrance = get_entrance_to_region ( region )
for location in region . locations :
if type ( location . address ) == int : # skips events and crystals
2020-06-20 12:22:50 +02:00
if lookup_vanilla_location_to_entrance [ location . address ] != main_entrance . name :
er_hint_data [ region . player ] [ location . address ] = main_entrance . name
2020-05-18 05:40:36 +02:00
2020-10-29 15:18:21 -07:00
ordered_areas = ( ' Light World ' , ' Dark World ' , ' Hyrule Castle ' , ' Agahnims Tower ' , ' Eastern Palace ' , ' Desert Palace ' ,
' Tower of Hera ' , ' Palace of Darkness ' , ' Swamp Palace ' , ' Skull Woods ' , ' Thieves Town ' , ' Ice Palace ' ,
' Misery Mire ' , ' Turtle Rock ' , ' Ganons Tower ' , " Total " )
checks_in_area = { player : { area : list ( ) for area in ordered_areas }
for player in range ( 1 , world . players + 1 ) }
for player in range ( 1 , world . players + 1 ) :
checks_in_area [ player ] [ " Total " ] = 0
for location in [ loc for loc in world . get_filled_locations ( ) if type ( loc . address ) is int ] :
2020-10-29 15:32:05 -07:00
main_entrance = get_entrance_to_region ( location . parent_region )
2020-10-29 15:18:21 -07:00
if location . parent_region . dungeon :
checks_in_area [ location . player ] [ location . parent_region . dungeon . name ] . append ( location . address )
elif main_entrance . parent_region . type == RegionType . LightWorld :
checks_in_area [ location . player ] [ " Light World " ] . append ( location . address )
elif main_entrance . parent_region . type == RegionType . DarkWorld :
checks_in_area [ location . player ] [ " Dark World " ] . append ( location . address )
checks_in_area [ location . player ] [ " Total " ] + = 1
2020-06-23 23:50:37 +02:00
precollected_items = [ [ ] for player in range ( world . players ) ]
for item in world . precollected_items :
precollected_items [ item . player - 1 ] . append ( item . code )
2020-08-21 18:35:48 +02:00
def write_multidata ( roms ) :
for future in roms :
rom_name = future . result ( )
rom_names . append ( rom_name )
2020-10-21 02:02:13 -07:00
multidatatags = [ " ER " ]
if args . race :
multidatatags . append ( " Race " )
if args . create_spoiler :
multidatatags . append ( " Spoiler " )
if not args . skip_playthrough :
multidatatags . append ( " Play through " )
2020-08-21 18:35:48 +02:00
multidata = zlib . compress ( json . dumps ( { " names " : parsed_names ,
# backwards compat for < 2.4.1
" roms " : [ ( slot , team , list ( name . encode ( ) ) )
for ( slot , team , name ) in rom_names ] ,
" rom_strings " : rom_names ,
" remote_items " : [ player for player in range ( 1 , world . players + 1 ) if
world . remote_items [ player ] ] ,
" locations " : [ ( ( location . address , location . player ) ,
( location . item . code , location . item . player ) )
for location in world . get_filled_locations ( ) if
type ( location . address ) is int ] ,
2020-10-29 15:18:21 -07:00
" checks_in_area " : checks_in_area ,
2020-08-21 18:35:48 +02:00
" server_options " : get_options ( ) [ " server_options " ] ,
" er_hint_data " : er_hint_data ,
" precollected_items " : precollected_items ,
" version " : _version_tuple ,
2020-10-21 02:02:13 -07:00
" tags " : multidatatags
2020-08-21 18:35:48 +02:00
} ) . encode ( " utf-8 " ) , 9 )
with open ( output_path ( ' %s .multidata ' % outfilebase ) , ' wb ' ) as f :
f . write ( multidata )
2020-08-23 12:06:00 +02:00
multidata_task = pool . submit ( write_multidata , rom_futures )
2017-05-20 14:07:40 +02:00
2019-04-18 11:23:24 +02:00
if not args . skip_playthrough :
logger . info ( ' Calculating playthrough. ' )
create_playthrough ( world )
2020-08-23 12:06:00 +02:00
if multidata_task :
multidata_task . result ( ) # retrieve exception if one exists
2020-08-21 18:35:48 +02:00
pool . shutdown ( ) # wait for all queued tasks to complete
if args . create_spoiler : # needs spoiler.hashes to be filled, that depend on rom_futures being done
2019-04-18 11:23:24 +02:00
world . spoiler . to_file ( output_path ( ' %s _Spoiler.txt ' % outfilebase ) )
2020-08-21 18:35:48 +02:00
logger . info ( ' Done. Enjoy. Total Time: %s ' , time . perf_counter ( ) - start )
2017-05-15 20:28:04 +02:00
return world
2018-03-22 23:18:40 -04:00
2020-05-09 10:00:41 +10:00
def copy_world ( world ) :
# ToDo: Not good yet
ret = World ( world . players , world . shuffle , world . logic , world . mode , world . swords , world . difficulty , world . difficulty_adjustments , world . timer , world . progressive , world . goal , world . algorithm , world . accessibility , world . shuffle_ganon , world . retro , world . custom , world . customitemarray , world . hints )
ret . teams = world . teams
ret . player_names = copy . deepcopy ( world . player_names )
ret . remote_items = world . remote_items . copy ( )
ret . required_medallions = world . required_medallions . copy ( )
ret . swamp_patch_required = world . swamp_patch_required . copy ( )
ret . ganon_at_pyramid = world . ganon_at_pyramid . copy ( )
ret . powder_patch_required = world . powder_patch_required . copy ( )
ret . ganonstower_vanilla = world . ganonstower_vanilla . copy ( )
ret . treasure_hunt_count = world . treasure_hunt_count . copy ( )
ret . treasure_hunt_icon = world . treasure_hunt_icon . copy ( )
ret . sewer_light_cone = world . sewer_light_cone . copy ( )
ret . light_world_light_cone = world . light_world_light_cone
ret . dark_world_light_cone = world . dark_world_light_cone
ret . seed = world . seed
ret . can_access_trock_eyebridge = world . can_access_trock_eyebridge . copy ( )
ret . can_access_trock_front = world . can_access_trock_front . copy ( )
ret . can_access_trock_big_chest = world . can_access_trock_big_chest . copy ( )
ret . can_access_trock_middle = world . can_access_trock_middle . copy ( )
ret . can_take_damage = world . can_take_damage
ret . difficulty_requirements = world . difficulty_requirements . copy ( )
ret . fix_fake_world = world . fix_fake_world . copy ( )
ret . mapshuffle = world . mapshuffle . copy ( )
ret . compassshuffle = world . compassshuffle . copy ( )
ret . keyshuffle = world . keyshuffle . copy ( )
ret . bigkeyshuffle = world . bigkeyshuffle . copy ( )
ret . crystals_needed_for_ganon = world . crystals_needed_for_ganon . copy ( )
ret . crystals_needed_for_gt = world . crystals_needed_for_gt . copy ( )
ret . open_pyramid = world . open_pyramid . copy ( )
ret . boss_shuffle = world . boss_shuffle . copy ( )
ret . enemy_shuffle = world . enemy_shuffle . copy ( )
ret . enemy_health = world . enemy_health . copy ( )
ret . enemy_damage = world . enemy_damage . copy ( )
ret . beemizer = world . beemizer . copy ( )
ret . timer = world . timer . copy ( )
ret . shufflepots = world . shufflepots . copy ( )
2020-10-07 19:51:46 +02:00
ret . shuffle_prizes = world . shuffle_prizes . copy ( )
ret . dark_room_logic = world . dark_room_logic . copy ( )
ret . restrict_dungeon_item_on_boss = world . restrict_dungeon_item_on_boss . copy ( )
2020-05-10 14:56:52 +10:00
for player in range ( 1 , world . players + 1 ) :
if world . mode [ player ] != ' inverted ' :
create_regions ( ret , player )
else :
create_inverted_regions ( ret , player )
create_shops ( ret , player )
create_dungeons ( ret , player )
copy_dynamic_regions_and_locations ( world , ret )
# copy bosses
for dungeon in world . dungeons :
for level , boss in dungeon . bosses . items ( ) :
ret . get_dungeon ( dungeon . name , dungeon . player ) . bosses [ level ] = boss
for shop in world . shops :
copied_shop = ret . get_region ( shop . region . name , shop . region . player ) . shop
copied_shop . inventory = copy . copy ( shop . inventory )
# connect copied world
for region in world . regions :
copied_region = ret . get_region ( region . name , region . player )
copied_region . is_light_world = region . is_light_world
copied_region . is_dark_world = region . is_dark_world
for exit in copied_region . exits :
old_connection = world . get_entrance ( exit . name , exit . player ) . connected_region
exit . connect ( ret . get_region ( old_connection . name , old_connection . player ) )
# fill locations
for location in world . get_locations ( ) :
if location . item is not None :
item = Item ( location . item . name , location . item . advancement , location . item . priority , location . item . type , player = location . item . player )
ret . get_location ( location . name , location . player ) . item = item
item . location = ret . get_location ( location . name , location . player )
item . world = ret
if location . event :
ret . get_location ( location . name , location . player ) . event = True
if location . locked :
ret . get_location ( location . name , location . player ) . locked = True
# copy remaining itempool. No item in itempool should have an assigned location
for item in world . itempool :
ret . itempool . append ( Item ( item . name , item . advancement , item . priority , item . type , player = item . player ) )
for item in world . precollected_items :
ret . push_precollected ( ItemFactory ( item . name , item . player ) )
# copy progress items in state
2020-05-09 10:00:41 +10:00
ret . state . prog_items = world . state . prog_items . copy ( )
ret . state . stale = { player : True for player in range ( 1 , world . players + 1 ) }
2020-05-10 14:56:52 +10:00
for player in range ( 1 , world . players + 1 ) :
set_rules ( ret , player )
2020-05-09 10:00:41 +10:00
return ret
2020-05-10 14:56:52 +10:00
def copy_dynamic_regions_and_locations ( world , ret ) :
for region in world . dynamic_regions :
new_reg = Region ( region . name , region . type , region . hint_text , region . player )
ret . regions . append ( new_reg )
ret . initialize_regions ( [ new_reg ] )
ret . dynamic_regions . append ( new_reg )
# Note: ideally exits should be copied here, but the current use case (Take anys) do not require this
if region . shop :
2020-08-25 14:31:20 +02:00
new_reg . shop = region . shop . __class__ ( new_reg , region . shop . room_id , region . shop . shopkeeper_config ,
region . shop . custom , region . shop . locked )
2020-05-10 14:56:52 +10:00
ret . shops . append ( new_reg . shop )
for location in world . dynamic_locations :
new_reg = ret . get_region ( location . parent_region . name , location . parent_region . player )
new_loc = Location ( location . player , location . name , location . address , location . crystal , location . hint_text , new_reg )
# todo: this is potentially dangerous. later refactor so we
# can apply dynamic region rules on top of copied world like other rules
new_loc . access_rule = location . access_rule
new_loc . always_allow = location . always_allow
new_loc . item_rule = location . item_rule
new_reg . locations . append ( new_loc )
2020-08-27 04:05:11 +02:00
ret . clear_location_cache ( )
2020-05-10 14:56:52 +10:00
2017-05-16 21:23:47 +02:00
def create_playthrough ( world ) :
# create a copy as we will modify it
2017-06-04 16:15:59 +02:00
old_world = world
2020-05-09 10:00:41 +10:00
world = copy_world ( world )
2017-05-26 09:53:34 +02:00
2017-06-23 22:15:29 +02:00
# if we only check for beatable, we can do this sanity check first before writing down spheres
2019-12-17 12:14:29 +01:00
if not world . can_beat_game ( ) :
2017-06-23 22:15:29 +02:00
raise RuntimeError ( ' Cannot beat game. Something went terribly wrong here! ' )
2017-05-16 21:23:47 +02:00
# get locations containing progress items
2018-01-02 20:01:16 -05:00
prog_locations = [ location for location in world . get_filled_locations ( ) if location . item . advancement ]
2018-01-01 15:55:13 -05:00
state_cache = [ None ]
2017-05-16 21:23:47 +02:00
collection_spheres = [ ]
state = CollectionState ( world )
sphere_candidates = list ( prog_locations )
2020-08-14 00:34:41 +02:00
logging . debug ( ' Building up collection spheres. ' )
2017-05-16 21:23:47 +02:00
while sphere_candidates :
2019-12-13 22:37:52 +01:00
state . sweep_for_events ( key_only = True )
2017-06-24 11:11:56 +02:00
2017-05-16 21:23:47 +02:00
sphere = [ ]
# build up spheres of collection radius. Everything in each sphere is independent from each other in dependencies and only depends on lower spheres
for location in sphere_candidates :
if state . can_reach ( location ) :
sphere . append ( location )
for location in sphere :
sphere_candidates . remove ( location )
2018-01-01 15:55:13 -05:00
state . collect ( location . item , True , location )
2017-06-17 14:40:37 +02:00
2017-05-16 21:23:47 +02:00
collection_spheres . append ( sphere )
2018-01-01 15:55:13 -05:00
state_cache . append ( state . copy ( ) )
2017-05-26 09:55:24 +02:00
2020-08-14 00:34:41 +02:00
logging . debug ( ' Calculated sphere %i , containing %i of %i progress items. ' , len ( collection_spheres ) , len ( sphere ) ,
len ( prog_locations ) )
2017-05-26 09:55:24 +02:00
if not sphere :
2020-08-14 00:34:41 +02:00
logging . debug ( ' The following items could not be reached: %s ' , [ ' %s (Player %d ) at %s (Player %d ) ' % (
location . item . name , location . item . player , location . name , location . player ) for location in
sphere_candidates ] )
2019-12-17 12:14:29 +01:00
if any ( [ world . accessibility [ location . item . player ] != ' none ' for location in sphere_candidates ] ) :
2020-08-14 00:34:41 +02:00
raise RuntimeError ( f ' Not all progression items reachable ( { sphere_candidates } ). '
f ' Something went terribly wrong here. ' )
2017-06-23 22:15:29 +02:00
else :
2019-12-21 13:33:07 +01:00
old_world . spoiler . unreachables = sphere_candidates . copy ( )
2017-06-23 22:15:29 +02:00
break
2017-05-26 09:55:24 +02:00
2017-05-16 21:23:47 +02:00
# in the second phase, we cull each sphere such that the game is still beatable, reducing each range of influence to the bare minimum required inside it
2018-01-01 15:55:13 -05:00
for num , sphere in reversed ( list ( enumerate ( collection_spheres ) ) ) :
2017-05-16 21:23:47 +02:00
to_delete = [ ]
for location in sphere :
# we remove the item at location and check if game is still beatable
2019-04-18 11:23:24 +02:00
logging . getLogger ( ' ' ) . debug ( ' Checking if %s (Player %d ) is required to beat the game. ' , location . item . name , location . item . player )
2017-05-16 21:23:47 +02:00
old_item = location . item
location . item = None
2019-07-11 00:12:09 -04:00
if world . can_beat_game ( state_cache [ num ] ) :
2017-05-16 21:23:47 +02:00
to_delete . append ( location )
else :
# still required, got to keep it around
location . item = old_item
# cull entries in spheres for spoiler walkthrough at end
for location in to_delete :
sphere . remove ( location )
2020-01-09 08:31:49 +01:00
# second phase, sphere 0
for item in [ i for i in world . precollected_items if i . advancement ] :
logging . getLogger ( ' ' ) . debug ( ' Checking if %s (Player %d ) is required to beat the game. ' , item . name , item . player )
world . precollected_items . remove ( item )
world . state . remove ( item )
if not world . can_beat_game ( ) :
world . push_precollected ( item )
2018-01-06 14:25:49 -05:00
# we are now down to just the required progress items in collection_spheres. Unfortunately
# the previous pruning stage could potentially have made certain items dependant on others
# in the same or later sphere (because the location had 2 ways to access but the item originally
# used to access it was deemed not required.) So we need to do one final sphere collection pass
# to build up the correct spheres
required_locations = [ item for sphere in collection_spheres for item in sphere ]
state = CollectionState ( world )
collection_spheres = [ ]
while required_locations :
2019-12-13 22:37:52 +01:00
state . sweep_for_events ( key_only = True )
2018-01-06 14:25:49 -05:00
sphere = list ( filter ( state . can_reach , required_locations ) )
for location in sphere :
required_locations . remove ( location )
state . collect ( location . item , True , location )
collection_spheres . append ( sphere )
logging . getLogger ( ' ' ) . debug ( ' Calculated final sphere %i , containing %i of %i progress items. ' , len ( collection_spheres ) , len ( sphere ) , len ( required_locations ) )
if not sphere :
raise RuntimeError ( ' Not all required items reachable. Something went terribly wrong here. ' )
2017-05-16 21:23:47 +02:00
2017-06-04 16:15:59 +02:00
# store the required locations for statistical analysis
2019-04-18 11:23:24 +02:00
old_world . required_locations = [ ( location . name , location . player ) for sphere in collection_spheres for location in sphere ]
2017-06-04 16:15:59 +02:00
2018-01-01 15:55:13 -05:00
def flist_to_iter ( node ) :
while node :
value , node = node
yield value
2018-01-06 16:25:14 -05:00
def get_path ( state , region ) :
reversed_path_as_flist = state . path . get ( region , ( region , None ) )
string_path_flat = reversed ( list ( map ( str , flist_to_iter ( reversed_path_as_flist ) ) ) )
# Now we combine the flat string list into (region, exit) pairs
pathsiter = iter ( string_path_flat )
pathpairs = zip_longest ( pathsiter , pathsiter )
return list ( pathpairs )
2019-04-18 11:23:24 +02:00
old_world . spoiler . paths = dict ( )
for player in range ( 1 , world . players + 1 ) :
old_world . spoiler . paths . update ( { str ( location ) : get_path ( state , location . parent_region ) for sphere in collection_spheres for location in sphere if location . player == player } )
for _ , path in dict ( old_world . spoiler . paths ) . items ( ) :
if any ( exit == ' Pyramid Fairy ' for ( _ , exit ) in path ) :
2019-12-16 16:54:46 +01:00
if world . mode [ player ] != ' inverted ' :
2019-07-27 09:13:13 -04:00
old_world . spoiler . paths [ str ( world . get_region ( ' Big Bomb Shop ' , player ) ) ] = get_path ( state , world . get_region ( ' Big Bomb Shop ' , player ) )
else :
old_world . spoiler . paths [ str ( world . get_region ( ' Inverted Big Bomb Shop ' , player ) ) ] = get_path ( state , world . get_region ( ' Inverted Big Bomb Shop ' , player ) )
2018-01-01 15:55:13 -05:00
2017-05-16 21:23:47 +02:00
# we can finally output our playthrough
2020-01-10 07:02:44 +01:00
old_world . spoiler . playthrough = OrderedDict ( [ ( " 0 " , [ str ( item ) for item in world . precollected_items if item . advancement ] ) ] )
2020-01-09 08:31:49 +01:00
for i , sphere in enumerate ( collection_spheres ) :
old_world . spoiler . playthrough [ str ( i + 1 ) ] = { str ( location ) : str ( location . item ) for location in sphere }