| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | import argparse | 
					
						
							|  |  |  | import asyncio | 
					
						
							|  |  |  | import json | 
					
						
							|  |  |  | import logging | 
					
						
							| 
									
										
										
										
											2020-01-10 22:43:01 +01:00
										 |  |  | import urllib.parse | 
					
						
							| 
									
										
										
										
											2020-02-02 06:20:08 +01:00
										 |  |  | import atexit | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  | import time | 
					
						
							| 
									
										
										
										
											2020-03-06 01:27:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | from Utils import get_item_name_from_id, get_location_name_from_address, ReceivedItem | 
					
						
							| 
									
										
										
										
											2020-02-02 06:20:08 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | exit_func = atexit.register(input, "Press enter to close.") | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 15:45:52 +01:00
										 |  |  | import ModuleUpdate | 
					
						
							| 
									
										
										
										
											2020-03-06 01:27:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 15:45:52 +01:00
										 |  |  | ModuleUpdate.update() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 15:04:39 +01:00
										 |  |  | import colorama | 
					
						
							| 
									
										
										
										
											2020-01-18 15:45:52 +01:00
										 |  |  | import websockets | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  | import prompt_toolkit | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  | import typing | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  | from prompt_toolkit.patch_stdout import patch_stdout | 
					
						
							| 
									
										
										
										
											2020-01-18 15:45:52 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | import Regions | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | import Utils | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 19:42:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | class Context: | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  |     def __init__(self, snes_address, server_address, password, found_items): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         self.snes_address = snes_address | 
					
						
							|  |  |  |         self.server_address = server_address | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.exit_event = asyncio.Event() | 
					
						
							| 
									
										
										
										
											2020-01-18 11:28:08 +01:00
										 |  |  |         self.watcher_event = asyncio.Event() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.input_queue = asyncio.Queue() | 
					
						
							|  |  |  |         self.input_requests = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.snes_socket = None | 
					
						
							|  |  |  |         self.snes_state = SNES_DISCONNECTED | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |         self.snes_attached_device = None | 
					
						
							|  |  |  |         self.snes_reconnect_address = None | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         self.snes_recv_queue = asyncio.Queue() | 
					
						
							|  |  |  |         self.snes_request_lock = asyncio.Lock() | 
					
						
							|  |  |  |         self.is_sd2snes = False | 
					
						
							|  |  |  |         self.snes_write_buffer = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.server_task = None | 
					
						
							|  |  |  |         self.socket = None | 
					
						
							|  |  |  |         self.password = password | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |         self.server_version = (0, 0, 0) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         self.team = None | 
					
						
							|  |  |  |         self.slot = None | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |         self.player_names: typing.Dict[int: str] = {} | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         self.locations_checked = set() | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |         self.locations_scouted = set() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         self.items_received = [] | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |         self.locations_info = {} | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         self.awaiting_rom = False | 
					
						
							|  |  |  |         self.rom = None | 
					
						
							|  |  |  |         self.auth = None | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  |         self.found_items = found_items | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |         self.finished_game = False | 
					
						
							|  |  |  |         self.slow_mode = False | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | color_codes = {'reset': 0, 'bold': 1, 'underline': 4, 'black': 30, 'red': 31, 'green': 32, 'yellow': 33, 'blue': 34, | 
					
						
							|  |  |  |                'magenta': 35, 'cyan': 36, 'white': 37, 'black_bg': 40, 'red_bg': 41, 'green_bg': 42, 'yellow_bg': 43, | 
					
						
							|  |  |  |                'blue_bg': 44, 'purple_bg': 45, 'cyan_bg': 46, 'white_bg': 47} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | def color_code(*args): | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     return '\033[' + ';'.join([str(color_codes[arg]) for arg in args]) + 'm' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | def color(text, *args): | 
					
						
							|  |  |  |     return color_code(*args) + text + color_code('reset') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:04:35 +01:00
										 |  |  | RECONNECT_DELAY = 5 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | ROM_START = 0x000000 | 
					
						
							|  |  |  | WRAM_START = 0xF50000 | 
					
						
							|  |  |  | WRAM_SIZE = 0x20000 | 
					
						
							|  |  |  | SRAM_START = 0xE00000 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ROMNAME_START = SRAM_START + 0x2000 | 
					
						
							|  |  |  | ROMNAME_SIZE = 0x15 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | INGAME_MODES = {0x07, 0x09, 0x0b} | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  | ENDGAME_MODES = {0x19, 0x1a} | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | SAVEDATA_START = WRAM_START + 0xF000 | 
					
						
							|  |  |  | SAVEDATA_SIZE = 0x500 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  | RECV_PROGRESS_ADDR = SAVEDATA_START + 0x4D0         # 2 bytes | 
					
						
							|  |  |  | RECV_ITEM_ADDR = SAVEDATA_START + 0x4D2             # 1 byte | 
					
						
							|  |  |  | RECV_ITEM_PLAYER_ADDR = SAVEDATA_START + 0x4D3      # 1 byte | 
					
						
							|  |  |  | ROOMID_ADDR = SAVEDATA_START + 0x4D4                # 2 bytes | 
					
						
							|  |  |  | ROOMDATA_ADDR = SAVEDATA_START + 0x4D6              # 1 byte | 
					
						
							|  |  |  | SCOUT_LOCATION_ADDR = SAVEDATA_START + 0x4D7        # 1 byte | 
					
						
							|  |  |  | SCOUTREPLY_LOCATION_ADDR = SAVEDATA_START + 0x4D8   # 1 byte | 
					
						
							|  |  |  | SCOUTREPLY_ITEM_ADDR = SAVEDATA_START + 0x4D9       # 1 byte | 
					
						
							|  |  |  | SCOUTREPLY_PLAYER_ADDR = SAVEDATA_START + 0x4DA     # 1 byte | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | location_table_uw = {"Blind's Hideout - Top": (0x11d, 0x10), | 
					
						
							|  |  |  |                      "Blind's Hideout - Left": (0x11d, 0x20), | 
					
						
							|  |  |  |                      "Blind's Hideout - Right": (0x11d, 0x40), | 
					
						
							|  |  |  |                      "Blind's Hideout - Far Left": (0x11d, 0x80), | 
					
						
							|  |  |  |                      "Blind's Hideout - Far Right": (0x11d, 0x100), | 
					
						
							|  |  |  |                      'Secret Passage': (0x55, 0x10), | 
					
						
							|  |  |  |                      'Waterfall Fairy - Left': (0x114, 0x10), | 
					
						
							|  |  |  |                      'Waterfall Fairy - Right': (0x114, 0x20), | 
					
						
							|  |  |  |                      "King's Tomb": (0x113, 0x10), | 
					
						
							|  |  |  |                      'Floodgate Chest': (0x10b, 0x10), | 
					
						
							|  |  |  |                      "Link's House": (0x104, 0x10), | 
					
						
							|  |  |  |                      'Kakariko Tavern': (0x103, 0x10), | 
					
						
							|  |  |  |                      'Chicken House': (0x108, 0x10), | 
					
						
							|  |  |  |                      "Aginah's Cave": (0x10a, 0x10), | 
					
						
							|  |  |  |                      "Sahasrahla's Hut - Left": (0x105, 0x10), | 
					
						
							|  |  |  |                      "Sahasrahla's Hut - Middle": (0x105, 0x20), | 
					
						
							|  |  |  |                      "Sahasrahla's Hut - Right": (0x105, 0x40), | 
					
						
							|  |  |  |                      'Kakariko Well - Top': (0x2f, 0x10), | 
					
						
							|  |  |  |                      'Kakariko Well - Left': (0x2f, 0x20), | 
					
						
							|  |  |  |                      'Kakariko Well - Middle': (0x2f, 0x40), | 
					
						
							|  |  |  |                      'Kakariko Well - Right': (0x2f, 0x80), | 
					
						
							|  |  |  |                      'Kakariko Well - Bottom': (0x2f, 0x100), | 
					
						
							|  |  |  |                      'Lost Woods Hideout': (0xe1, 0x200), | 
					
						
							|  |  |  |                      'Lumberjack Tree': (0xe2, 0x200), | 
					
						
							|  |  |  |                      'Cave 45': (0x11b, 0x400), | 
					
						
							|  |  |  |                      'Graveyard Cave': (0x11b, 0x200), | 
					
						
							|  |  |  |                      'Checkerboard Cave': (0x126, 0x200), | 
					
						
							|  |  |  |                      'Mini Moldorm Cave - Far Left': (0x123, 0x10), | 
					
						
							|  |  |  |                      'Mini Moldorm Cave - Left': (0x123, 0x20), | 
					
						
							|  |  |  |                      'Mini Moldorm Cave - Right': (0x123, 0x40), | 
					
						
							|  |  |  |                      'Mini Moldorm Cave - Far Right': (0x123, 0x80), | 
					
						
							|  |  |  |                      'Mini Moldorm Cave - Generous Guy': (0x123, 0x400), | 
					
						
							|  |  |  |                      'Ice Rod Cave': (0x120, 0x10), | 
					
						
							|  |  |  |                      'Bonk Rock Cave': (0x124, 0x10), | 
					
						
							|  |  |  |                      'Desert Palace - Big Chest': (0x73, 0x10), | 
					
						
							|  |  |  |                      'Desert Palace - Torch': (0x73, 0x400), | 
					
						
							|  |  |  |                      'Desert Palace - Map Chest': (0x74, 0x10), | 
					
						
							|  |  |  |                      'Desert Palace - Compass Chest': (0x85, 0x10), | 
					
						
							|  |  |  |                      'Desert Palace - Big Key Chest': (0x75, 0x10), | 
					
						
							|  |  |  |                      'Desert Palace - Boss': (0x33, 0x800), | 
					
						
							|  |  |  |                      'Eastern Palace - Compass Chest': (0xa8, 0x10), | 
					
						
							|  |  |  |                      'Eastern Palace - Big Chest': (0xa9, 0x10), | 
					
						
							|  |  |  |                      'Eastern Palace - Cannonball Chest': (0xb9, 0x10), | 
					
						
							|  |  |  |                      'Eastern Palace - Big Key Chest': (0xb8, 0x10), | 
					
						
							|  |  |  |                      'Eastern Palace - Map Chest': (0xaa, 0x10), | 
					
						
							|  |  |  |                      'Eastern Palace - Boss': (0xc8, 0x800), | 
					
						
							|  |  |  |                      'Hyrule Castle - Boomerang Chest': (0x71, 0x10), | 
					
						
							|  |  |  |                      'Hyrule Castle - Map Chest': (0x72, 0x10), | 
					
						
							|  |  |  |                      "Hyrule Castle - Zelda's Chest": (0x80, 0x10), | 
					
						
							|  |  |  |                      'Sewers - Dark Cross': (0x32, 0x10), | 
					
						
							|  |  |  |                      'Sewers - Secret Room - Left': (0x11, 0x10), | 
					
						
							|  |  |  |                      'Sewers - Secret Room - Middle': (0x11, 0x20), | 
					
						
							|  |  |  |                      'Sewers - Secret Room - Right': (0x11, 0x40), | 
					
						
							|  |  |  |                      'Sanctuary': (0x12, 0x10), | 
					
						
							|  |  |  |                      'Castle Tower - Room 03': (0xe0, 0x10), | 
					
						
							|  |  |  |                      'Castle Tower - Dark Maze': (0xd0, 0x10), | 
					
						
							|  |  |  |                      'Spectacle Rock Cave': (0xea, 0x400), | 
					
						
							|  |  |  |                      'Paradox Cave Lower - Far Left': (0xef, 0x10), | 
					
						
							|  |  |  |                      'Paradox Cave Lower - Left': (0xef, 0x20), | 
					
						
							|  |  |  |                      'Paradox Cave Lower - Right': (0xef, 0x40), | 
					
						
							|  |  |  |                      'Paradox Cave Lower - Far Right': (0xef, 0x80), | 
					
						
							|  |  |  |                      'Paradox Cave Lower - Middle': (0xef, 0x100), | 
					
						
							|  |  |  |                      'Paradox Cave Upper - Left': (0xff, 0x10), | 
					
						
							|  |  |  |                      'Paradox Cave Upper - Right': (0xff, 0x20), | 
					
						
							|  |  |  |                      'Spiral Cave': (0xfe, 0x10), | 
					
						
							|  |  |  |                      'Tower of Hera - Basement Cage': (0x87, 0x400), | 
					
						
							|  |  |  |                      'Tower of Hera - Map Chest': (0x77, 0x10), | 
					
						
							|  |  |  |                      'Tower of Hera - Big Key Chest': (0x87, 0x10), | 
					
						
							|  |  |  |                      'Tower of Hera - Compass Chest': (0x27, 0x20), | 
					
						
							|  |  |  |                      'Tower of Hera - Big Chest': (0x27, 0x10), | 
					
						
							|  |  |  |                      'Tower of Hera - Boss': (0x7, 0x800), | 
					
						
							|  |  |  |                      'Hype Cave - Top': (0x11e, 0x10), | 
					
						
							|  |  |  |                      'Hype Cave - Middle Right': (0x11e, 0x20), | 
					
						
							|  |  |  |                      'Hype Cave - Middle Left': (0x11e, 0x40), | 
					
						
							|  |  |  |                      'Hype Cave - Bottom': (0x11e, 0x80), | 
					
						
							|  |  |  |                      'Hype Cave - Generous Guy': (0x11e, 0x400), | 
					
						
							|  |  |  |                      'Peg Cave': (0x127, 0x400), | 
					
						
							|  |  |  |                      'Pyramid Fairy - Left': (0x116, 0x10), | 
					
						
							|  |  |  |                      'Pyramid Fairy - Right': (0x116, 0x20), | 
					
						
							|  |  |  |                      'Brewery': (0x106, 0x10), | 
					
						
							|  |  |  |                      'C-Shaped House': (0x11c, 0x10), | 
					
						
							|  |  |  |                      'Chest Game': (0x106, 0x400), | 
					
						
							|  |  |  |                      'Mire Shed - Left': (0x10d, 0x10), | 
					
						
							|  |  |  |                      'Mire Shed - Right': (0x10d, 0x20), | 
					
						
							|  |  |  |                      'Superbunny Cave - Top': (0xf8, 0x10), | 
					
						
							|  |  |  |                      'Superbunny Cave - Bottom': (0xf8, 0x20), | 
					
						
							|  |  |  |                      'Spike Cave': (0x117, 0x10), | 
					
						
							|  |  |  |                      'Hookshot Cave - Top Right': (0x3c, 0x10), | 
					
						
							|  |  |  |                      'Hookshot Cave - Top Left': (0x3c, 0x20), | 
					
						
							|  |  |  |                      'Hookshot Cave - Bottom Right': (0x3c, 0x80), | 
					
						
							|  |  |  |                      'Hookshot Cave - Bottom Left': (0x3c, 0x40), | 
					
						
							|  |  |  |                      'Mimic Cave': (0x10c, 0x10), | 
					
						
							|  |  |  |                      'Swamp Palace - Entrance': (0x28, 0x10), | 
					
						
							|  |  |  |                      'Swamp Palace - Map Chest': (0x37, 0x10), | 
					
						
							|  |  |  |                      'Swamp Palace - Big Chest': (0x36, 0x10), | 
					
						
							|  |  |  |                      'Swamp Palace - Compass Chest': (0x46, 0x10), | 
					
						
							|  |  |  |                      'Swamp Palace - Big Key Chest': (0x35, 0x10), | 
					
						
							|  |  |  |                      'Swamp Palace - West Chest': (0x34, 0x10), | 
					
						
							|  |  |  |                      'Swamp Palace - Flooded Room - Left': (0x76, 0x10), | 
					
						
							|  |  |  |                      'Swamp Palace - Flooded Room - Right': (0x76, 0x20), | 
					
						
							|  |  |  |                      'Swamp Palace - Waterfall Room': (0x66, 0x10), | 
					
						
							|  |  |  |                      'Swamp Palace - Boss': (0x6, 0x800), | 
					
						
							|  |  |  |                      "Thieves' Town - Big Key Chest": (0xdb, 0x20), | 
					
						
							|  |  |  |                      "Thieves' Town - Map Chest": (0xdb, 0x10), | 
					
						
							|  |  |  |                      "Thieves' Town - Compass Chest": (0xdc, 0x10), | 
					
						
							|  |  |  |                      "Thieves' Town - Ambush Chest": (0xcb, 0x10), | 
					
						
							|  |  |  |                      "Thieves' Town - Attic": (0x65, 0x10), | 
					
						
							|  |  |  |                      "Thieves' Town - Big Chest": (0x44, 0x10), | 
					
						
							|  |  |  |                      "Thieves' Town - Blind's Cell": (0x45, 0x10), | 
					
						
							|  |  |  |                      "Thieves' Town - Boss": (0xac, 0x800), | 
					
						
							|  |  |  |                      'Skull Woods - Compass Chest': (0x67, 0x10), | 
					
						
							|  |  |  |                      'Skull Woods - Map Chest': (0x58, 0x20), | 
					
						
							|  |  |  |                      'Skull Woods - Big Chest': (0x58, 0x10), | 
					
						
							|  |  |  |                      'Skull Woods - Pot Prison': (0x57, 0x20), | 
					
						
							|  |  |  |                      'Skull Woods - Pinball Room': (0x68, 0x10), | 
					
						
							|  |  |  |                      'Skull Woods - Big Key Chest': (0x57, 0x10), | 
					
						
							|  |  |  |                      'Skull Woods - Bridge Room': (0x59, 0x10), | 
					
						
							|  |  |  |                      'Skull Woods - Boss': (0x29, 0x800), | 
					
						
							|  |  |  |                      'Ice Palace - Compass Chest': (0x2e, 0x10), | 
					
						
							|  |  |  |                      'Ice Palace - Freezor Chest': (0x7e, 0x10), | 
					
						
							|  |  |  |                      'Ice Palace - Big Chest': (0x9e, 0x10), | 
					
						
							|  |  |  |                      'Ice Palace - Iced T Room': (0xae, 0x10), | 
					
						
							|  |  |  |                      'Ice Palace - Spike Room': (0x5f, 0x10), | 
					
						
							|  |  |  |                      'Ice Palace - Big Key Chest': (0x1f, 0x10), | 
					
						
							|  |  |  |                      'Ice Palace - Map Chest': (0x3f, 0x10), | 
					
						
							|  |  |  |                      'Ice Palace - Boss': (0xde, 0x800), | 
					
						
							|  |  |  |                      'Misery Mire - Big Chest': (0xc3, 0x10), | 
					
						
							|  |  |  |                      'Misery Mire - Map Chest': (0xc3, 0x20), | 
					
						
							|  |  |  |                      'Misery Mire - Main Lobby': (0xc2, 0x10), | 
					
						
							|  |  |  |                      'Misery Mire - Bridge Chest': (0xa2, 0x10), | 
					
						
							|  |  |  |                      'Misery Mire - Spike Chest': (0xb3, 0x10), | 
					
						
							|  |  |  |                      'Misery Mire - Compass Chest': (0xc1, 0x10), | 
					
						
							|  |  |  |                      'Misery Mire - Big Key Chest': (0xd1, 0x10), | 
					
						
							|  |  |  |                      'Misery Mire - Boss': (0x90, 0x800), | 
					
						
							|  |  |  |                      'Turtle Rock - Compass Chest': (0xd6, 0x10), | 
					
						
							|  |  |  |                      'Turtle Rock - Roller Room - Left': (0xb7, 0x10), | 
					
						
							|  |  |  |                      'Turtle Rock - Roller Room - Right': (0xb7, 0x20), | 
					
						
							|  |  |  |                      'Turtle Rock - Chain Chomps': (0xb6, 0x10), | 
					
						
							|  |  |  |                      'Turtle Rock - Big Key Chest': (0x14, 0x10), | 
					
						
							|  |  |  |                      'Turtle Rock - Big Chest': (0x24, 0x10), | 
					
						
							|  |  |  |                      'Turtle Rock - Crystaroller Room': (0x4, 0x10), | 
					
						
							|  |  |  |                      'Turtle Rock - Eye Bridge - Bottom Left': (0xd5, 0x80), | 
					
						
							|  |  |  |                      'Turtle Rock - Eye Bridge - Bottom Right': (0xd5, 0x40), | 
					
						
							|  |  |  |                      'Turtle Rock - Eye Bridge - Top Left': (0xd5, 0x20), | 
					
						
							|  |  |  |                      'Turtle Rock - Eye Bridge - Top Right': (0xd5, 0x10), | 
					
						
							|  |  |  |                      'Turtle Rock - Boss': (0xa4, 0x800), | 
					
						
							|  |  |  |                      'Palace of Darkness - Shooter Room': (0x9, 0x10), | 
					
						
							|  |  |  |                      'Palace of Darkness - The Arena - Bridge': (0x2a, 0x20), | 
					
						
							|  |  |  |                      'Palace of Darkness - Stalfos Basement': (0xa, 0x10), | 
					
						
							|  |  |  |                      'Palace of Darkness - Big Key Chest': (0x3a, 0x10), | 
					
						
							|  |  |  |                      'Palace of Darkness - The Arena - Ledge': (0x2a, 0x10), | 
					
						
							|  |  |  |                      'Palace of Darkness - Map Chest': (0x2b, 0x10), | 
					
						
							|  |  |  |                      'Palace of Darkness - Compass Chest': (0x1a, 0x20), | 
					
						
							|  |  |  |                      'Palace of Darkness - Dark Basement - Left': (0x6a, 0x10), | 
					
						
							|  |  |  |                      'Palace of Darkness - Dark Basement - Right': (0x6a, 0x20), | 
					
						
							|  |  |  |                      'Palace of Darkness - Dark Maze - Top': (0x19, 0x10), | 
					
						
							|  |  |  |                      'Palace of Darkness - Dark Maze - Bottom': (0x19, 0x20), | 
					
						
							|  |  |  |                      'Palace of Darkness - Big Chest': (0x1a, 0x10), | 
					
						
							|  |  |  |                      'Palace of Darkness - Harmless Hellway': (0x1a, 0x40), | 
					
						
							|  |  |  |                      'Palace of Darkness - Boss': (0x5a, 0x800), | 
					
						
							|  |  |  |                      "Ganons Tower - Bob's Torch": (0x8c, 0x400), | 
					
						
							|  |  |  |                      'Ganons Tower - Hope Room - Left': (0x8c, 0x20), | 
					
						
							|  |  |  |                      'Ganons Tower - Hope Room - Right': (0x8c, 0x40), | 
					
						
							|  |  |  |                      'Ganons Tower - Tile Room': (0x8d, 0x10), | 
					
						
							|  |  |  |                      'Ganons Tower - Compass Room - Top Left': (0x9d, 0x10), | 
					
						
							|  |  |  |                      'Ganons Tower - Compass Room - Top Right': (0x9d, 0x20), | 
					
						
							|  |  |  |                      'Ganons Tower - Compass Room - Bottom Left': (0x9d, 0x40), | 
					
						
							|  |  |  |                      'Ganons Tower - Compass Room - Bottom Right': (0x9d, 0x80), | 
					
						
							|  |  |  |                      'Ganons Tower - DMs Room - Top Left': (0x7b, 0x10), | 
					
						
							|  |  |  |                      'Ganons Tower - DMs Room - Top Right': (0x7b, 0x20), | 
					
						
							|  |  |  |                      'Ganons Tower - DMs Room - Bottom Left': (0x7b, 0x40), | 
					
						
							|  |  |  |                      'Ganons Tower - DMs Room - Bottom Right': (0x7b, 0x80), | 
					
						
							|  |  |  |                      'Ganons Tower - Map Chest': (0x8b, 0x10), | 
					
						
							|  |  |  |                      'Ganons Tower - Firesnake Room': (0x7d, 0x10), | 
					
						
							|  |  |  |                      'Ganons Tower - Randomizer Room - Top Left': (0x7c, 0x10), | 
					
						
							|  |  |  |                      'Ganons Tower - Randomizer Room - Top Right': (0x7c, 0x20), | 
					
						
							|  |  |  |                      'Ganons Tower - Randomizer Room - Bottom Left': (0x7c, 0x40), | 
					
						
							|  |  |  |                      'Ganons Tower - Randomizer Room - Bottom Right': (0x7c, 0x80), | 
					
						
							|  |  |  |                      "Ganons Tower - Bob's Chest": (0x8c, 0x80), | 
					
						
							|  |  |  |                      'Ganons Tower - Big Chest': (0x8c, 0x10), | 
					
						
							|  |  |  |                      'Ganons Tower - Big Key Room - Left': (0x1c, 0x20), | 
					
						
							|  |  |  |                      'Ganons Tower - Big Key Room - Right': (0x1c, 0x40), | 
					
						
							|  |  |  |                      'Ganons Tower - Big Key Chest': (0x1c, 0x10), | 
					
						
							|  |  |  |                      'Ganons Tower - Mini Helmasaur Room - Left': (0x3d, 0x10), | 
					
						
							|  |  |  |                      'Ganons Tower - Mini Helmasaur Room - Right': (0x3d, 0x20), | 
					
						
							|  |  |  |                      'Ganons Tower - Pre-Moldorm Chest': (0x3d, 0x40), | 
					
						
							|  |  |  |                      'Ganons Tower - Validation Chest': (0x4d, 0x10)} | 
					
						
							|  |  |  | location_table_npc = {'Mushroom': 0x1000, | 
					
						
							|  |  |  |                       'King Zora': 0x2, | 
					
						
							|  |  |  |                       'Sahasrahla': 0x10, | 
					
						
							|  |  |  |                       'Blacksmith': 0x400, | 
					
						
							|  |  |  |                       'Magic Bat': 0x8000, | 
					
						
							|  |  |  |                       'Sick Kid': 0x4, | 
					
						
							|  |  |  |                       'Library': 0x80, | 
					
						
							|  |  |  |                       'Potion Shop': 0x2000, | 
					
						
							|  |  |  |                       'Old Man': 0x1, | 
					
						
							|  |  |  |                       'Ether Tablet': 0x100, | 
					
						
							|  |  |  |                       'Catfish': 0x20, | 
					
						
							|  |  |  |                       'Stumpy': 0x8, | 
					
						
							|  |  |  |                       'Bombos Tablet': 0x200} | 
					
						
							|  |  |  | location_table_ow = {'Flute Spot': 0x2a, | 
					
						
							|  |  |  |                      'Sunken Treasure': 0x3b, | 
					
						
							|  |  |  |                      "Zora's Ledge": 0x81, | 
					
						
							|  |  |  |                      'Lake Hylia Island': 0x35, | 
					
						
							|  |  |  |                      'Maze Race': 0x28, | 
					
						
							|  |  |  |                      'Desert Ledge': 0x30, | 
					
						
							|  |  |  |                      'Master Sword Pedestal': 0x80, | 
					
						
							|  |  |  |                      'Spectacle Rock': 0x3, | 
					
						
							|  |  |  |                      'Pyramid': 0x5b, | 
					
						
							|  |  |  |                      'Digging Game': 0x68, | 
					
						
							|  |  |  |                      'Bumper Cave Ledge': 0x4a, | 
					
						
							|  |  |  |                      'Floating Island': 0x5} | 
					
						
							|  |  |  | location_table_misc = {'Bottle Merchant': (0x3c9, 0x2), | 
					
						
							|  |  |  |                        'Purple Chest': (0x3c9, 0x10), | 
					
						
							|  |  |  |                        "Link's Uncle": (0x3c6, 0x1), | 
					
						
							|  |  |  |                        'Hobo': (0x3c9, 0x1)} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SNES_DISCONNECTED = 0 | 
					
						
							|  |  |  | SNES_CONNECTING = 1 | 
					
						
							|  |  |  | SNES_CONNECTED = 2 | 
					
						
							|  |  |  | SNES_ATTACHED = 3 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  | async def snes_connect(ctx : Context, address): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     if ctx.snes_socket is not None: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.error('Already connected to snes') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ctx.snes_state = SNES_CONNECTING | 
					
						
							|  |  |  |     recv_task = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |     address = f"ws://{address}" if "://" not in address else address | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |     logging.info("Connecting to QUsb2snes at %s ..." % address) | 
					
						
							| 
									
										
										
										
											2020-03-07 00:07:32 +01:00
										 |  |  |     seen_problems = set() | 
					
						
							|  |  |  |     while ctx.snes_state == SNES_CONNECTING: | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             ctx.snes_socket = await websockets.connect(address, ping_timeout=None, ping_interval=None) | 
					
						
							|  |  |  |         except Exception as e: | 
					
						
							|  |  |  |             problem = "%s" % e | 
					
						
							|  |  |  |             # only tell the user about new problems, otherwise silently lay in wait for a working connection | 
					
						
							|  |  |  |             if problem not in seen_problems: | 
					
						
							|  |  |  |                 seen_problems.add(problem) | 
					
						
							|  |  |  |                 logging.error(f"Error connecting to QUsb2snes ({problem})") | 
					
						
							| 
									
										
										
										
											2020-04-07 04:18:26 +02:00
										 |  |  |                 if len(seen_problems) == 1: | 
					
						
							| 
									
										
										
										
											2020-04-12 04:38:57 +02:00
										 |  |  |                     # this is the first problem. Let's try launching QUsb2snes if it isn't already running | 
					
						
							| 
									
										
										
										
											2020-04-07 04:18:26 +02:00
										 |  |  |                     qusb2snes_path = Utils.get_options()["general_options"]["qusb2snes"] | 
					
						
							| 
									
										
										
										
											2020-04-12 04:44:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 04:18:26 +02:00
										 |  |  |                     import os | 
					
						
							| 
									
										
										
										
											2020-04-12 04:44:03 +02:00
										 |  |  |                     if not os.path.isfile(qusb2snes_path): | 
					
						
							| 
									
										
										
										
											2020-04-12 04:38:57 +02:00
										 |  |  |                         qusb2snes_path = Utils.local_path(qusb2snes_path) | 
					
						
							| 
									
										
										
										
											2020-04-12 04:44:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-12 04:38:57 +02:00
										 |  |  |                     if os.path.isfile(qusb2snes_path): | 
					
						
							| 
									
										
										
										
											2020-04-12 04:44:03 +02:00
										 |  |  |                         logging.info(f"Attempting to start {qusb2snes_path}") | 
					
						
							| 
									
										
										
										
											2020-04-07 04:18:26 +02:00
										 |  |  |                         import subprocess | 
					
						
							|  |  |  |                         subprocess.Popen(qusb2snes_path, cwd=os.path.dirname(qusb2snes_path)) | 
					
						
							|  |  |  |                     else: | 
					
						
							| 
									
										
										
										
											2020-04-12 04:38:57 +02:00
										 |  |  |                         logging.info( | 
					
						
							|  |  |  |                             f"Attempt to start (Q)Usb2Snes was aborted as path {qusb2snes_path} was not found, please start it yourself if it is not running") | 
					
						
							| 
									
										
										
										
											2020-04-07 04:18:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 00:07:32 +01:00
										 |  |  |             await asyncio.sleep(1) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             ctx.snes_state = SNES_CONNECTED | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     try: | 
					
						
							|  |  |  |         DeviceList_Request = { | 
					
						
							| 
									
										
										
										
											2020-03-07 00:07:32 +01:00
										 |  |  |             "Opcode": "DeviceList", | 
					
						
							|  |  |  |             "Space": "SNES" | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         await ctx.snes_socket.send(json.dumps(DeviceList_Request)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         reply = json.loads(await ctx.snes_socket.recv()) | 
					
						
							|  |  |  |         devices = reply['Results'] if 'Results' in reply and len(reply['Results']) > 0 else None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not devices: | 
					
						
							| 
									
										
										
										
											2020-03-07 00:07:32 +01:00
										 |  |  |             logging.info('No device found, waiting for device. Run multibridge and connect it to QUSB2SNES.') | 
					
						
							|  |  |  |             while not devices: | 
					
						
							|  |  |  |                 await asyncio.sleep(1) | 
					
						
							|  |  |  |                 await ctx.snes_socket.send(json.dumps(DeviceList_Request)) | 
					
						
							|  |  |  |                 reply = json.loads(await ctx.snes_socket.recv()) | 
					
						
							|  |  |  |                 devices = reply['Results'] if 'Results' in reply and len(reply['Results']) > 0 else None | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.info("Available devices:") | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         for id, device in enumerate(devices): | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.info("[%d] %s" % (id + 1, device)) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         device = None | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |         if len(devices) == 1: | 
					
						
							|  |  |  |             device = devices[0] | 
					
						
							|  |  |  |         elif ctx.snes_reconnect_address: | 
					
						
							|  |  |  |             if ctx.snes_attached_device[1] in devices: | 
					
						
							|  |  |  |                 device = ctx.snes_attached_device[1] | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 device = devices[ctx.snes_attached_device[0]] | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             while True: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |                 logging.info("Select a device:") | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |                 choice = await console_input(ctx) | 
					
						
							|  |  |  |                 if choice is None: | 
					
						
							|  |  |  |                     raise Exception('Abort input') | 
					
						
							|  |  |  |                 if not choice.isdigit() or int(choice) < 1 or int(choice) > len(devices): | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |                     logging.warning("Invalid choice (%s)" % choice) | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |                 device = devices[int(choice) - 1] | 
					
						
							|  |  |  |                 break | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.info("Attaching to " + device) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         Attach_Request = { | 
					
						
							|  |  |  |             "Opcode" : "Attach", | 
					
						
							|  |  |  |             "Space" : "SNES", | 
					
						
							|  |  |  |             "Operands" : [device] | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         await ctx.snes_socket.send(json.dumps(Attach_Request)) | 
					
						
							|  |  |  |         ctx.snes_state = SNES_ATTACHED | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |         ctx.snes_attached_device = (devices.index(device), device) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if 'SD2SNES'.lower() in device.lower() or (len(device) == 4 and device[:3] == 'COM'): | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.info("SD2SNES Detected") | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             ctx.is_sd2snes = True | 
					
						
							|  |  |  |             await ctx.snes_socket.send(json.dumps({"Opcode" : "Info", "Space" : "SNES"})) | 
					
						
							|  |  |  |             reply = json.loads(await ctx.snes_socket.recv()) | 
					
						
							|  |  |  |             if reply and 'Results' in reply: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |                 logging.info(reply['Results']) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         else: | 
					
						
							|  |  |  |             ctx.is_sd2snes = False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |         ctx.snes_reconnect_address = address | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         recv_task = asyncio.create_task(snes_recv_loop(ctx)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     except Exception as e: | 
					
						
							|  |  |  |         if recv_task is not None: | 
					
						
							|  |  |  |             if not ctx.snes_socket.closed: | 
					
						
							|  |  |  |                 await ctx.snes_socket.close() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             if ctx.snes_socket is not None: | 
					
						
							|  |  |  |                 if not ctx.snes_socket.closed: | 
					
						
							|  |  |  |                     await ctx.snes_socket.close() | 
					
						
							|  |  |  |                 ctx.snes_socket = None | 
					
						
							|  |  |  |             ctx.snes_state = SNES_DISCONNECTED | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |         if not ctx.snes_reconnect_address: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.error("Error connecting to snes (%s)" % e) | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.error(f"Error connecting to snes, attempt again in {RECONNECT_DELAY}s") | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |             asyncio.create_task(snes_autoreconnect(ctx)) | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  | async def snes_autoreconnect(ctx: Context): | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  |     # unfortunately currently broken. See: https://github.com/prompt-toolkit/python-prompt-toolkit/issues/1033 | 
					
						
							|  |  |  |     # with prompt_toolkit.shortcuts.ProgressBar() as pb: | 
					
						
							|  |  |  |     #    for _ in pb(range(100)): | 
					
						
							|  |  |  |     #        await asyncio.sleep(RECONNECT_DELAY/100) | 
					
						
							|  |  |  |     await asyncio.sleep(RECONNECT_DELAY) | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |     if ctx.snes_reconnect_address and ctx.snes_socket is None: | 
					
						
							|  |  |  |         await snes_connect(ctx, ctx.snes_reconnect_address) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | async def snes_recv_loop(ctx : Context): | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         async for msg in ctx.snes_socket: | 
					
						
							|  |  |  |             ctx.snes_recv_queue.put_nowait(msg) | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.warning("Snes disconnected") | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     except Exception as e: | 
					
						
							| 
									
										
										
										
											2019-12-16 18:39:00 +01:00
										 |  |  |         if not isinstance(e, websockets.WebSocketException): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             logging.exception(e) | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.error("Lost connection to the snes, type /snes to reconnect") | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     finally: | 
					
						
							|  |  |  |         socket, ctx.snes_socket = ctx.snes_socket, None | 
					
						
							|  |  |  |         if socket is not None and not socket.closed: | 
					
						
							|  |  |  |             await socket.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ctx.snes_state = SNES_DISCONNECTED | 
					
						
							|  |  |  |         ctx.snes_recv_queue = asyncio.Queue() | 
					
						
							|  |  |  |         ctx.hud_message_queue = [] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         ctx.rom = None | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |         if ctx.snes_reconnect_address: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.info(f"...reconnecting in {RECONNECT_DELAY}s") | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |             asyncio.create_task(snes_autoreconnect(ctx)) | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | async def snes_read(ctx : Context, address, size): | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         await ctx.snes_request_lock.acquire() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ctx.snes_state != SNES_ATTACHED or ctx.snes_socket is None or not ctx.snes_socket.open or ctx.snes_socket.closed: | 
					
						
							|  |  |  |             return None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         GetAddress_Request = { | 
					
						
							|  |  |  |             "Opcode" : "GetAddress", | 
					
						
							|  |  |  |             "Space" : "SNES", | 
					
						
							|  |  |  |             "Operands" : [hex(address)[2:], hex(size)[2:]] | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             await ctx.snes_socket.send(json.dumps(GetAddress_Request)) | 
					
						
							|  |  |  |         except websockets.ConnectionClosed: | 
					
						
							|  |  |  |             return None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         data = bytes() | 
					
						
							|  |  |  |         while len(data) < size: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 data += await asyncio.wait_for(ctx.snes_recv_queue.get(), 5) | 
					
						
							|  |  |  |             except asyncio.TimeoutError: | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if len(data) != size: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.error('Error reading %s, requested %d bytes, received %d' % (hex(address), size, len(data))) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             if len(data): | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |                 logging.error(str(data)) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             if ctx.snes_socket is not None and not ctx.snes_socket.closed: | 
					
						
							|  |  |  |                 await ctx.snes_socket.close() | 
					
						
							|  |  |  |             return None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return data | 
					
						
							|  |  |  |     finally: | 
					
						
							|  |  |  |         ctx.snes_request_lock.release() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async def snes_write(ctx : Context, write_list): | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         await ctx.snes_request_lock.acquire() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ctx.snes_state != SNES_ATTACHED or ctx.snes_socket is None or not ctx.snes_socket.open or ctx.snes_socket.closed: | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         PutAddress_Request = { | 
					
						
							|  |  |  |             "Opcode" : "PutAddress", | 
					
						
							|  |  |  |             "Operands" : [] | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ctx.is_sd2snes: | 
					
						
							|  |  |  |             cmd = b'\x00\xE2\x20\x48\xEB\x48' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for address, data in write_list: | 
					
						
							|  |  |  |                 if (address < WRAM_START) or ((address + len(data)) > (WRAM_START + WRAM_SIZE)): | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |                     logging.error("SD2SNES: Write out of range %s (%d)" % (hex(address), len(data))) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |                     return False | 
					
						
							|  |  |  |                 for ptr, byte in enumerate(data, address + 0x7E0000 - WRAM_START): | 
					
						
							|  |  |  |                     cmd += b'\xA9' # LDA | 
					
						
							|  |  |  |                     cmd += bytes([byte]) | 
					
						
							|  |  |  |                     cmd += b'\x8F' # STA.l | 
					
						
							|  |  |  |                     cmd += bytes([ptr & 0xFF, (ptr >> 8) & 0xFF, (ptr >> 16) & 0xFF]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             cmd += b'\xA9\x00\x8F\x00\x2C\x00\x68\xEB\x68\x28\x6C\xEA\xFF\x08' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             PutAddress_Request['Space'] = 'CMD' | 
					
						
							|  |  |  |             PutAddress_Request['Operands'] = ["2C00", hex(len(cmd)-1)[2:], "2C00", "1"] | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 if ctx.snes_socket is not None: | 
					
						
							|  |  |  |                     await ctx.snes_socket.send(json.dumps(PutAddress_Request)) | 
					
						
							|  |  |  |                 if ctx.snes_socket is not None: | 
					
						
							|  |  |  |                     await ctx.snes_socket.send(cmd) | 
					
						
							|  |  |  |             except websockets.ConnectionClosed: | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             PutAddress_Request['Space'] = 'SNES' | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 #will pack those requests as soon as qusb2snes actually supports that for real | 
					
						
							|  |  |  |                 for address, data in write_list: | 
					
						
							|  |  |  |                     PutAddress_Request['Operands'] = [hex(address)[2:], hex(len(data))[2:]] | 
					
						
							|  |  |  |                     if ctx.snes_socket is not None: | 
					
						
							|  |  |  |                         await ctx.snes_socket.send(json.dumps(PutAddress_Request)) | 
					
						
							|  |  |  |                     if ctx.snes_socket is not None: | 
					
						
							|  |  |  |                         await ctx.snes_socket.send(data) | 
					
						
							|  |  |  |             except websockets.ConnectionClosed: | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return True | 
					
						
							|  |  |  |     finally: | 
					
						
							|  |  |  |         ctx.snes_request_lock.release() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def snes_buffered_write(ctx : Context, address, data): | 
					
						
							|  |  |  |     if len(ctx.snes_write_buffer) > 0 and (ctx.snes_write_buffer[-1][0] + len(ctx.snes_write_buffer[-1][1])) == address: | 
					
						
							|  |  |  |         ctx.snes_write_buffer[-1] = (ctx.snes_write_buffer[-1][0], ctx.snes_write_buffer[-1][1] + data) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         ctx.snes_write_buffer.append((address, data)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async def snes_flush_writes(ctx : Context): | 
					
						
							|  |  |  |     if not ctx.snes_write_buffer: | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await snes_write(ctx, ctx.snes_write_buffer) | 
					
						
							|  |  |  |     ctx.snes_write_buffer = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async def send_msgs(websocket, msgs): | 
					
						
							|  |  |  |     if not websocket or not websocket.open or websocket.closed: | 
					
						
							|  |  |  |         return | 
					
						
							| 
									
										
										
										
											2020-04-19 13:44:22 +02:00
										 |  |  |     await websocket.send(json.dumps(msgs)) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  | async def server_loop(ctx : Context, address = None): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     if ctx.socket is not None: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.error('Already connected') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         return | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 05:29:02 +02:00
										 |  |  |     if address is None:  # set through CLI or BMBP | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         address = ctx.server_address | 
					
						
							| 
									
										
										
										
											2020-04-24 05:29:02 +02:00
										 |  |  |     if address is None:  # see if this is an old connection | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2020-04-25 04:10:30 +02:00
										 |  |  |             address = Utils.persistent_load()["servers"]["default"] | 
					
						
							| 
									
										
										
										
											2020-04-24 05:29:02 +02:00
										 |  |  |         except Exception as e: | 
					
						
							|  |  |  |             logging.debug(f"Could not find cached server address. {e}") | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2020-04-25 04:10:30 +02:00
										 |  |  |             logging.info(f'Enter multiworld server address. Press enter to connect to {address}') | 
					
						
							|  |  |  |             text = await console_input(ctx) | 
					
						
							|  |  |  |             if text: | 
					
						
							|  |  |  |                 address = text | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |     while not address: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.info('Enter multiworld server address') | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         address = await console_input(ctx) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-25 04:10:30 +02:00
										 |  |  |     Utils.persistent_store("servers", "default", address) | 
					
						
							| 
									
										
										
										
											2020-04-24 05:29:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |     address = f"ws://{address}" if "://" not in address else address | 
					
						
							| 
									
										
										
										
											2020-01-10 22:43:01 +01:00
										 |  |  |     port = urllib.parse.urlparse(address).port or 38281 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |     logging.info('Connecting to multiworld server at %s' % address) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     try: | 
					
						
							| 
									
										
										
										
											2020-04-19 13:41:01 +02:00
										 |  |  |         ctx.socket = await websockets.connect(address, port=port, ping_timeout=None, ping_interval=None) | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.info('Connected') | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         ctx.server_address = address | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         async for data in ctx.socket: | 
					
						
							|  |  |  |             for msg in json.loads(data): | 
					
						
							|  |  |  |                 cmd, args = (msg[0], msg[1]) if len(msg) > 1 else (msg, None) | 
					
						
							|  |  |  |                 await process_server_cmd(ctx, cmd, args) | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.warning('Disconnected from multiworld server, type /connect to reconnect') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     except ConnectionRefusedError: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.error('Connection refused by the multiworld server') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     except (OSError, websockets.InvalidURI): | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.error('Failed to connect to the multiworld server') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     except Exception as e: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.error('Lost connection to the multiworld server, type /connect to reconnect') | 
					
						
							| 
									
										
										
										
											2019-12-16 18:39:00 +01:00
										 |  |  |         if not isinstance(e, websockets.WebSocketException): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             logging.exception(e) | 
					
						
							|  |  |  |     finally: | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         ctx.awaiting_rom = False | 
					
						
							|  |  |  |         ctx.auth = None | 
					
						
							|  |  |  |         ctx.items_received = [] | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |         ctx.locations_info = {} | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |         ctx.server_version = (0, 0, 0) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         socket, ctx.socket = ctx.socket, None | 
					
						
							|  |  |  |         if socket is not None and not socket.closed: | 
					
						
							|  |  |  |             await socket.close() | 
					
						
							|  |  |  |         ctx.server_task = None | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         if ctx.server_address: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.info(f"... reconnecting in {RECONNECT_DELAY}s") | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |             asyncio.create_task(server_autoreconnect(ctx)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  | async def server_autoreconnect(ctx: Context): | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  |     # unfortunately currently broken. See: https://github.com/prompt-toolkit/python-prompt-toolkit/issues/1033 | 
					
						
							|  |  |  |     # with prompt_toolkit.shortcuts.ProgressBar() as pb: | 
					
						
							|  |  |  |     #    for _ in pb(range(100)): | 
					
						
							|  |  |  |     #        await asyncio.sleep(RECONNECT_DELAY/100) | 
					
						
							|  |  |  |     await asyncio.sleep(RECONNECT_DELAY) | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |     if ctx.server_address and ctx.server_task is None: | 
					
						
							|  |  |  |         ctx.server_task = asyncio.create_task(server_loop(ctx)) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | async def process_server_cmd(ctx : Context, cmd, args): | 
					
						
							|  |  |  |     if cmd == 'RoomInfo': | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.info('--------------------------------') | 
					
						
							|  |  |  |         logging.info('Room Information:') | 
					
						
							|  |  |  |         logging.info('--------------------------------') | 
					
						
							| 
									
										
										
										
											2020-02-16 16:20:00 +01:00
										 |  |  |         version = args.get("version", "unknown Bonta Protocol") | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |         if isinstance(version, list): | 
					
						
							|  |  |  |             ctx.server_version = tuple(item for item in version) | 
					
						
							| 
									
										
										
										
											2020-02-16 16:20:00 +01:00
										 |  |  |             version = ".".join(str(item) for item in version) | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |         else: | 
					
						
							|  |  |  |             ctx.server_version = (0, 0, 0) | 
					
						
							| 
									
										
										
										
											2020-02-16 16:20:00 +01:00
										 |  |  |         logging.info(f'Server protocol version: {version}') | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |         if "tags" in args: | 
					
						
							|  |  |  |             logging.info("Server protocol tags: " + ", ".join(args["tags"])) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         if args['password']: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.info('Password required') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         if len(args['players']) < 1: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.info('No player connected') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2020-01-15 03:00:30 +01:00
										 |  |  |             args['players'].sort() | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |             current_team = -1 | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.info('Connected players:') | 
					
						
							| 
									
										
										
										
											2020-01-15 03:00:30 +01:00
										 |  |  |             for team, slot, name in args['players']: | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |                 if team != current_team: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |                     logging.info(f'  Team #{team + 1}') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |                     current_team = team | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |                 logging.info('    %s (Player %d)' % (name, slot)) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         await server_auth(ctx, args['password']) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     elif cmd == 'ConnectionRefused': | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         if 'InvalidPassword' in args: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.error('Invalid password') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             ctx.password = None | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |             await server_auth(ctx, True) | 
					
						
							|  |  |  |         if 'InvalidRom' in args: | 
					
						
							| 
									
										
										
										
											2020-04-16 15:23:08 -07:00
										 |  |  |             if ctx.snes_socket is not None and not ctx.snes_socket.closed: | 
					
						
							|  |  |  |                 asyncio.create_task(ctx.snes_socket.close()) | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |             raise Exception( | 
					
						
							|  |  |  |                 'Invalid ROM detected, please verify that you have loaded the correct rom and reconnect your snes (/snes)') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         if 'SlotAlreadyTaken' in args: | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |             raise Exception('Player slot already in use for that team') | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |         if 'IncompatibleVersion' in args: | 
					
						
							|  |  |  |             raise Exception('Server reported your client version as incompatible') | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         raise Exception('Connection refused by the multiworld host') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     elif cmd == 'Connected': | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         ctx.team, ctx.slot = args[0] | 
					
						
							|  |  |  |         ctx.player_names = {p: n for p, n in args[1]} | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |         msgs = [] | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         if ctx.locations_checked: | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |             msgs.append(['LocationChecks', [Regions.location_table[loc][0] for loc in ctx.locations_checked]]) | 
					
						
							|  |  |  |         if ctx.locations_scouted: | 
					
						
							|  |  |  |             msgs.append(['LocationScouts', list(ctx.locations_scouted)]) | 
					
						
							|  |  |  |         if msgs: | 
					
						
							|  |  |  |             await send_msgs(ctx.socket, msgs) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     elif cmd == 'ReceivedItems': | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         start_index, items = args | 
					
						
							|  |  |  |         if start_index == 0: | 
					
						
							|  |  |  |             ctx.items_received = [] | 
					
						
							|  |  |  |         elif start_index != len(ctx.items_received): | 
					
						
							|  |  |  |             sync_msg = [['Sync']] | 
					
						
							|  |  |  |             if ctx.locations_checked: | 
					
						
							|  |  |  |                 sync_msg.append(['LocationChecks', [Regions.location_table[loc][0] for loc in ctx.locations_checked]]) | 
					
						
							|  |  |  |             await send_msgs(ctx.socket, sync_msg) | 
					
						
							|  |  |  |         if start_index == len(ctx.items_received): | 
					
						
							|  |  |  |             for item in items: | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |                 ctx.items_received.append(ReceivedItem(*item)) | 
					
						
							| 
									
										
										
										
											2020-01-18 11:28:08 +01:00
										 |  |  |         ctx.watcher_event.set() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     elif cmd == 'LocationInfo': | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |         for location, item, player in args: | 
					
						
							|  |  |  |             if location not in ctx.locations_info: | 
					
						
							|  |  |  |                 replacements = {0xA2: 'Small Key', 0x9D: 'Big Key', 0x8D: 'Compass', 0x7D: 'Map'} | 
					
						
							|  |  |  |                 item_name = replacements.get(item, get_item_name_from_id(item)) | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |                 logging.info( | 
					
						
							|  |  |  |                     f"Saw {color(item_name, 'red', 'bold')} at {list(Regions.location_table.keys())[location - 1]}") | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |                 ctx.locations_info[location] = (item, player) | 
					
						
							| 
									
										
										
										
											2020-01-18 11:28:08 +01:00
										 |  |  |         ctx.watcher_event.set() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     elif cmd == 'ItemSent': | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         player_sent, location, player_recvd, item = args | 
					
						
							|  |  |  |         item = color(get_item_name_from_id(item), 'cyan' if player_sent != ctx.slot else 'green') | 
					
						
							|  |  |  |         player_sent = color(ctx.player_names[player_sent], 'yellow' if player_sent != ctx.slot else 'magenta') | 
					
						
							|  |  |  |         player_recvd = color(ctx.player_names[player_recvd], 'yellow' if player_recvd != ctx.slot else 'magenta') | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |         logging.info( | 
					
						
							|  |  |  |             '%s sent %s to %s (%s)' % (player_sent, item, player_recvd, get_location_name_from_address(location))) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  |     elif cmd == 'ItemFound': | 
					
						
							|  |  |  |         found = ReceivedItem(*args) | 
					
						
							|  |  |  |         item = color(get_item_name_from_id(found.item), 'cyan' if found.player != ctx.slot else 'green') | 
					
						
							|  |  |  |         player_sent = color(ctx.player_names[found.player], 'yellow' if found.player != ctx.slot else 'magenta') | 
					
						
							|  |  |  |         logging.info('%s found %s (%s)' % (player_sent, item, get_location_name_from_address(found.location))) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     elif cmd == 'Hint': | 
					
						
							|  |  |  |         hints = [Utils.Hint(*hint) for hint in args] | 
					
						
							|  |  |  |         for hint in hints: | 
					
						
							|  |  |  |             item = color(get_item_name_from_id(hint.item), 'green' if hint.found else 'cyan') | 
					
						
							|  |  |  |             player_find = color(ctx.player_names[hint.finding_player], | 
					
						
							|  |  |  |                                 'yellow' if hint.finding_player != ctx.slot else 'magenta') | 
					
						
							|  |  |  |             player_recvd = color(ctx.player_names[hint.receiving_player], | 
					
						
							|  |  |  |                                  'yellow' if hint.receiving_player != ctx.slot else 'magenta') | 
					
						
							|  |  |  |             logging.info(f"[Hint]: {player_recvd}'s {item} can be found " | 
					
						
							|  |  |  |                          f"at {get_location_name_from_address(hint.location)} in {player_find}'s World." + | 
					
						
							|  |  |  |                          (" (found)" if hint.found else "")) | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |     elif cmd == "AliasUpdate": | 
					
						
							|  |  |  |         ctx.player_names = {p: n for p, n in args} | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     elif cmd == 'Print': | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.info(args) | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |     else: | 
					
						
							|  |  |  |         logging.debug(f"unknown command {args}") | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  | def get_tags(ctx: Context): | 
					
						
							|  |  |  |     tags = ['Berserker'] | 
					
						
							|  |  |  |     if ctx.found_items: | 
					
						
							|  |  |  |         tags.append('FoundItems') | 
					
						
							|  |  |  |     return tags | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | async def server_auth(ctx: Context, password_requested): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     if password_requested and not ctx.password: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.info('Enter the password required to join this game:') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         ctx.password = await console_input(ctx) | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |     if ctx.rom is None: | 
					
						
							|  |  |  |         ctx.awaiting_rom = True | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |         logging.info('No ROM detected, awaiting snes connection to authenticate to the multiworld server (/snes)') | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         return | 
					
						
							|  |  |  |     ctx.awaiting_rom = False | 
					
						
							|  |  |  |     ctx.auth = ctx.rom.copy() | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     await send_msgs(ctx.socket, [['Connect', { | 
					
						
							| 
									
										
										
										
											2020-04-20 19:50:13 +02:00
										 |  |  |         'password': ctx.password, 'rom': ctx.auth, 'version': Utils._version_tuple, 'tags': get_tags(ctx) | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     }]]) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | async def console_input(ctx : Context): | 
					
						
							|  |  |  |     ctx.input_requests += 1 | 
					
						
							|  |  |  |     return await ctx.input_queue.get() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  | async def disconnect(ctx: Context): | 
					
						
							|  |  |  |     if ctx.socket is not None and not ctx.socket.closed: | 
					
						
							|  |  |  |         await ctx.socket.close() | 
					
						
							|  |  |  |     if ctx.server_task is not None: | 
					
						
							|  |  |  |         await ctx.server_task | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  | async def connect(ctx: Context, address=None): | 
					
						
							|  |  |  |     await disconnect(ctx) | 
					
						
							|  |  |  |     ctx.server_task = asyncio.create_task(server_loop(ctx, address)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | from MultiServer import CommandProcessor | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ClientCommandProcessor(CommandProcessor): | 
					
						
							|  |  |  |     def __init__(self, ctx: Context): | 
					
						
							|  |  |  |         self.ctx = ctx | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_exit(self) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """Close connections and client""" | 
					
						
							|  |  |  |         self.ctx.exit_event.set() | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |         return True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_snes(self, snes_address: str = "") -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """Connect to a snes. Optionally include network address of a snes to connect to, otherwise show available devices""" | 
					
						
							|  |  |  |         self.ctx.snes_reconnect_address = None | 
					
						
							|  |  |  |         asyncio.create_task(snes_connect(self.ctx, snes_address if snes_address else self.ctx.snes_address)) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |         return True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_snes_close(self) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """Close connection to a currently connected snes""" | 
					
						
							|  |  |  |         self.ctx.snes_reconnect_address = None | 
					
						
							|  |  |  |         if self.ctx.snes_socket is not None and not self.ctx.snes_socket.closed: | 
					
						
							|  |  |  |             asyncio.create_task(self.ctx.snes_socket.close()) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |             return True | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_connect(self, address: str = "") -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """Connect to a MultiWorld Server""" | 
					
						
							|  |  |  |         self.ctx.server_address = None | 
					
						
							|  |  |  |         asyncio.create_task(connect(self.ctx, address if address else None)) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |         return True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_disconnect(self) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """Disconnect from a MultiWorld Server""" | 
					
						
							|  |  |  |         self.ctx.server_address = None | 
					
						
							|  |  |  |         asyncio.create_task(disconnect(self.ctx)) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |         return True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_received(self) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """List all received items""" | 
					
						
							|  |  |  |         logging.info('Received items:') | 
					
						
							|  |  |  |         for index, item in enumerate(self.ctx.items_received, 1): | 
					
						
							|  |  |  |             logging.info('%s from %s (%s) (%d/%d in list)' % ( | 
					
						
							|  |  |  |                 color(get_item_name_from_id(item.item), 'red', 'bold'), | 
					
						
							|  |  |  |                 color(self.ctx.player_names[item.player], 'yellow'), | 
					
						
							|  |  |  |                 get_location_name_from_address(item.location), index, len(self.ctx.items_received))) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |         return True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_missing(self) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-20 11:47:50 +02:00
										 |  |  |         """List all missing location checks, from your local game state""" | 
					
						
							|  |  |  |         count = 0 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         for location in [k for k, v in Regions.location_table.items() if type(v[0]) is int]: | 
					
						
							|  |  |  |             if location not in self.ctx.locations_checked: | 
					
						
							| 
									
										
										
										
											2020-04-20 11:47:50 +02:00
										 |  |  |                 self.output('Missing: ' + location) | 
					
						
							|  |  |  |                 count += 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if count: | 
					
						
							|  |  |  |             self.output(f"Found {count} missing location checks") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.output("No missing location checks found.") | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |         return True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_show_items(self, toggle: str = "") -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """Toggle showing of items received across the team""" | 
					
						
							|  |  |  |         if toggle: | 
					
						
							|  |  |  |             self.ctx.found_items = toggle.lower() in {"1", "true", "on"} | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.ctx.found_items = not self.ctx.found_items | 
					
						
							|  |  |  |         logging.info(f"Set showing team items to {self.ctx.found_items}") | 
					
						
							|  |  |  |         asyncio.create_task(send_msgs(self.ctx.socket, [['UpdateTags', get_tags(self.ctx)]])) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |         return True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |     def _cmd_slow_mode(self, toggle: str = ""): | 
					
						
							|  |  |  |         """Toggle slow mode, which limits how fast you send / receive items.""" | 
					
						
							|  |  |  |         if toggle: | 
					
						
							|  |  |  |             self.ctx.slow_mode = toggle.lower() in {"1", "true", "on"} | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.ctx.slow_mode = not self.ctx.slow_mode | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         logging.info(f"Setting slow mode to {self.ctx.slow_mode}") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |     def default(self, raw: str): | 
					
						
							|  |  |  |         asyncio.create_task(send_msgs(self.ctx.socket, [['Say', raw]])) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async def console_loop(ctx: Context): | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  |     session = prompt_toolkit.PromptSession() | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |     commandprocessor = ClientCommandProcessor(ctx) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     while not ctx.exit_event.is_set(): | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  |         try: | 
					
						
							|  |  |  |             with patch_stdout(): | 
					
						
							|  |  |  |                 input_text = await session.prompt_async() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  |             if ctx.input_requests > 0: | 
					
						
							|  |  |  |                 ctx.input_requests -= 1 | 
					
						
							|  |  |  |                 ctx.input_queue.put_nowait(input_text) | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 03:24:27 +02:00
										 |  |  |             if not input_text: | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |             commandprocessor(input_text) | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  |         except Exception as e: | 
					
						
							|  |  |  |             logging.exception(e) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         await snes_flush_writes(ctx) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async def track_locations(ctx : Context, roomid, roomdata): | 
					
						
							|  |  |  |     new_locations = [] | 
					
						
							|  |  |  |     def new_check(location): | 
					
						
							|  |  |  |         ctx.locations_checked.add(location) | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.info("New check: %s (%d/216)" % (location, len(ctx.locations_checked))) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         new_locations.append(Regions.location_table[location][0]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for location, (loc_roomid, loc_mask) in location_table_uw.items(): | 
					
						
							|  |  |  |         if location not in ctx.locations_checked and loc_roomid == roomid and (roomdata << 4) & loc_mask != 0: | 
					
						
							|  |  |  |             new_check(location) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     uw_begin = 0x129 | 
					
						
							|  |  |  |     uw_end = 0 | 
					
						
							|  |  |  |     uw_unchecked = {} | 
					
						
							|  |  |  |     for location, (roomid, mask) in location_table_uw.items(): | 
					
						
							|  |  |  |         if location not in ctx.locations_checked: | 
					
						
							|  |  |  |             uw_unchecked[location] = (roomid, mask) | 
					
						
							|  |  |  |             uw_begin = min(uw_begin, roomid) | 
					
						
							|  |  |  |             uw_end = max(uw_end, roomid + 1) | 
					
						
							|  |  |  |     if uw_begin < uw_end: | 
					
						
							|  |  |  |         uw_data = await snes_read(ctx, SAVEDATA_START + (uw_begin * 2), (uw_end - uw_begin) * 2) | 
					
						
							|  |  |  |         if uw_data is not None: | 
					
						
							|  |  |  |             for location, (roomid, mask) in uw_unchecked.items(): | 
					
						
							|  |  |  |                 offset = (roomid - uw_begin) * 2 | 
					
						
							|  |  |  |                 roomdata = uw_data[offset] | (uw_data[offset + 1] << 8) | 
					
						
							|  |  |  |                 if roomdata & mask != 0: | 
					
						
							|  |  |  |                     new_check(location) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ow_begin = 0x82 | 
					
						
							|  |  |  |     ow_end = 0 | 
					
						
							|  |  |  |     ow_unchecked = {} | 
					
						
							|  |  |  |     for location, screenid in location_table_ow.items(): | 
					
						
							|  |  |  |         if location not in ctx.locations_checked: | 
					
						
							|  |  |  |             ow_unchecked[location] = screenid | 
					
						
							|  |  |  |             ow_begin = min(ow_begin, screenid) | 
					
						
							|  |  |  |             ow_end = max(ow_end, screenid + 1) | 
					
						
							|  |  |  |     if ow_begin < ow_end: | 
					
						
							|  |  |  |         ow_data = await snes_read(ctx, SAVEDATA_START + 0x280 + ow_begin, ow_end - ow_begin) | 
					
						
							|  |  |  |         if ow_data is not None: | 
					
						
							|  |  |  |             for location, screenid in ow_unchecked.items(): | 
					
						
							|  |  |  |                 if ow_data[screenid - ow_begin] & 0x40 != 0: | 
					
						
							|  |  |  |                     new_check(location) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if not all([location in ctx.locations_checked for location in location_table_npc.keys()]): | 
					
						
							|  |  |  |         npc_data = await snes_read(ctx, SAVEDATA_START + 0x410, 2) | 
					
						
							|  |  |  |         if npc_data is not None: | 
					
						
							|  |  |  |             npc_value = npc_data[0] | (npc_data[1] << 8) | 
					
						
							|  |  |  |             for location, mask in location_table_npc.items(): | 
					
						
							|  |  |  |                 if npc_value & mask != 0 and location not in ctx.locations_checked: | 
					
						
							|  |  |  |                     new_check(location) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if not all([location in ctx.locations_checked for location in location_table_misc.keys()]): | 
					
						
							|  |  |  |         misc_data = await snes_read(ctx, SAVEDATA_START + 0x3c6, 4) | 
					
						
							|  |  |  |         if misc_data is not None: | 
					
						
							|  |  |  |             for location, (offset, mask) in location_table_misc.items(): | 
					
						
							|  |  |  |                 assert(0x3c6 <= offset <= 0x3c9) | 
					
						
							|  |  |  |                 if misc_data[offset - 0x3c6] & mask != 0 and location not in ctx.locations_checked: | 
					
						
							|  |  |  |                     new_check(location) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await send_msgs(ctx.socket, [['LocationChecks', new_locations]]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async def game_watcher(ctx : Context): | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |     prev_game_timer = 0 | 
					
						
							|  |  |  |     perf_counter = time.perf_counter() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     while not ctx.exit_event.is_set(): | 
					
						
							| 
									
										
										
										
											2020-01-18 11:28:08 +01:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |             await asyncio.wait_for(ctx.watcher_event.wait(), 0.125) | 
					
						
							| 
									
										
										
										
											2020-01-18 11:28:08 +01:00
										 |  |  |         except asyncio.TimeoutError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |         ctx.watcher_event.clear() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         if not ctx.rom: | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |             ctx.finished_game = False | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             rom = await snes_read(ctx, ROMNAME_START, ROMNAME_SIZE) | 
					
						
							|  |  |  |             if rom is None or rom == bytes([0] * ROMNAME_SIZE): | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             ctx.rom = list(rom) | 
					
						
							|  |  |  |             ctx.locations_checked = set() | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |             ctx.locations_scouted = set() | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |             if ctx.awaiting_rom: | 
					
						
							|  |  |  |                 await server_auth(ctx, False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ctx.auth and ctx.auth != ctx.rom: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.warning("ROM change detected, please reconnect to the multiworld server") | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |             await disconnect(ctx) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         gamemode = await snes_read(ctx, WRAM_START + 0x10, 1) | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |         gameend = await snes_read(ctx, SAVEDATA_START + 0x443, 1) | 
					
						
							|  |  |  |         game_timer = await snes_read(ctx, SAVEDATA_START + 0x42E, 4) | 
					
						
							|  |  |  |         if gamemode is None or gameend is None or game_timer is None or \ | 
					
						
							|  |  |  |                 (gamemode[0] not in INGAME_MODES and gamemode[0] not in ENDGAME_MODES): | 
					
						
							|  |  |  |             continue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         delay = 7 if ctx.slow_mode else 2 | 
					
						
							|  |  |  |         if gameend[0]: | 
					
						
							|  |  |  |             if not ctx.finished_game: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     if ctx.server_version >= (2, 0, 5): | 
					
						
							|  |  |  |                         await send_msgs(ctx.socket, [['GameFinished', '']]) | 
					
						
							|  |  |  |                         ctx.finished_game = True | 
					
						
							|  |  |  |                 except Exception as ex: | 
					
						
							|  |  |  |                     logging.exception(ex) | 
					
						
							|  |  |  |                     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if time.perf_counter() - perf_counter < delay: | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 perf_counter = time.perf_counter() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             game_timer = game_timer[0] | (game_timer[1] << 8) | (game_timer[2] << 16) | (game_timer[3] << 24) | 
					
						
							|  |  |  |             if abs(game_timer - prev_game_timer) < (delay * 60): | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 prev_game_timer = game_timer | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if gamemode in ENDGAME_MODES: | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             continue | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |         data = await snes_read(ctx, RECV_PROGRESS_ADDR, 8) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         if data is None: | 
					
						
							|  |  |  |             continue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         recv_index = data[0] | (data[1] << 8) | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |         assert RECV_ITEM_ADDR == RECV_PROGRESS_ADDR + 2 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         recv_item = data[2] | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |         assert ROOMID_ADDR == RECV_PROGRESS_ADDR + 4 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         roomid = data[4] | (data[5] << 8) | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |         assert ROOMDATA_ADDR == RECV_PROGRESS_ADDR + 6 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         roomdata = data[6] | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |         assert SCOUT_LOCATION_ADDR == RECV_PROGRESS_ADDR + 7 | 
					
						
							|  |  |  |         scout_location = data[7] | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if recv_index < len(ctx.items_received) and recv_item == 0: | 
					
						
							|  |  |  |             item = ctx.items_received[recv_index] | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.info('Received %s from %s (%s) (%d/%d in list)' % ( | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |                 color(get_item_name_from_id(item.item), 'red', 'bold'), color(ctx.player_names[item.player], 'yellow'), | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |                 get_location_name_from_address(item.location), recv_index + 1, len(ctx.items_received))) | 
					
						
							|  |  |  |             recv_index += 1 | 
					
						
							|  |  |  |             snes_buffered_write(ctx, RECV_PROGRESS_ADDR, bytes([recv_index & 0xFF, (recv_index >> 8) & 0xFF])) | 
					
						
							|  |  |  |             snes_buffered_write(ctx, RECV_ITEM_ADDR, bytes([item.item])) | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |             snes_buffered_write(ctx, RECV_ITEM_PLAYER_ADDR, bytes([item.player if item.player != ctx.slot else 0])) | 
					
						
							|  |  |  |         if scout_location > 0 and scout_location in ctx.locations_info: | 
					
						
							|  |  |  |             snes_buffered_write(ctx, SCOUTREPLY_LOCATION_ADDR, bytes([scout_location])) | 
					
						
							|  |  |  |             snes_buffered_write(ctx, SCOUTREPLY_ITEM_ADDR, bytes([ctx.locations_info[scout_location][0]])) | 
					
						
							|  |  |  |             snes_buffered_write(ctx, SCOUTREPLY_PLAYER_ADDR, bytes([ctx.locations_info[scout_location][1]])) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         await snes_flush_writes(ctx) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 11:28:08 +01:00
										 |  |  |         if scout_location > 0 and scout_location not in ctx.locations_scouted: | 
					
						
							|  |  |  |             ctx.locations_scouted.add(scout_location) | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.info(f'Scouting item at {list(Regions.location_table.keys())[scout_location - 1]}') | 
					
						
							| 
									
										
										
										
											2020-01-18 11:28:08 +01:00
										 |  |  |             await send_msgs(ctx.socket, [['LocationScouts', [scout_location]]]) | 
					
						
							|  |  |  |         await track_locations(ctx, roomid, roomdata) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | async def run_game(romfile): | 
					
						
							|  |  |  |     import webbrowser | 
					
						
							|  |  |  |     webbrowser.open(romfile) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | async def main(): | 
					
						
							|  |  |  |     parser = argparse.ArgumentParser() | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  |     parser.add_argument('diff_file', default="", type=str, nargs="?", | 
					
						
							|  |  |  |                         help='Path to a Berserker Multiworld Binary Patch file') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     parser.add_argument('--snes', default='localhost:8080', help='Address of the QUsb2snes server.') | 
					
						
							|  |  |  |     parser.add_argument('--connect', default=None, help='Address of the multiworld host.') | 
					
						
							|  |  |  |     parser.add_argument('--password', default=None, help='Password of the multiworld host.') | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |     parser.add_argument('--loglevel', default='info', choices=['debug', 'info', 'warning', 'error', 'critical']) | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  |     parser.add_argument('--founditems', default=False, action='store_true', help='Show items found by other players for themselves.') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     args = parser.parse_args() | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     logging.basicConfig(format='%(message)s', level=getattr(logging, args.loglevel.upper(), logging.INFO)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  |     if args.diff_file: | 
					
						
							|  |  |  |         import Patch | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  |         meta, romfile = Patch.create_rom_file(args.diff_file) | 
					
						
							|  |  |  |         args.connect = meta["server"] | 
					
						
							|  |  |  |         logging.info(f"Wrote rom file to {romfile}") | 
					
						
							|  |  |  |         asyncio.create_task(run_game(romfile)) | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  |     ctx = Context(args.snes, args.connect, args.password, args.founditems) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     input_task = asyncio.create_task(console_loop(ctx)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |     await snes_connect(ctx, ctx.snes_address) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if ctx.server_task is None: | 
					
						
							|  |  |  |         ctx.server_task = asyncio.create_task(server_loop(ctx)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     watcher_task = asyncio.create_task(game_watcher(ctx)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await ctx.exit_event.wait() | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |     ctx.server_address = None | 
					
						
							| 
									
										
										
										
											2020-01-13 03:55:33 +01:00
										 |  |  |     ctx.snes_reconnect_address = None | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     await watcher_task | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ctx.socket is not None and not ctx.socket.closed: | 
					
						
							|  |  |  |         await ctx.socket.close() | 
					
						
							|  |  |  |     if ctx.server_task is not None: | 
					
						
							|  |  |  |         await ctx.server_task | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ctx.snes_socket is not None and not ctx.snes_socket.closed: | 
					
						
							|  |  |  |         await ctx.snes_socket.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while ctx.input_requests > 0: | 
					
						
							|  |  |  |         ctx.input_queue.put_nowait(None) | 
					
						
							|  |  |  |         ctx.input_requests -= 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     await input_task | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							| 
									
										
										
										
											2020-01-18 10:05:59 +01:00
										 |  |  |     colorama.init() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     loop = asyncio.get_event_loop() | 
					
						
							|  |  |  |     loop.run_until_complete(main()) | 
					
						
							|  |  |  |     loop.close() | 
					
						
							| 
									
										
										
										
											2020-01-18 10:05:59 +01:00
										 |  |  |     colorama.deinit() | 
					
						
							| 
									
										
										
										
											2020-02-02 06:20:08 +01:00
										 |  |  |     atexit.unregister(exit_func) |