| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | from __future__ import annotations | 
					
						
							|  |  |  | import logging | 
					
						
							|  |  |  | import asyncio | 
					
						
							|  |  |  | import urllib.parse | 
					
						
							| 
									
										
										
										
											2021-08-20 22:31:17 +02:00
										 |  |  | import sys | 
					
						
							| 
									
										
										
										
											2021-10-25 09:58:08 +02:00
										 |  |  | import typing | 
					
						
							| 
									
										
										
										
											2021-11-01 19:37:47 +01:00
										 |  |  | import time | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 10:19:18 +02:00
										 |  |  | import ModuleUpdate | 
					
						
							|  |  |  | ModuleUpdate.update() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | import websockets | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import Utils | 
					
						
							| 
									
										
										
										
											2021-11-10 15:35:43 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							| 
									
										
										
										
											2021-11-17 22:46:32 +01:00
										 |  |  |     Utils.init_logging("TextClient", exception_logger="Client") | 
					
						
							| 
									
										
										
										
											2021-11-10 15:35:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | from MultiServer import CommandProcessor | 
					
						
							| 
									
										
										
										
											2022-05-24 00:20:02 +02:00
										 |  |  | from NetUtils import Endpoint, decode, NetworkItem, encode, JSONtoTextParser, ClientStatus, Permission, NetworkSlot | 
					
						
							| 
									
										
										
										
											2021-11-28 04:06:30 +01:00
										 |  |  | from Utils import Version, stream_input | 
					
						
							| 
									
										
										
										
											2021-07-12 20:07:02 +02:00
										 |  |  | from worlds import network_data_package, AutoWorldRegister | 
					
						
							| 
									
										
										
										
											2022-04-06 00:06:48 +02:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | logger = logging.getLogger("Client") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-21 22:24:49 +02:00
										 |  |  | # without terminal, we have to use gui mode | 
					
						
							| 
									
										
										
										
											2021-11-09 12:53:05 +01:00
										 |  |  | gui_enabled = not sys.stdout or "--nogui" not in sys.argv | 
					
						
							| 
									
										
										
										
											2021-08-04 18:38:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | class ClientCommandProcessor(CommandProcessor): | 
					
						
							|  |  |  |     def __init__(self, ctx: CommonContext): | 
					
						
							|  |  |  |         self.ctx = ctx | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def output(self, text: str): | 
					
						
							|  |  |  |         logger.info(text) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _cmd_exit(self) -> bool: | 
					
						
							|  |  |  |         """Close connections and client""" | 
					
						
							|  |  |  |         self.ctx.exit_event.set() | 
					
						
							|  |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _cmd_connect(self, address: str = "") -> bool: | 
					
						
							|  |  |  |         """Connect to a MultiWorld Server""" | 
					
						
							|  |  |  |         self.ctx.server_address = None | 
					
						
							| 
									
										
										
										
											2021-11-21 02:02:40 +01:00
										 |  |  |         asyncio.create_task(self.ctx.connect(address if address else None), name="connecting") | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _cmd_disconnect(self) -> bool: | 
					
						
							|  |  |  |         """Disconnect from a MultiWorld Server""" | 
					
						
							|  |  |  |         self.ctx.server_address = None | 
					
						
							| 
									
										
										
										
											2021-11-21 02:02:40 +01:00
										 |  |  |         asyncio.create_task(self.ctx.disconnect(), name="disconnecting") | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _cmd_received(self) -> bool: | 
					
						
							|  |  |  |         """List all received items""" | 
					
						
							| 
									
										
										
										
											2021-06-21 02:14:25 +02:00
										 |  |  |         logger.info(f'{len(self.ctx.items_received)} received items:') | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         for index, item in enumerate(self.ctx.items_received, 1): | 
					
						
							| 
									
										
										
										
											2021-06-21 02:14:25 +02:00
										 |  |  |             self.output(f"{self.ctx.item_name_getter(item.item)} from {self.ctx.player_names[item.player]}") | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _cmd_missing(self) -> bool: | 
					
						
							|  |  |  |         """List all missing location checks, from your local game state""" | 
					
						
							| 
									
										
										
										
											2021-10-30 07:33:05 +02:00
										 |  |  |         if not self.ctx.game: | 
					
						
							|  |  |  |             self.output("No game set, cannot determine missing checks.") | 
					
						
							| 
									
										
										
										
											2021-11-09 12:53:05 +01:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         count = 0 | 
					
						
							|  |  |  |         checked_count = 0 | 
					
						
							| 
									
										
										
										
											2021-08-10 04:38:29 +02:00
										 |  |  |         for location, location_id in AutoWorldRegister.world_types[self.ctx.game].location_name_to_id.items(): | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |             if location_id < 0: | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             if location_id not in self.ctx.locations_checked: | 
					
						
							|  |  |  |                 if location_id in self.ctx.missing_locations: | 
					
						
							|  |  |  |                     self.output('Missing: ' + location) | 
					
						
							|  |  |  |                     count += 1 | 
					
						
							|  |  |  |                 elif location_id in self.ctx.checked_locations: | 
					
						
							|  |  |  |                     self.output('Checked: ' + location) | 
					
						
							|  |  |  |                     count += 1 | 
					
						
							|  |  |  |                     checked_count += 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if count: | 
					
						
							|  |  |  |             self.output( | 
					
						
							|  |  |  |                 f"Found {count} missing location checks{f'. {checked_count} location checks previously visited.' if checked_count else ''}") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.output("No missing location checks found.") | 
					
						
							|  |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-23 21:47:23 +01:00
										 |  |  |     def _cmd_items(self): | 
					
						
							| 
									
										
										
										
											2021-12-03 05:14:44 +01:00
										 |  |  |         """List all item names for the currently running game.""" | 
					
						
							| 
									
										
										
										
											2021-11-23 21:47:23 +01:00
										 |  |  |         self.output(f"Item Names for {self.ctx.game}") | 
					
						
							|  |  |  |         for item_name in AutoWorldRegister.world_types[self.ctx.game].item_name_to_id: | 
					
						
							|  |  |  |             self.output(item_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _cmd_locations(self): | 
					
						
							| 
									
										
										
										
											2021-12-03 05:14:44 +01:00
										 |  |  |         """List all location names for the currently running game.""" | 
					
						
							| 
									
										
										
										
											2021-11-23 21:47:23 +01:00
										 |  |  |         self.output(f"Location Names for {self.ctx.game}") | 
					
						
							|  |  |  |         for location_name in AutoWorldRegister.world_types[self.ctx.game].location_name_to_id: | 
					
						
							|  |  |  |             self.output(location_name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |     def _cmd_ready(self): | 
					
						
							| 
									
										
										
										
											2021-12-03 07:04:17 +01:00
										 |  |  |         """Send ready status to server.""" | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         self.ctx.ready = not self.ctx.ready | 
					
						
							|  |  |  |         if self.ctx.ready: | 
					
						
							|  |  |  |             state = ClientStatus.CLIENT_READY | 
					
						
							|  |  |  |             self.output("Readied up.") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             state = ClientStatus.CLIENT_CONNECTED | 
					
						
							|  |  |  |             self.output("Unreadied.") | 
					
						
							| 
									
										
										
										
											2021-11-21 02:02:40 +01:00
										 |  |  |         asyncio.create_task(self.ctx.send_msgs([{"cmd": "StatusUpdate", "status": state}]), name="send StatusUpdate") | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def default(self, raw: str): | 
					
						
							| 
									
										
										
										
											2021-11-29 21:35:06 +01:00
										 |  |  |         raw = self.ctx.on_user_say(raw) | 
					
						
							|  |  |  |         if raw: | 
					
						
							|  |  |  |             asyncio.create_task(self.ctx.send_msgs([{"cmd": "Say", "text": raw}]), name="send Say") | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-04 18:38:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | class CommonContext(): | 
					
						
							| 
									
										
										
										
											2021-10-29 10:03:15 +02:00
										 |  |  |     tags: typing.Set[str] = {"AP"} | 
					
						
							| 
									
										
										
										
											2021-09-11 03:59:12 +02:00
										 |  |  |     starting_reconnect_delay: int = 5 | 
					
						
							|  |  |  |     current_reconnect_delay: int = starting_reconnect_delay | 
					
						
							|  |  |  |     command_processor: int = ClientCommandProcessor | 
					
						
							| 
									
										
										
										
											2022-04-27 22:11:11 +02:00
										 |  |  |     game: typing.Optional[str] = None | 
					
						
							| 
									
										
										
										
											2021-09-11 03:59:12 +02:00
										 |  |  |     ui = None | 
					
						
							| 
									
										
										
										
											2022-04-27 22:11:11 +02:00
										 |  |  |     ui_task: typing.Optional[asyncio.Task] = None | 
					
						
							|  |  |  |     input_task: typing.Optional[asyncio.Task] = None | 
					
						
							|  |  |  |     keep_alive_task: typing.Optional[asyncio.Task] = None | 
					
						
							| 
									
										
										
										
											2022-01-23 06:38:46 +01:00
										 |  |  |     items_handling: typing.Optional[int] = None | 
					
						
							| 
									
										
										
										
											2022-05-24 00:20:02 +02:00
										 |  |  |     slot_info: typing.Dict[int, NetworkSlot] | 
					
						
							| 
									
										
										
										
											2022-05-21 22:24:49 +02:00
										 |  |  |     current_energy_link_value: int = 0  # to display in UI, gets set by server | 
					
						
							| 
									
										
										
										
											2021-07-12 20:07:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-19 21:52:08 +02:00
										 |  |  |     def __init__(self, server_address, password): | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         # server state | 
					
						
							|  |  |  |         self.server_address = server_address | 
					
						
							|  |  |  |         self.password = password | 
					
						
							|  |  |  |         self.server_task = None | 
					
						
							|  |  |  |         self.server: typing.Optional[Endpoint] = None | 
					
						
							|  |  |  |         self.server_version = Version(0, 0, 0) | 
					
						
							| 
									
										
										
										
											2021-10-29 10:03:15 +02:00
										 |  |  |         self.hint_cost: typing.Optional[int] = None | 
					
						
							| 
									
										
										
										
											2021-10-30 07:33:05 +02:00
										 |  |  |         self.games: typing.Dict[int, str] = {} | 
					
						
							| 
									
										
										
										
											2022-05-24 00:20:02 +02:00
										 |  |  |         self.slot_info = {} | 
					
						
							| 
									
										
										
										
											2021-10-22 05:25:09 +02:00
										 |  |  |         self.permissions = { | 
					
						
							|  |  |  |             "forfeit": "disabled", | 
					
						
							|  |  |  |             "collect": "disabled", | 
					
						
							|  |  |  |             "remaining": "disabled", | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # own state | 
					
						
							|  |  |  |         self.finished_game = False | 
					
						
							|  |  |  |         self.ready = False | 
					
						
							|  |  |  |         self.team = None | 
					
						
							|  |  |  |         self.slot = None | 
					
						
							|  |  |  |         self.auth = None | 
					
						
							| 
									
										
										
										
											2021-05-16 00:21:00 +02:00
										 |  |  |         self.seed_name = None | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-29 10:03:15 +02:00
										 |  |  |         self.locations_checked: typing.Set[int] = set()  # local state | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         self.locations_scouted: typing.Set[int] = set() | 
					
						
							|  |  |  |         self.items_received = [] | 
					
						
							| 
									
										
										
										
											2021-10-18 22:58:29 +02:00
										 |  |  |         self.missing_locations: typing.Set[int] = set() | 
					
						
							| 
									
										
										
										
											2021-10-29 10:03:15 +02:00
										 |  |  |         self.checked_locations: typing.Set[int] = set()  # server state | 
					
						
							| 
									
										
										
										
											2022-03-21 15:26:38 +01:00
										 |  |  |         self.locations_info: typing.Dict[int, NetworkItem] = {} | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.input_queue = asyncio.Queue() | 
					
						
							|  |  |  |         self.input_requests = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-01 19:37:47 +01:00
										 |  |  |         self.last_death_link: float = time.time()  # last send/received death link on AP layer | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         # game state | 
					
						
							| 
									
										
										
										
											2021-05-11 23:08:50 +02:00
										 |  |  |         self.player_names: typing.Dict[int: str] = {0: "Archipelago"} | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         self.exit_event = asyncio.Event() | 
					
						
							|  |  |  |         self.watcher_event = asyncio.Event() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.slow_mode = False | 
					
						
							|  |  |  |         self.jsontotextparser = JSONtoTextParser(self) | 
					
						
							|  |  |  |         self.set_getters(network_data_package) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-11 03:59:12 +02:00
										 |  |  |         # execution | 
					
						
							| 
									
										
										
										
											2021-11-21 02:02:40 +01:00
										 |  |  |         self.keep_alive_task = asyncio.create_task(keep_alive(self), name="Bouncy") | 
					
						
							| 
									
										
										
										
											2021-09-11 03:59:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-29 10:03:15 +02:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def total_locations(self) -> typing.Optional[int]: | 
					
						
							|  |  |  |         """Will return None until connected.""" | 
					
						
							|  |  |  |         if self.checked_locations or self.missing_locations: | 
					
						
							|  |  |  |             return len(self.checked_locations | self.missing_locations) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |     async def connection_closed(self): | 
					
						
							| 
									
										
										
										
											2022-05-21 22:24:49 +02:00
										 |  |  |         self.reset_server_state() | 
					
						
							|  |  |  |         if self.server and self.server.socket is not None: | 
					
						
							|  |  |  |             await self.server.socket.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def reset_server_state(self): | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         self.auth = None | 
					
						
							| 
									
										
										
										
											2022-05-24 00:20:02 +02:00
										 |  |  |         self.slot = None | 
					
						
							|  |  |  |         self.team = None | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         self.items_received = [] | 
					
						
							|  |  |  |         self.locations_info = {} | 
					
						
							|  |  |  |         self.server_version = Version(0, 0, 0) | 
					
						
							|  |  |  |         self.server = None | 
					
						
							|  |  |  |         self.server_task = None | 
					
						
							| 
									
										
										
										
											2022-05-21 22:24:49 +02:00
										 |  |  |         self.games = {} | 
					
						
							|  |  |  |         self.hint_cost = None | 
					
						
							|  |  |  |         self.permissions = { | 
					
						
							|  |  |  |             "forfeit": "disabled", | 
					
						
							|  |  |  |             "collect": "disabled", | 
					
						
							|  |  |  |             "remaining": "disabled", | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-04 08:57:27 +01:00
										 |  |  |     # noinspection PyAttributeOutsideInit | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |     def set_getters(self, data_package: dict, network=False): | 
					
						
							|  |  |  |         if not network:  # local data; check if newer data was already downloaded | 
					
						
							|  |  |  |             local_package = Utils.persistent_load().get("datapackage", {}).get("latest", {}) | 
					
						
							|  |  |  |             if local_package and local_package["version"] > network_data_package["version"]: | 
					
						
							|  |  |  |                 data_package: dict = local_package | 
					
						
							|  |  |  |         elif network:  # check if data from server is newer | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if data_package["version"] > network_data_package["version"]: | 
					
						
							|  |  |  |                 Utils.persistent_store("datapackage", "latest", network_data_package) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-04 15:54:32 +02:00
										 |  |  |         item_lookup: dict = {} | 
					
						
							|  |  |  |         locations_lookup: dict = {} | 
					
						
							|  |  |  |         for game, gamedata in data_package["games"].items(): | 
					
						
							|  |  |  |             for item_name, item_id in gamedata["item_name_to_id"].items(): | 
					
						
							|  |  |  |                 item_lookup[item_id] = item_name | 
					
						
							|  |  |  |             for location_name, location_id in gamedata["location_name_to_id"].items(): | 
					
						
							|  |  |  |                 locations_lookup[location_id] = location_name | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def get_item_name_from_id(code: int): | 
					
						
							|  |  |  |             return item_lookup.get(code, f'Unknown item (ID:{code})') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.item_name_getter = get_item_name_from_id | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def get_location_name_from_address(address: int): | 
					
						
							|  |  |  |             return locations_lookup.get(address, f'Unknown location (ID:{address})') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.location_name_getter = get_location_name_from_address | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     async def disconnect(self): | 
					
						
							|  |  |  |         if self.server and not self.server.socket.closed: | 
					
						
							|  |  |  |             await self.server.socket.close() | 
					
						
							|  |  |  |         if self.server_task is not None: | 
					
						
							|  |  |  |             await self.server_task | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     async def send_msgs(self, msgs): | 
					
						
							|  |  |  |         if not self.server or not self.server.socket.open or self.server.socket.closed: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         await self.server.socket.send(encode(msgs)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def consume_players_package(self, package: typing.List[tuple]): | 
					
						
							|  |  |  |         self.player_names = {slot: name for team, slot, name, orig_name in package if self.team == team} | 
					
						
							| 
									
										
										
										
											2021-05-11 23:08:50 +02:00
										 |  |  |         self.player_names[0] = "Archipelago" | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def event_invalid_slot(self): | 
					
						
							|  |  |  |         raise Exception('Invalid Slot; please verify that you have connected to the correct world.') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-17 04:32:09 +02:00
										 |  |  |     def event_invalid_game(self): | 
					
						
							|  |  |  |         raise Exception('Invalid Game; please verify that you connected with the right game to the correct world.') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  |     async def server_auth(self, password_requested: bool = False): | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         if password_requested and not self.password: | 
					
						
							|  |  |  |             logger.info('Enter the password required to join this game:') | 
					
						
							|  |  |  |             self.password = await self.console_input() | 
					
						
							|  |  |  |             return self.password | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-21 02:50:24 +01:00
										 |  |  |     async def send_connect(self, **kwargs): | 
					
						
							|  |  |  |         payload = { | 
					
						
							| 
									
										
										
										
											2022-01-23 06:38:46 +01:00
										 |  |  |             'cmd': 'Connect', | 
					
						
							| 
									
										
										
										
											2021-11-21 02:50:24 +01:00
										 |  |  |             'password': self.password, 'name': self.auth, 'version': Utils.version_tuple, | 
					
						
							| 
									
										
										
										
											2022-01-23 06:38:46 +01:00
										 |  |  |             'tags': self.tags, 'items_handling': self.items_handling, | 
					
						
							| 
									
										
										
										
											2021-11-21 02:50:24 +01:00
										 |  |  |             'uuid': Utils.get_unique_identifier(), 'game': self.game | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if kwargs: | 
					
						
							|  |  |  |             payload.update(kwargs) | 
					
						
							|  |  |  |         await self.send_msgs([payload]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |     async def console_input(self): | 
					
						
							|  |  |  |         self.input_requests += 1 | 
					
						
							|  |  |  |         return await self.input_queue.get() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-04 18:38:49 +02:00
										 |  |  |     async def connect(self, address=None): | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         await self.disconnect() | 
					
						
							| 
									
										
										
										
											2021-11-21 02:02:40 +01:00
										 |  |  |         self.server_task = asyncio.create_task(server_loop(self, address), name="server loop") | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-13 14:49:32 +02:00
										 |  |  |     def on_print(self, args: dict): | 
					
						
							|  |  |  |         logger.info(args["text"]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def on_print_json(self, args: dict): | 
					
						
							| 
									
										
										
										
											2021-07-31 01:53:06 +02:00
										 |  |  |         if self.ui: | 
					
						
							|  |  |  |             self.ui.print_json(args["data"]) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             text = self.jsontotextparser(args["data"]) | 
					
						
							|  |  |  |             logger.info(text) | 
					
						
							| 
									
										
										
										
											2021-04-13 14:49:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-31 19:45:17 +02:00
										 |  |  |     def on_package(self, cmd: str, args: dict): | 
					
						
							|  |  |  |         """For custom package handling in subclasses.""" | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-29 21:35:06 +01:00
										 |  |  |     def on_user_say(self, text: str) -> typing.Optional[str]: | 
					
						
							|  |  |  |         """Gets called before sending a Say to the server from the user.
 | 
					
						
							|  |  |  |         Returned text is sent, or sending is aborted if None is returned."""
 | 
					
						
							|  |  |  |         return text | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-22 05:25:09 +02:00
										 |  |  |     def update_permissions(self, permissions: typing.Dict[str, int]): | 
					
						
							|  |  |  |         for permission_name, permission_flag in permissions.items(): | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 flag = Permission(permission_flag) | 
					
						
							|  |  |  |                 logger.info(f"{permission_name.capitalize()} permission: {flag.name}") | 
					
						
							|  |  |  |                 self.permissions[permission_name] = flag.name | 
					
						
							|  |  |  |             except Exception as e:  # safeguard against permissions that may be implemented in the future | 
					
						
							|  |  |  |                 logger.exception(e) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-29 21:35:06 +01:00
										 |  |  |     async def shutdown(self): | 
					
						
							|  |  |  |         self.server_address = None | 
					
						
							|  |  |  |         if self.server and not self.server.socket.closed: | 
					
						
							|  |  |  |             await self.server.socket.close() | 
					
						
							|  |  |  |         if self.server_task: | 
					
						
							|  |  |  |             await self.server_task | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while self.input_requests > 0: | 
					
						
							|  |  |  |             self.input_queue.put_nowait(None) | 
					
						
							|  |  |  |             self.input_requests -= 1 | 
					
						
							|  |  |  |         self.keep_alive_task.cancel() | 
					
						
							| 
									
										
										
										
											2022-04-27 22:11:11 +02:00
										 |  |  |         if self.ui_task: | 
					
						
							|  |  |  |             await self.ui_task | 
					
						
							|  |  |  |         if self.input_task: | 
					
						
							|  |  |  |             self.input_task.cancel() | 
					
						
							| 
									
										
										
										
											2021-11-29 21:35:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # DeathLink hooks | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-01 19:37:47 +01:00
										 |  |  |     def on_deathlink(self, data: dict): | 
					
						
							|  |  |  |         """Gets dispatched when a new DeathLink is triggered by another linked player.""" | 
					
						
							| 
									
										
										
										
											2021-11-04 13:23:13 +01:00
										 |  |  |         self.last_death_link = max(data["time"], self.last_death_link) | 
					
						
							| 
									
										
										
										
											2021-11-06 11:19:49 +01:00
										 |  |  |         text = data.get("cause", "") | 
					
						
							|  |  |  |         if text: | 
					
						
							|  |  |  |             logger.info(f"DeathLink: {text}") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             logger.info(f"DeathLink: Received from {data['source']}") | 
					
						
							| 
									
										
										
										
											2021-11-01 19:37:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-06 16:17:10 +01:00
										 |  |  |     async def send_death(self, death_text: str = ""): | 
					
						
							| 
									
										
										
										
											2022-02-24 06:17:39 +01:00
										 |  |  |         if self.server and self.server.socket: | 
					
						
							|  |  |  |             logger.info("DeathLink: Sending death to your friends...") | 
					
						
							|  |  |  |             self.last_death_link = time.time() | 
					
						
							|  |  |  |             await self.send_msgs([{ | 
					
						
							|  |  |  |                 "cmd": "Bounce", "tags": ["DeathLink"], | 
					
						
							|  |  |  |                 "data": { | 
					
						
							|  |  |  |                     "time": self.last_death_link, | 
					
						
							|  |  |  |                     "source": self.player_names[self.slot], | 
					
						
							|  |  |  |                     "cause": death_text | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             }]) | 
					
						
							| 
									
										
										
										
											2021-11-01 19:37:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-24 05:22:52 +02:00
										 |  |  |     async def update_death_link(self, death_link: bool): | 
					
						
							| 
									
										
										
										
											2021-11-24 01:38:58 -08:00
										 |  |  |         old_tags = self.tags.copy() | 
					
						
							|  |  |  |         if death_link: | 
					
						
							|  |  |  |             self.tags.add("DeathLink") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.tags -= {"DeathLink"} | 
					
						
							|  |  |  |         if old_tags != self.tags and self.server and not self.server.socket.closed: | 
					
						
							|  |  |  |             await self.send_msgs([{"cmd": "ConnectUpdate", "tags": self.tags}]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-27 22:11:11 +02:00
										 |  |  |     def run_gui(self): | 
					
						
							|  |  |  |         """Import kivy UI system and start running it as self.ui_task.""" | 
					
						
							|  |  |  |         from kvui import GameManager | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         class TextManager(GameManager): | 
					
						
							|  |  |  |             logging_pairs = [ | 
					
						
							|  |  |  |                 ("Client", "Archipelago") | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |             base_title = "Archipelago Text Client" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.ui = TextManager(self) | 
					
						
							|  |  |  |         self.ui_task = asyncio.create_task(self.ui.async_run(), name="UI") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def run_cli(self): | 
					
						
							|  |  |  |         if sys.stdin: | 
					
						
							|  |  |  |             # steam overlay breaks when starting console_loop | 
					
						
							|  |  |  |             if 'gameoverlayrenderer' in os.environ.get('LD_PRELOAD', ''): | 
					
						
							|  |  |  |                 logger.info("Skipping terminal input, due to conflicting Steam Overlay detected. Please use GUI only.") | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.input_task = asyncio.create_task(console_loop(self), name="Input") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-12 21:15:03 +02:00
										 |  |  | async def keep_alive(ctx: CommonContext, seconds_between_checks=100): | 
					
						
							|  |  |  |     """some ISPs/network configurations drop TCP connections if no payload is sent (ignore TCP-keep-alive)
 | 
					
						
							|  |  |  |      so we send a payload to prevent drop and if we were dropped anyway this will cause an auto-reconnect."""
 | 
					
						
							|  |  |  |     seconds_elapsed = 0 | 
					
						
							| 
									
										
										
										
											2021-09-11 03:59:12 +02:00
										 |  |  |     while not ctx.exit_event.is_set(): | 
					
						
							| 
									
										
										
										
											2021-09-12 21:15:03 +02:00
										 |  |  |         await asyncio.sleep(1)  # short sleep to not block program shutdown | 
					
						
							| 
									
										
										
										
											2021-09-11 03:59:12 +02:00
										 |  |  |         if ctx.server and ctx.slot: | 
					
						
							| 
									
										
										
										
											2021-09-12 21:15:03 +02:00
										 |  |  |             seconds_elapsed += 1 | 
					
						
							|  |  |  |             if seconds_elapsed > seconds_between_checks: | 
					
						
							|  |  |  |                 await ctx.send_msgs([{"cmd": "Bounce", "slots": [ctx.slot]}]) | 
					
						
							|  |  |  |                 seconds_elapsed = 0 | 
					
						
							| 
									
										
										
										
											2021-09-11 03:59:12 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | async def server_loop(ctx: CommonContext, address=None): | 
					
						
							|  |  |  |     if ctx.server and ctx.server.socket: | 
					
						
							|  |  |  |         logger.error('Already connected') | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if address is None:  # set through CLI or APBP | 
					
						
							|  |  |  |         address = ctx.server_address | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Wait for the user to provide a multiworld server address | 
					
						
							|  |  |  |     if not address: | 
					
						
							|  |  |  |         logger.info('Please connect to an Archipelago server.') | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     address = f"ws://{address}" if "://" not in address else address | 
					
						
							|  |  |  |     port = urllib.parse.urlparse(address).port or 38281 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     logger.info(f'Connecting to Archipelago server at {address}') | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         socket = await websockets.connect(address, port=port, ping_timeout=None, ping_interval=None) | 
					
						
							|  |  |  |         ctx.server = Endpoint(socket) | 
					
						
							|  |  |  |         logger.info('Connected') | 
					
						
							|  |  |  |         ctx.server_address = address | 
					
						
							|  |  |  |         ctx.current_reconnect_delay = ctx.starting_reconnect_delay | 
					
						
							|  |  |  |         async for data in ctx.server.socket: | 
					
						
							|  |  |  |             for msg in decode(data): | 
					
						
							|  |  |  |                 await process_server_cmd(ctx, msg) | 
					
						
							|  |  |  |         logger.warning('Disconnected from multiworld server, type /connect to reconnect') | 
					
						
							|  |  |  |     except ConnectionRefusedError: | 
					
						
							| 
									
										
										
										
											2022-05-21 22:24:49 +02:00
										 |  |  |         logger.exception('Connection refused by the server. May not be running Archipelago on that address or port.') | 
					
						
							| 
									
										
										
										
											2021-09-09 16:02:45 +02:00
										 |  |  |     except websockets.InvalidURI: | 
					
						
							|  |  |  |         logger.exception('Failed to connect to the multiworld server (invalid URI)') | 
					
						
							| 
									
										
										
										
											2022-05-21 22:24:49 +02:00
										 |  |  |     except OSError: | 
					
						
							| 
									
										
										
										
											2021-09-09 16:02:45 +02:00
										 |  |  |         logger.exception('Failed to connect to the multiworld server') | 
					
						
							| 
									
										
										
										
											2022-05-21 22:24:49 +02:00
										 |  |  |     except Exception: | 
					
						
							| 
									
										
										
										
											2021-09-09 16:02:45 +02:00
										 |  |  |         logger.exception('Lost connection to the multiworld server, type /connect to reconnect') | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |     finally: | 
					
						
							|  |  |  |         await ctx.connection_closed() | 
					
						
							|  |  |  |         if ctx.server_address: | 
					
						
							|  |  |  |             logger.info(f"... reconnecting in {ctx.current_reconnect_delay}s") | 
					
						
							| 
									
										
										
										
											2021-11-21 02:02:40 +01:00
										 |  |  |             asyncio.create_task(server_autoreconnect(ctx), name="server auto reconnect") | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         ctx.current_reconnect_delay *= 2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async def server_autoreconnect(ctx: CommonContext): | 
					
						
							|  |  |  |     await asyncio.sleep(ctx.current_reconnect_delay) | 
					
						
							|  |  |  |     if ctx.server_address and ctx.server_task is None: | 
					
						
							| 
									
										
										
										
											2021-11-21 02:02:40 +01:00
										 |  |  |         ctx.server_task = asyncio.create_task(server_loop(ctx), name="server loop") | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async def process_server_cmd(ctx: CommonContext, args: dict): | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         cmd = args["cmd"] | 
					
						
							|  |  |  |     except: | 
					
						
							|  |  |  |         logger.exception(f"Could not get command from {args}") | 
					
						
							|  |  |  |         raise | 
					
						
							|  |  |  |     if cmd == 'RoomInfo': | 
					
						
							| 
									
										
										
										
											2021-05-16 00:21:00 +02:00
										 |  |  |         if ctx.seed_name and ctx.seed_name != args["seed_name"]: | 
					
						
							|  |  |  |             logger.info("The server is running a different multiworld than your client is. (invalid seed_name)") | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2021-05-16 00:21:00 +02:00
										 |  |  |             logger.info('--------------------------------') | 
					
						
							|  |  |  |             logger.info('Room Information:') | 
					
						
							|  |  |  |             logger.info('--------------------------------') | 
					
						
							|  |  |  |             version = args["version"] | 
					
						
							|  |  |  |             ctx.server_version = tuple(version) | 
					
						
							|  |  |  |             version = ".".join(str(item) for item in version) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             logger.info(f'Server protocol version: {version}') | 
					
						
							|  |  |  |             logger.info("Server protocol tags: " + ", ".join(args["tags"])) | 
					
						
							|  |  |  |             if args['password']: | 
					
						
							|  |  |  |                 logger.info('Password required') | 
					
						
							| 
									
										
										
										
											2021-10-22 05:25:09 +02:00
										 |  |  |             ctx.update_permissions(args.get("permissions", {})) | 
					
						
							| 
									
										
										
										
											2021-10-30 07:33:05 +02:00
										 |  |  |             if "games" in args: | 
					
						
							|  |  |  |                 ctx.games = {x: game for x, game in enumerate(args["games"], start=1)} | 
					
						
							| 
									
										
										
										
											2021-08-04 18:38:49 +02:00
										 |  |  |             logger.info( | 
					
						
							|  |  |  |                 f"A !hint costs {args['hint_cost']}% of your total location count as points" | 
					
						
							|  |  |  |                 f" and you get {args['location_check_points']}" | 
					
						
							|  |  |  |                 f" for each location checked. Use !hint for more information.") | 
					
						
							| 
									
										
										
										
											2021-05-16 00:21:00 +02:00
										 |  |  |             ctx.hint_cost = int(args['hint_cost']) | 
					
						
							|  |  |  |             ctx.check_points = int(args['location_check_points']) | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-16 00:21:00 +02:00
										 |  |  |             if len(args['players']) < 1: | 
					
						
							|  |  |  |                 logger.info('No player connected') | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 args['players'].sort() | 
					
						
							|  |  |  |                 current_team = -1 | 
					
						
							| 
									
										
										
										
											2022-02-24 00:51:31 +01:00
										 |  |  |                 logger.info('Connected Players:') | 
					
						
							| 
									
										
										
										
											2021-05-16 00:21:00 +02:00
										 |  |  |                 for network_player in args['players']: | 
					
						
							|  |  |  |                     if network_player.team != current_team: | 
					
						
							|  |  |  |                         logger.info(f'  Team #{network_player.team + 1}') | 
					
						
							|  |  |  |                         current_team = network_player.team | 
					
						
							|  |  |  |                     logger.info('    %s (Player %d)' % (network_player.alias, network_player.slot)) | 
					
						
							|  |  |  |             if args["datapackage_version"] > network_data_package["version"] or args["datapackage_version"] == 0: | 
					
						
							|  |  |  |                 await ctx.send_msgs([{"cmd": "GetDataPackage"}]) | 
					
						
							|  |  |  |             await ctx.server_auth(args['password']) | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     elif cmd == 'DataPackage': | 
					
						
							|  |  |  |         logger.info("Got new ID/Name Datapackage") | 
					
						
							|  |  |  |         ctx.set_getters(args['data'], network=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     elif cmd == 'ConnectionRefused': | 
					
						
							|  |  |  |         errors = args["errors"] | 
					
						
							|  |  |  |         if 'InvalidSlot' in errors: | 
					
						
							|  |  |  |             ctx.event_invalid_slot() | 
					
						
							| 
									
										
										
										
											2021-09-17 04:32:09 +02:00
										 |  |  |         elif 'InvalidGame' in errors: | 
					
						
							|  |  |  |             ctx.event_invalid_game() | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         elif 'IncompatibleVersion' in errors: | 
					
						
							|  |  |  |             raise Exception('Server reported your client version as incompatible') | 
					
						
							| 
									
										
										
										
											2022-01-29 15:39:14 +01:00
										 |  |  |         elif 'InvalidItemsHandling' in errors: | 
					
						
							|  |  |  |             raise Exception('The item handling flags requested by the client are not supported') | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         # last to check, recoverable problem | 
					
						
							|  |  |  |         elif 'InvalidPassword' in errors: | 
					
						
							|  |  |  |             logger.error('Invalid password') | 
					
						
							|  |  |  |             ctx.password = None | 
					
						
							|  |  |  |             await ctx.server_auth(True) | 
					
						
							| 
									
										
										
										
											2021-06-25 07:25:03 +02:00
										 |  |  |         elif errors: | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |             raise Exception("Unknown connection errors: " + str(errors)) | 
					
						
							| 
									
										
										
										
											2021-06-25 07:25:03 +02:00
										 |  |  |         else: | 
					
						
							|  |  |  |             raise Exception('Connection refused by the multiworld host, no reason provided') | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     elif cmd == 'Connected': | 
					
						
							|  |  |  |         ctx.team = args["team"] | 
					
						
							|  |  |  |         ctx.slot = args["slot"] | 
					
						
							| 
									
										
										
										
											2022-05-24 00:20:02 +02:00
										 |  |  |         # int keys get lost in JSON transfer | 
					
						
							|  |  |  |         ctx.slot_info = {int(pid): data for pid, data in args["slot_info"].items()} | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         ctx.consume_players_package(args["players"]) | 
					
						
							|  |  |  |         msgs = [] | 
					
						
							|  |  |  |         if ctx.locations_checked: | 
					
						
							|  |  |  |             msgs.append({"cmd": "LocationChecks", | 
					
						
							|  |  |  |                          "locations": list(ctx.locations_checked)}) | 
					
						
							|  |  |  |         if ctx.locations_scouted: | 
					
						
							|  |  |  |             msgs.append({"cmd": "LocationScouts", | 
					
						
							|  |  |  |                          "locations": list(ctx.locations_scouted)}) | 
					
						
							|  |  |  |         if msgs: | 
					
						
							|  |  |  |             await ctx.send_msgs(msgs) | 
					
						
							|  |  |  |         if ctx.finished_game: | 
					
						
							| 
									
										
										
										
											2021-04-03 20:02:15 +02:00
										 |  |  |             await ctx.send_msgs([{"cmd": "StatusUpdate", "status": ClientStatus.CLIENT_GOAL}]) | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Get the server side view of missing as of time of connecting. | 
					
						
							|  |  |  |         # This list is used to only send to the server what is reported as ACTUALLY Missing. | 
					
						
							|  |  |  |         # This also serves to allow an easy visual of what locations were already checked previously | 
					
						
							|  |  |  |         # when /missing is used for the client side view of what is missing. | 
					
						
							| 
									
										
										
										
											2021-10-18 22:58:29 +02:00
										 |  |  |         ctx.missing_locations = set(args["missing_locations"]) | 
					
						
							|  |  |  |         ctx.checked_locations = set(args["checked_locations"]) | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     elif cmd == 'ReceivedItems': | 
					
						
							|  |  |  |         start_index = args["index"] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if start_index == 0: | 
					
						
							|  |  |  |             ctx.items_received = [] | 
					
						
							|  |  |  |         elif start_index != len(ctx.items_received): | 
					
						
							|  |  |  |             sync_msg = [{'cmd': 'Sync'}] | 
					
						
							|  |  |  |             if ctx.locations_checked: | 
					
						
							|  |  |  |                 sync_msg.append({"cmd": "LocationChecks", | 
					
						
							|  |  |  |                                  "locations": list(ctx.locations_checked)}) | 
					
						
							|  |  |  |             await ctx.send_msgs(sync_msg) | 
					
						
							|  |  |  |         if start_index == len(ctx.items_received): | 
					
						
							|  |  |  |             for item in args['items']: | 
					
						
							|  |  |  |                 ctx.items_received.append(NetworkItem(*item)) | 
					
						
							|  |  |  |         ctx.watcher_event.set() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     elif cmd == 'LocationInfo': | 
					
						
							| 
									
										
										
										
											2022-03-07 11:21:29 -08:00
										 |  |  |         for item in [NetworkItem(*item) for item in args['locations']]: | 
					
						
							| 
									
										
										
										
											2022-03-21 15:26:38 +01:00
										 |  |  |             ctx.locations_info[item.location] = item | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         ctx.watcher_event.set() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     elif cmd == "RoomUpdate": | 
					
						
							|  |  |  |         if "players" in args: | 
					
						
							|  |  |  |             ctx.consume_players_package(args["players"]) | 
					
						
							|  |  |  |         if "hint_points" in args: | 
					
						
							|  |  |  |             ctx.hint_points = args['hint_points'] | 
					
						
							| 
									
										
										
										
											2021-10-22 05:25:09 +02:00
										 |  |  |         if "checked_locations" in args: | 
					
						
							|  |  |  |             checked = set(args["checked_locations"]) | 
					
						
							|  |  |  |             ctx.checked_locations |= checked | 
					
						
							|  |  |  |             ctx.missing_locations -= checked | 
					
						
							|  |  |  |         if "permissions" in args: | 
					
						
							|  |  |  |             ctx.update_permissions(args["permissions"]) | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     elif cmd == 'Print': | 
					
						
							| 
									
										
										
										
											2021-04-13 14:49:32 +02:00
										 |  |  |         ctx.on_print(args) | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     elif cmd == 'PrintJSON': | 
					
						
							| 
									
										
										
										
											2021-04-13 14:49:32 +02:00
										 |  |  |         ctx.on_print_json(args) | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-14 10:02:39 +02:00
										 |  |  |     elif cmd == 'InvalidPacket': | 
					
						
							|  |  |  |         logger.warning(f"Invalid Packet of {args['type']}: {args['text']}") | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-02 01:35:24 +02:00
										 |  |  |     elif cmd == "Bounced": | 
					
						
							| 
									
										
										
										
											2021-11-01 19:37:47 +01:00
										 |  |  |         tags = args.get("tags", []) | 
					
						
							| 
									
										
										
										
											2021-11-02 11:11:57 +01:00
										 |  |  |         # we can skip checking "DeathLink" in ctx.tags, as otherwise we wouldn't have been send this | 
					
						
							| 
									
										
										
										
											2021-11-01 19:37:47 +01:00
										 |  |  |         if "DeathLink" in tags and ctx.last_death_link != args["data"]["time"]: | 
					
						
							|  |  |  |             ctx.on_deathlink(args["data"]) | 
					
						
							| 
									
										
										
										
											2022-02-24 04:47:01 +01:00
										 |  |  |     elif cmd == "SetReply": | 
					
						
							|  |  |  |         if args["key"] == "EnergyLink": | 
					
						
							|  |  |  |             ctx.current_energy_link_value = args["value"] | 
					
						
							|  |  |  |             if ctx.ui: | 
					
						
							|  |  |  |                 ctx.ui.set_new_energy_link_value() | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |     else: | 
					
						
							|  |  |  |         logger.debug(f"unknown command {cmd}") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-31 19:45:17 +02:00
										 |  |  |     ctx.on_package(cmd, args) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | async def console_loop(ctx: CommonContext): | 
					
						
							|  |  |  |     commandprocessor = ctx.command_processor(ctx) | 
					
						
							| 
									
										
										
										
											2021-11-28 03:27:18 +01:00
										 |  |  |     queue = asyncio.Queue() | 
					
						
							|  |  |  |     stream_input(sys.stdin, queue) | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |     while not ctx.exit_event.is_set(): | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2021-11-28 03:27:18 +01:00
										 |  |  |             input_text = await queue.get() | 
					
						
							| 
									
										
										
										
											2021-11-28 04:06:30 +01:00
										 |  |  |             queue.task_done() | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if ctx.input_requests > 0: | 
					
						
							|  |  |  |                 ctx.input_requests -= 1 | 
					
						
							|  |  |  |                 ctx.input_queue.put_nowait(input_text) | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if input_text: | 
					
						
							|  |  |  |                 commandprocessor(input_text) | 
					
						
							|  |  |  |         except Exception as e: | 
					
						
							| 
									
										
										
										
											2021-07-31 19:45:17 +02:00
										 |  |  |             logger.exception(e) | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-09 12:53:05 +01:00
										 |  |  | def get_base_parser(description=None): | 
					
						
							|  |  |  |     import argparse | 
					
						
							|  |  |  |     parser = argparse.ArgumentParser(description=description) | 
					
						
							|  |  |  |     parser.add_argument('--connect', default=None, help='Address of the multiworld host.') | 
					
						
							|  |  |  |     parser.add_argument('--password', default=None, help='Password of the multiworld host.') | 
					
						
							|  |  |  |     if sys.stdout:  # If terminal output exists, offer gui-less mode | 
					
						
							|  |  |  |         parser.add_argument('--nogui', default=False, action='store_true', help="Turns off Client GUI.") | 
					
						
							|  |  |  |     return parser | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     # Text Mode to use !hint and such with games that have no text entry | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     class TextContext(CommonContext): | 
					
						
							| 
									
										
										
										
											2022-01-21 02:51:25 +01:00
										 |  |  |         tags = {"AP", "IgnoreGame", "TextOnly"} | 
					
						
							| 
									
										
										
										
											2022-04-08 11:16:36 +02:00
										 |  |  |         game = ""  # empty matches any game since 0.3.2 | 
					
						
							| 
									
										
										
										
											2022-01-23 06:38:46 +01:00
										 |  |  |         items_handling = 0  # don't receive any NetworkItems | 
					
						
							| 
									
										
										
										
											2021-10-25 09:58:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  |         async def server_auth(self, password_requested: bool = False): | 
					
						
							|  |  |  |             if password_requested and not self.password: | 
					
						
							|  |  |  |                 await super(TextContext, self).server_auth(password_requested) | 
					
						
							|  |  |  |             if not self.auth: | 
					
						
							|  |  |  |                 logger.info('Enter slot name:') | 
					
						
							|  |  |  |                 self.auth = await self.console_input() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-21 02:50:24 +01:00
										 |  |  |             await self.send_connect() | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-30 07:33:05 +02:00
										 |  |  |         def on_package(self, cmd: str, args: dict): | 
					
						
							|  |  |  |             if cmd == "Connected": | 
					
						
							|  |  |  |                 self.game = self.games.get(self.slot, None) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-25 09:58:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  |     async def main(args): | 
					
						
							|  |  |  |         ctx = TextContext(args.connect, args.password) | 
					
						
							| 
									
										
										
										
											2022-05-24 00:20:02 +02:00
										 |  |  |         ctx.auth = args.name | 
					
						
							| 
									
										
										
										
											2021-11-21 02:02:40 +01:00
										 |  |  |         ctx.server_task = asyncio.create_task(server_loop(ctx), name="server loop") | 
					
						
							| 
									
										
										
										
											2022-04-06 00:06:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  |         if gui_enabled: | 
					
						
							| 
									
										
										
										
											2022-04-27 22:11:11 +02:00
										 |  |  |             ctx.run_gui() | 
					
						
							|  |  |  |         ctx.run_cli() | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-27 22:11:11 +02:00
										 |  |  |         await ctx.exit_event.wait() | 
					
						
							| 
									
										
										
										
											2021-11-21 02:02:40 +01:00
										 |  |  |         await ctx.shutdown() | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     import colorama | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-26 06:02:03 +01:00
										 |  |  |     parser = get_base_parser(description="Gameless Archipelago Client, for text interfacing.") | 
					
						
							| 
									
										
										
										
											2022-05-24 00:20:02 +02:00
										 |  |  |     parser.add_argument('--name', default=None, help="Slot Name to connect as.") | 
					
						
							|  |  |  |     parser.add_argument("url", nargs="?", help="Archipelago connection url") | 
					
						
							|  |  |  |     args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if args.url: | 
					
						
							|  |  |  |         url = urllib.parse.urlparse(args.url) | 
					
						
							|  |  |  |         args.connect = url.netloc | 
					
						
							| 
									
										
										
										
											2022-05-24 05:22:52 +02:00
										 |  |  |         if url.username: | 
					
						
							|  |  |  |             args.name = urllib.parse.unquote(url.username) | 
					
						
							|  |  |  |         if url.password: | 
					
						
							|  |  |  |             args.password = urllib.parse.unquote(url.password) | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     colorama.init() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-27 22:11:11 +02:00
										 |  |  |     asyncio.run(main(args)) | 
					
						
							| 
									
										
										
										
											2021-09-30 09:09:21 +02:00
										 |  |  |     colorama.deinit() |