| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | from __future__ import annotations | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | import argparse | 
					
						
							|  |  |  | import asyncio | 
					
						
							|  |  |  | import functools | 
					
						
							|  |  |  | import json | 
					
						
							|  |  |  | import logging | 
					
						
							| 
									
										
										
										
											2020-01-04 22:08:13 +01:00
										 |  |  | import zlib | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  | import collections | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | import typing | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  | import inspect | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  | import weakref | 
					
						
							| 
									
										
										
										
											2020-04-20 04:36:56 +02:00
										 |  |  | import datetime | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 15:45:52 +01:00
										 |  |  | import ModuleUpdate | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-18 15:45:52 +01:00
										 |  |  | ModuleUpdate.update() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import websockets | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  | import prompt_toolkit | 
					
						
							|  |  |  | from prompt_toolkit.patch_stdout import patch_stdout | 
					
						
							| 
									
										
										
										
											2020-02-17 13:57:48 +01:00
										 |  |  | from fuzzywuzzy import process as fuzzy_process | 
					
						
							| 
									
										
										
										
											2020-01-18 15:45:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | import Items | 
					
						
							|  |  |  | import Regions | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  | import Utils | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | from Utils import get_item_name_from_id, get_location_name_from_address, ReceivedItem | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 13:57:48 +01:00
										 |  |  | console_names = frozenset(set(Items.item_table) | set(Regions.location_table)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  | CLIENT_PLAYING = 0 | 
					
						
							|  |  |  | CLIENT_GOAL = 1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | class Client: | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     version: typing.List[int] = [0, 0, 0] | 
					
						
							|  |  |  |     tags: typing.List[str] = [] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |     def __init__(self, socket: websockets.server.WebSocketServerProtocol, ctx: Context): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         self.socket = socket | 
					
						
							|  |  |  |         self.auth = False | 
					
						
							|  |  |  |         self.name = None | 
					
						
							|  |  |  |         self.team = None | 
					
						
							|  |  |  |         self.slot = None | 
					
						
							|  |  |  |         self.send_index = 0 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |         self.tags = [] | 
					
						
							|  |  |  |         self.version = [0, 0, 0] | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         self.messageprocessor = ClientMessageProcessor(ctx, self) | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |         self.ctx = weakref.ref(ctx) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     async def disconnect(self): | 
					
						
							|  |  |  |         ctx = self.ctx() | 
					
						
							|  |  |  |         if ctx: | 
					
						
							|  |  |  |             await on_client_disconnected(ctx, self) | 
					
						
							|  |  |  |             ctx.clients.remove(self) | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def wants_item_notification(self): | 
					
						
							|  |  |  |         return self.auth and "FoundItems" in self.tags | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | class Context: | 
					
						
							| 
									
										
										
										
											2020-02-22 19:42:44 +01:00
										 |  |  |     def __init__(self, host: str, port: int, password: str, location_check_points: int, hint_cost: int, | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |                  item_cheat: bool, forfeit_mode: str = "disabled", remaining_mode: str = "disabled"): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         self.data_filename = None | 
					
						
							|  |  |  |         self.save_filename = None | 
					
						
							|  |  |  |         self.disable_save = False | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         self.player_names = {} | 
					
						
							| 
									
										
										
										
											2020-01-04 22:08:13 +01:00
										 |  |  |         self.rom_names = {} | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |         self.remote_items = set() | 
					
						
							| 
									
										
										
										
											2020-01-04 22:08:13 +01:00
										 |  |  |         self.locations = {} | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         self.host = host | 
					
						
							|  |  |  |         self.port = port | 
					
						
							|  |  |  |         self.password = password | 
					
						
							|  |  |  |         self.server = None | 
					
						
							| 
									
										
										
										
											2020-01-10 22:44:07 +01:00
										 |  |  |         self.countdown_timer = 0 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         self.clients = [] | 
					
						
							|  |  |  |         self.received_items = {} | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |         self.name_aliases: typing.Dict[typing.Tuple[int, int], str] = {} | 
					
						
							| 
									
										
										
										
											2020-02-09 12:10:12 +01:00
										 |  |  |         self.location_checks = collections.defaultdict(set) | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  |         self.hint_cost = hint_cost | 
					
						
							|  |  |  |         self.location_check_points = location_check_points | 
					
						
							| 
									
										
										
										
											2020-02-22 18:04:35 +01:00
										 |  |  |         self.hints_used = collections.defaultdict(int) | 
					
						
							| 
									
										
										
										
											2020-04-22 05:09:46 +02:00
										 |  |  |         self.hints: typing.Dict[typing.Tuple[int, int], typing.Set[Utils.Hint]] = collections.defaultdict(set) | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |         self.forfeit_mode: str = forfeit_mode | 
					
						
							|  |  |  |         self.remaining_mode: str = remaining_mode | 
					
						
							| 
									
										
										
										
											2020-02-22 19:42:44 +01:00
										 |  |  |         self.item_cheat = item_cheat | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |         self.running = True | 
					
						
							| 
									
										
										
										
											2020-04-20 04:36:56 +02:00
										 |  |  |         self.client_activity_timers = {} | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |         self.client_game_state: typing.Dict[typing.Tuple[int, int], int] = collections.defaultdict(int) | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         self.commandprocessor = ServerCommandProcessor(self) | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def get_save(self) -> dict: | 
					
						
							| 
									
										
										
										
											2020-04-22 15:50:14 +02:00
										 |  |  |         d = { | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  |             "rom_names": list(self.rom_names.items()), | 
					
						
							| 
									
										
										
										
											2020-02-22 19:42:44 +01:00
										 |  |  |             "received_items": tuple((k, v) for k, v in self.received_items.items()), | 
					
						
							| 
									
										
										
										
											2020-04-22 05:09:46 +02:00
										 |  |  |             "hints_used": tuple((key, value) for key, value in self.hints_used.items()), | 
					
						
							| 
									
										
										
										
											2020-04-22 15:50:14 +02:00
										 |  |  |             "hints": tuple((key, list(value)) for key, value in self.hints.items()), | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |             "location_checks": tuple((key, tuple(value)) for key, value in self.location_checks.items()), | 
					
						
							|  |  |  |             "name_aliases": tuple((key, value) for key, value in self.name_aliases.items()), | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |             "client_game_state": tuple((key, value) for key, value in self.client_game_state.items()) | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-22 15:50:14 +02:00
										 |  |  |         return d | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  |     def set_save(self, savedata: dict): | 
					
						
							|  |  |  |         rom_names = savedata["rom_names"] | 
					
						
							| 
									
										
										
										
											2020-02-22 19:42:44 +01:00
										 |  |  |         received_items = {tuple(k): [ReceivedItem(*i) for i in v] for k, v in savedata["received_items"]} | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  |         if not all([self.rom_names[tuple(rom)] == (team, slot) for rom, (team, slot) in rom_names]): | 
					
						
							|  |  |  |             raise Exception('Save file mismatch, will start a new game') | 
					
						
							|  |  |  |         self.received_items = received_items | 
					
						
							|  |  |  |         self.hints_used.update({tuple(key): value for key, value in savedata["hints_used"]}) | 
					
						
							| 
									
										
										
										
											2020-04-22 05:09:46 +02:00
										 |  |  |         if "hints" in savedata: | 
					
						
							|  |  |  |             self.hints.update( | 
					
						
							|  |  |  |                 {tuple(key): set(Utils.Hint(*hint) for hint in value) for key, value in savedata["hints"]}) | 
					
						
							|  |  |  |         else:  # backwards compatiblity for <= 2.0.2 | 
					
						
							|  |  |  |             old_hints = {tuple(key): set(value) for key, value in savedata["hints_sent"]} | 
					
						
							|  |  |  |             for team_slot, item_or_location_s in old_hints.items(): | 
					
						
							|  |  |  |                 team, slot = team_slot | 
					
						
							|  |  |  |                 for item_or_location in item_or_location_s: | 
					
						
							|  |  |  |                     if item_or_location in Items.item_table: | 
					
						
							|  |  |  |                         hints = collect_hints(self, team, slot, item_or_location) | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         hints = collect_hints_location(self, team, slot, item_or_location) | 
					
						
							|  |  |  |                     for hint in hints: | 
					
						
							|  |  |  |                         self.hints[team, hint.receiving_player].add(hint) | 
					
						
							|  |  |  |                         # even if it is the same hint, it won't be duped due to set | 
					
						
							|  |  |  |                         self.hints[team, hint.finding_player].add(hint) | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |         if "name_aliases" in savedata: | 
					
						
							|  |  |  |             self.name_aliases.update({tuple(key): value for key, value in savedata["name_aliases"]}) | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |             if "client_game_state" in savedata: | 
					
						
							|  |  |  |                 self.client_game_state.update({tuple(key): value for key, value in savedata["client_game_state"]}) | 
					
						
							| 
									
										
										
										
											2020-02-09 12:10:12 +01:00
										 |  |  |         self.location_checks.update({tuple(key): set(value) for key, value in savedata["location_checks"]}) | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  |         logging.info(f'Loaded save file with {sum([len(p) for p in received_items.values()])} received items ' | 
					
						
							|  |  |  |                      f'for {len(received_items)} players') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |     def get_aliased_name(self, team: int, slot: int): | 
					
						
							|  |  |  |         if (team, slot) in self.name_aliases: | 
					
						
							|  |  |  |             return f"{self.name_aliases[team, slot]} ({self.player_names[team, slot]})" | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return self.player_names[team, slot] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  | async def send_msgs(client: Client, msgs): | 
					
						
							|  |  |  |     websocket = client.socket | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     if not websocket or not websocket.open or websocket.closed: | 
					
						
							|  |  |  |         return | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |     try: | 
					
						
							|  |  |  |         await websocket.send(json.dumps(msgs)) | 
					
						
							|  |  |  |     except websockets.ConnectionClosed: | 
					
						
							|  |  |  |         logging.exception("Exception during send_msgs") | 
					
						
							|  |  |  |         await client.disconnect() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | async def send_json_msgs(client: Client, msg: str): | 
					
						
							|  |  |  |     websocket = client.socket | 
					
						
							|  |  |  |     if not websocket or not websocket.open or websocket.closed: | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         await websocket.send(msg) | 
					
						
							|  |  |  |     except websockets.ConnectionClosed: | 
					
						
							|  |  |  |         logging.exception("Exception during send_msgs") | 
					
						
							|  |  |  |         await client.disconnect() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def broadcast_all(ctx: Context, msgs): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     for client in ctx.clients: | 
					
						
							|  |  |  |         if client.auth: | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |             asyncio.create_task(send_msgs(client, msgs)) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | def broadcast_team(ctx: Context, team, msgs): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     for client in ctx.clients: | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         if client.auth and client.team == team: | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |             asyncio.create_task(send_msgs(client, msgs)) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | def notify_all(ctx : Context, text): | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |     logging.info("Notice (all): %s" % text) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     broadcast_all(ctx, [['Print', text]]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | def notify_team(ctx: Context, team: int, text: str): | 
					
						
							|  |  |  |     logging.info("Notice (Team #%d): %s" % (team + 1, text)) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     broadcast_team(ctx, team, [['Print', text]]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | def notify_client(client: Client, text: str): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     if not client.auth: | 
					
						
							|  |  |  |         return | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     logging.info("Notice (Player %s in team %d): %s" % (client.name, client.team + 1, text)) | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |     asyncio.create_task(send_msgs(client, [['Print', text]])) | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  | # separated out, due to compatibilty between clients | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | def notify_hints(ctx: Context, team: int, hints: typing.List[Utils.Hint]): | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |     cmd = json.dumps([["Hint", hints]])  # make sure it is a list, as it can be set internally | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |     texts = [['Print', format_hint(ctx, team, hint)] for hint in hints] | 
					
						
							| 
									
										
										
										
											2020-02-17 05:34:02 +01:00
										 |  |  |     for _, text in texts: | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |         logging.info("Notice (Team #%d): %s" % (team + 1, text)) | 
					
						
							|  |  |  |     for client in ctx.clients: | 
					
						
							|  |  |  |         if client.auth and client.team == team: | 
					
						
							|  |  |  |             if "Berserker" in client.tags: | 
					
						
							|  |  |  |                 payload = cmd | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |                 asyncio.create_task(send_json_msgs(client, payload)) | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 payload = texts | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |                 asyncio.create_task(send_msgs(client, payload)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def update_aliases(ctx: Context, team: int, client: typing.Optional[Client] = None): | 
					
						
							|  |  |  |     cmd = json.dumps([["AliasUpdate", | 
					
						
							|  |  |  |                        [(key[1], ctx.get_aliased_name(*key)) for key, value in ctx.player_names.items() if | 
					
						
							|  |  |  |                         key[0] == team]]])  # make sure it is a list, as it can be set internally | 
					
						
							|  |  |  |     if client is None: | 
					
						
							|  |  |  |         for client in ctx.clients: | 
					
						
							|  |  |  |             if client.team == team and client.auth and client.version > [2, 0, 3]: | 
					
						
							|  |  |  |                 asyncio.create_task(send_json_msgs(client, cmd)) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         asyncio.create_task(send_json_msgs(client, cmd)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | async def server(websocket, path, ctx: Context): | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |     client = Client(websocket, ctx) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     ctx.clients.append(client) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         await on_client_connected(ctx, client) | 
					
						
							|  |  |  |         async for data in websocket: | 
					
						
							|  |  |  |             for msg in json.loads(data): | 
					
						
							|  |  |  |                 if len(msg) == 1: | 
					
						
							|  |  |  |                     cmd = msg | 
					
						
							|  |  |  |                     args = None | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     cmd = msg[0] | 
					
						
							|  |  |  |                     args = msg[1] | 
					
						
							|  |  |  |                 await process_client_cmd(ctx, client, cmd, args) | 
					
						
							|  |  |  |     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) | 
					
						
							|  |  |  |     finally: | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |         await client.disconnect() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | async def on_client_connected(ctx: Context, client: Client): | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |     await send_msgs(client, [['RoomInfo', { | 
					
						
							| 
									
										
										
										
											2020-01-15 03:00:30 +01:00
										 |  |  |         'password': ctx.password is not None, | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |         'players': [(client.team, client.slot, ctx.name_aliases.get((client.team, client.slot), client.name)) for client | 
					
						
							|  |  |  |                     in ctx.clients if client.auth], | 
					
						
							| 
									
										
										
										
											2020-02-16 15:35:01 +01:00
										 |  |  |         # tags are for additional features in the communication. | 
					
						
							|  |  |  |         # Name them by feature or fork, as you feel is appropriate. | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |         'tags': ['Berserker'], | 
					
						
							| 
									
										
										
										
											2020-04-20 14:50:49 +02:00
										 |  |  |         'version': Utils._version_tuple | 
					
						
							| 
									
										
										
										
											2020-01-15 03:00:30 +01:00
										 |  |  |     }]]) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | async def on_client_disconnected(ctx: Context, client: Client): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     if client.auth: | 
					
						
							|  |  |  |         await on_client_left(ctx, client) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | async def on_client_joined(ctx: Context, client: Client): | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |     notify_all(ctx, | 
					
						
							|  |  |  |                "%s (Team #%d) has joined the game. Client(%s, %s)." % (ctx.get_aliased_name(client.team, client.slot), | 
					
						
							|  |  |  |                                                                        client.team + 1, | 
					
						
							|  |  |  |                                                                        ".".join(str(x) for x in client.version), | 
					
						
							|  |  |  |                                                                        client.tags)) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | async def on_client_left(ctx: Context, client: Client): | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |     notify_all(ctx, "%s (Team #%d) has left the game" % (client.name, client.team + 1)) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | async def countdown(ctx: Context, timer): | 
					
						
							| 
									
										
										
										
											2020-01-10 22:44:07 +01:00
										 |  |  |     notify_all(ctx, f'[Server]: Starting countdown of {timer}s') | 
					
						
							|  |  |  |     if ctx.countdown_timer: | 
					
						
							| 
									
										
										
										
											2020-03-11 23:08:16 +01:00
										 |  |  |         ctx.countdown_timer = timer  # timer is already running, set it to a different time | 
					
						
							|  |  |  |     else: | 
					
						
							| 
									
										
										
										
											2020-01-10 22:44:07 +01:00
										 |  |  |         ctx.countdown_timer = timer | 
					
						
							| 
									
										
										
										
											2020-03-11 23:08:16 +01:00
										 |  |  |         while ctx.countdown_timer > 0: | 
					
						
							|  |  |  |             notify_all(ctx, f'[Server]: {ctx.countdown_timer}') | 
					
						
							|  |  |  |             ctx.countdown_timer -= 1 | 
					
						
							|  |  |  |             await asyncio.sleep(1) | 
					
						
							|  |  |  |         notify_all(ctx, f'[Server]: GO') | 
					
						
							| 
									
										
										
										
											2020-01-10 22:44:07 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 14:51:48 +02:00
										 |  |  | def get_players_string(ctx: Context): | 
					
						
							| 
									
										
										
										
											2020-03-11 09:15:39 +01:00
										 |  |  |     auth_clients = {(c.team, c.slot) for c in ctx.clients if c.auth} | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-11 09:15:39 +01:00
										 |  |  |     player_names = sorted(ctx.player_names.keys()) | 
					
						
							| 
									
										
										
										
											2020-03-05 02:31:26 +01:00
										 |  |  |     current_team = -1 | 
					
						
							|  |  |  |     text = '' | 
					
						
							| 
									
										
										
										
											2020-03-11 09:15:39 +01:00
										 |  |  |     for team, slot in player_names: | 
					
						
							|  |  |  |         player_name = ctx.player_names[team, slot] | 
					
						
							|  |  |  |         if team != current_team: | 
					
						
							|  |  |  |             text += f':: Team #{team + 1}: ' | 
					
						
							|  |  |  |             current_team = team | 
					
						
							|  |  |  |         if (team, slot) in auth_clients: | 
					
						
							|  |  |  |             text += f'{player_name} ' | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             text += f'({player_name}) ' | 
					
						
							|  |  |  |     return f'{len(auth_clients)} players of {len(ctx.player_names)} connected ' + text[:-1] | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  | def get_received_items(ctx: Context, team: int, player: int) -> typing.List[ReceivedItem]: | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |     return ctx.received_items.setdefault((team, player), []) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | def tuplize_received_items(items): | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |     return [(item.item, item.location, item.player) for item in items] | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | def send_new_items(ctx: Context): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     for client in ctx.clients: | 
					
						
							|  |  |  |         if not client.auth: | 
					
						
							|  |  |  |             continue | 
					
						
							|  |  |  |         items = get_received_items(ctx, client.team, client.slot) | 
					
						
							|  |  |  |         if len(items) > client.send_index: | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |             asyncio.create_task(send_msgs(client, [ | 
					
						
							|  |  |  |                 ['ReceivedItems', (client.send_index, tuplize_received_items(items)[client.send_index:])]])) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             client.send_index = len(items) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | def forfeit_player(ctx: Context, team: int, slot: int): | 
					
						
							|  |  |  |     all_locations = {values[0] for values in Regions.location_table.values() if type(values[0]) is int} | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |     notify_all(ctx, "%s (Team #%d) has forfeited" % (ctx.player_names[(team, slot)], team + 1)) | 
					
						
							|  |  |  |     register_location_checks(ctx, team, slot, all_locations) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  | def get_remaining(ctx: Context, team: int, slot: int) -> typing.List[int]: | 
					
						
							|  |  |  |     items = [] | 
					
						
							|  |  |  |     for (location, location_slot) in ctx.locations: | 
					
						
							|  |  |  |         if location_slot == slot and location not in ctx.location_checks[team, slot]: | 
					
						
							|  |  |  |             items.append(ctx.locations[location, slot][0])  # item ID | 
					
						
							|  |  |  |     return sorted(items) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | def register_location_checks(ctx: Context, team: int, slot: int, locations): | 
					
						
							| 
									
										
										
										
											2020-04-20 04:36:56 +02:00
										 |  |  |     ctx.client_activity_timers[team, slot] = datetime.datetime.now(datetime.timezone.utc) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     found_items = False | 
					
						
							|  |  |  |     for location in locations: | 
					
						
							| 
									
										
										
										
											2020-01-04 22:08:13 +01:00
										 |  |  |         if (location, slot) in ctx.locations: | 
					
						
							|  |  |  |             target_item, target_player = ctx.locations[(location, slot)] | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |             if target_player != slot or slot in ctx.remote_items: | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |                 found = False | 
					
						
							|  |  |  |                 recvd_items = get_received_items(ctx, team, target_player) | 
					
						
							|  |  |  |                 for recvd_item in recvd_items: | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |                     if recvd_item.location == location and recvd_item.player == slot: | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |                         found = True | 
					
						
							|  |  |  |                         break | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |                 if not found: | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |                     new_item = ReceivedItem(target_item, location, slot) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |                     recvd_items.append(new_item) | 
					
						
							| 
									
										
										
										
											2020-01-18 11:26:45 +01:00
										 |  |  |                     if slot != target_player: | 
					
						
							|  |  |  |                         broadcast_team(ctx, team, [['ItemSent', (slot, location, target_player, target_item)]]) | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  |                     logging.info('(Team #%d) %s sent %s to %s (%s)' % ( | 
					
						
							|  |  |  |                     team + 1, ctx.player_names[(team, slot)], get_item_name_from_id(target_item), | 
					
						
							|  |  |  |                     ctx.player_names[(team, target_player)], get_location_name_from_address(location))) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |                     found_items = True | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  |             elif target_player == slot:  # local pickup, notify clients of the pickup | 
					
						
							|  |  |  |                 if location not in ctx.location_checks[team, slot]: | 
					
						
							|  |  |  |                     for client in ctx.clients: | 
					
						
							|  |  |  |                         if client.team == team and client.wants_item_notification: | 
					
						
							|  |  |  |                             asyncio.create_task( | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |                                 send_msgs(client, [['ItemFound', (target_item, location, slot)]])) | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  |     ctx.location_checks[team, slot] |= set(locations) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     send_new_items(ctx) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  |     if found_items: | 
					
						
							|  |  |  |         save(ctx) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | def save(ctx: Context): | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  |     if not ctx.disable_save: | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2020-02-22 19:42:44 +01:00
										 |  |  |             jsonstr = json.dumps(ctx.get_save()) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             with open(ctx.save_filename, "wb") as f: | 
					
						
							| 
									
										
										
										
											2020-01-04 22:08:13 +01:00
										 |  |  |                 f.write(zlib.compress(jsonstr.encode("utf-8"))) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         except Exception as e: | 
					
						
							|  |  |  |             logging.exception(e) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | def collect_hints(ctx: Context, team: int, slot: int, item: str) -> typing.List[Utils.Hint]: | 
					
						
							| 
									
										
										
										
											2020-02-11 00:44:28 +01:00
										 |  |  |     hints = [] | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  |     seeked_item_id = Items.item_table[item][3] | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  |     for check, result in ctx.locations.items(): | 
					
						
							|  |  |  |         item_id, receiving_player = result | 
					
						
							|  |  |  |         if receiving_player == slot and item_id == seeked_item_id: | 
					
						
							|  |  |  |             location_id, finding_player = check | 
					
						
							| 
									
										
										
										
											2020-02-11 00:44:28 +01:00
										 |  |  |             found = location_id in ctx.location_checks[team, finding_player] | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |             hints.append(Utils.Hint(receiving_player, finding_player, location_id, item_id, found)) | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 00:44:28 +01:00
										 |  |  |     return hints | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 08:16:17 +01:00
										 |  |  | def collect_hints_location(ctx: Context, team: int, slot: int, location: str) -> typing.List[Utils.Hint]: | 
					
						
							|  |  |  |     hints = [] | 
					
						
							|  |  |  |     seeked_location = Regions.location_table[location][0] | 
					
						
							|  |  |  |     for check, result in ctx.locations.items(): | 
					
						
							|  |  |  |         location_id, finding_player = check | 
					
						
							|  |  |  |         if finding_player == slot and location_id == seeked_location: | 
					
						
							|  |  |  |             item_id, receiving_player = result | 
					
						
							|  |  |  |             found = location_id in ctx.location_checks[team, finding_player] | 
					
						
							|  |  |  |             hints.append(Utils.Hint(receiving_player, finding_player, location_id, item_id, found)) | 
					
						
							|  |  |  |             break # each location has 1 item | 
					
						
							|  |  |  |     return hints | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | def format_hint(ctx: Context, team: int, hint: Utils.Hint) -> str: | 
					
						
							|  |  |  |     return f"[Hint]: {ctx.player_names[team, hint.receiving_player]}'s " \ | 
					
						
							|  |  |  |            f"{Items.lookup_id_to_name[hint.item]} can be found " \ | 
					
						
							|  |  |  |            f"at {get_location_name_from_address(hint.location)} " \ | 
					
						
							|  |  |  |            f"in {ctx.player_names[team, hint.finding_player]}'s World." \ | 
					
						
							|  |  |  |            + (" (found)" if hint.found else "") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 13:57:48 +01:00
										 |  |  | def get_intended_text(input_text: str, possible_answers: typing.Iterable[str]= console_names) -> typing.Tuple[str, bool, str]: | 
					
						
							|  |  |  |     picks = fuzzy_process.extract(input_text, possible_answers, limit=2) | 
					
						
							| 
									
										
										
										
											2020-04-21 21:53:20 +02:00
										 |  |  |     if len(picks) > 1: | 
					
						
							|  |  |  |         dif = picks[0][1] - picks[1][1] | 
					
						
							|  |  |  |         if picks[0][1] == 100: | 
					
						
							|  |  |  |             return picks[0][0], True, "Perfect Match" | 
					
						
							|  |  |  |         elif picks[0][1] < 75: | 
					
						
							|  |  |  |             return picks[0][0], False, f"Didn't find something that closely matches, " \ | 
					
						
							|  |  |  |                                        f"did you mean {picks[0][0]}? ({picks[0][1]}% sure)" | 
					
						
							|  |  |  |         elif dif > 5: | 
					
						
							|  |  |  |             return picks[0][0], True, "Close Match" | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return picks[0][0], False, f"Too many close matches, did you mean {picks[0][0]}? ({picks[0][1]}% sure)" | 
					
						
							| 
									
										
										
										
											2020-02-17 13:57:48 +01:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2020-04-21 21:53:20 +02:00
										 |  |  |         if picks[0][1] > 90: | 
					
						
							|  |  |  |             return picks[0][0], True, "Only Option Match" | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return picks[0][0], False, f"Did you mean {picks[0][0]}? ({picks[0][1]}% sure)" | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | class CommandMeta(type): | 
					
						
							|  |  |  |     def __new__(cls, name, bases, attrs): | 
					
						
							|  |  |  |         commands = attrs["commands"] = {} | 
					
						
							|  |  |  |         for base in bases: | 
					
						
							|  |  |  |             commands.update(base.commands) | 
					
						
							|  |  |  |         commands.update({name[5:].lower(): method for name, method in attrs.items() if | 
					
						
							|  |  |  |                          name.startswith("_cmd_")}) | 
					
						
							|  |  |  |         return super(CommandMeta, cls).__new__(cls, name, bases, attrs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 03:24:27 +02:00
										 |  |  | def mark_raw(function): | 
					
						
							|  |  |  |     function.raw_text = True | 
					
						
							|  |  |  |     return function | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | class CommandProcessor(metaclass=CommandMeta): | 
					
						
							|  |  |  |     commands: typing.Dict[str, typing.Callable] | 
					
						
							|  |  |  |     marker = "/" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def output(self, text: str): | 
					
						
							|  |  |  |         print(text) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def __call__(self, raw: str) -> typing.Optional[bool]: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         if not raw: | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2020-04-15 08:49:30 +02:00
										 |  |  |         try: | 
					
						
							|  |  |  |             command = raw.split() | 
					
						
							|  |  |  |             basecommand = command[0] | 
					
						
							|  |  |  |             if basecommand[0] == self.marker: | 
					
						
							|  |  |  |                 method = self.commands.get(basecommand[1:].lower(), None) | 
					
						
							|  |  |  |                 if not method: | 
					
						
							|  |  |  |                     self._error_unknown_command(basecommand[1:]) | 
					
						
							|  |  |  |                 else: | 
					
						
							| 
									
										
										
										
											2020-04-19 15:32:27 +02:00
										 |  |  |                     if getattr(method, "raw_text", False):  # method is requesting unprocessed text data | 
					
						
							| 
									
										
										
										
											2020-04-19 15:31:15 +02:00
										 |  |  |                         arg = raw.split(maxsplit=1) | 
					
						
							|  |  |  |                         if len(arg) > 1: | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                             return method(self, arg[1])  # argument text was found, so pass it along | 
					
						
							| 
									
										
										
										
											2020-04-19 15:31:15 +02:00
										 |  |  |                         else: | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                             return method(self)  # argument may be optional, try running without args | 
					
						
							| 
									
										
										
										
											2020-04-19 03:24:27 +02:00
										 |  |  |                     else: | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                         return method(self, *command[1:])  # pass each word as argument | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2020-04-15 08:49:30 +02:00
										 |  |  |                 self.default(raw) | 
					
						
							|  |  |  |         except Exception as e: | 
					
						
							|  |  |  |             self._error_parsing_command(e) | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def get_help_text(self) -> str: | 
					
						
							|  |  |  |         s = "" | 
					
						
							|  |  |  |         for command, method in self.commands.items(): | 
					
						
							|  |  |  |             spec = inspect.signature(method).parameters | 
					
						
							|  |  |  |             argtext = "" | 
					
						
							|  |  |  |             for argname, parameter in spec.items(): | 
					
						
							|  |  |  |                 if argname == "self": | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2020-04-15 09:56:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |                 if isinstance(parameter.default, str): | 
					
						
							| 
									
										
										
										
											2020-04-15 09:56:28 +02:00
										 |  |  |                     if not parameter.default: | 
					
						
							|  |  |  |                         argname = f"[{argname}]" | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         argname += "=" + parameter.default | 
					
						
							|  |  |  |                 argtext += argname | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |                 argtext += " " | 
					
						
							|  |  |  |             s += f"{self.marker}{command} {argtext}\n    {method.__doc__}\n" | 
					
						
							|  |  |  |         return s | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _cmd_help(self): | 
					
						
							|  |  |  |         """Returns the help listing""" | 
					
						
							|  |  |  |         self.output(self.get_help_text()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _cmd_license(self): | 
					
						
							|  |  |  |         """Returns the licensing information""" | 
					
						
							| 
									
										
										
										
											2020-04-20 21:15:13 +02:00
										 |  |  |         license = getattr(CommandProcessor, "license", None) | 
					
						
							|  |  |  |         if not license: | 
					
						
							|  |  |  |             with open(Utils.local_path("LICENSE")) as f: | 
					
						
							|  |  |  |                 CommandProcessor.license = license = f.read() | 
					
						
							|  |  |  |         self.output(CommandProcessor.license) | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def default(self, raw: str): | 
					
						
							|  |  |  |         self.output("Echo: " + raw) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _error_unknown_command(self, raw: str): | 
					
						
							|  |  |  |         self.output(f"Could not find command {raw}. Known commands: {', '.join(self.commands)}") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-15 08:49:30 +02:00
										 |  |  |     def _error_parsing_command(self, exception: Exception): | 
					
						
							|  |  |  |         self.output(str(exception)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | class ClientMessageProcessor(CommandProcessor): | 
					
						
							|  |  |  |     marker = "!" | 
					
						
							|  |  |  |     ctx: Context | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, ctx: Context, client: Client): | 
					
						
							|  |  |  |         self.ctx = ctx | 
					
						
							|  |  |  |         self.client = client | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def output(self, text): | 
					
						
							|  |  |  |         notify_client(self.client, text) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:26:22 +02:00
										 |  |  |     def default(self, raw: str): | 
					
						
							|  |  |  |         pass  # default is client sending just text | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_players(self) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """Get information about connected and missing players""" | 
					
						
							| 
									
										
										
										
											2020-04-20 23:03:52 +02:00
										 |  |  |         if len(self.ctx.player_names) < 10: | 
					
						
							|  |  |  |             notify_all(self.ctx, get_players_string(self.ctx)) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.output(get_players_string(self.ctx)) | 
					
						
							| 
									
										
										
										
											2020-04-24 05:29:02 +02:00
										 |  |  |         return True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_forfeit(self) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """Surrender and send your remaining items out to their recipients""" | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |         if self.ctx.forfeit_mode == "enabled": | 
					
						
							| 
									
										
										
										
											2020-04-16 15:40:31 +02:00
										 |  |  |             forfeit_player(self.ctx, self.client.team, self.client.slot) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |             return True | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |         elif self.ctx.forfeit_mode == "disabled": | 
					
						
							| 
									
										
										
										
											2020-04-16 15:40:31 +02:00
										 |  |  |             self.output( | 
					
						
							|  |  |  |                 "Sorry, client forfeiting has been disabled on this server. You can ask the server admin for a /forfeit") | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |         else:  # is auto or goal | 
					
						
							|  |  |  |             if self.ctx.client_game_state[self.client.team, self.client.slot] == CLIENT_GOAL: | 
					
						
							|  |  |  |                 forfeit_player(self.ctx, self.client.team, self.client.slot) | 
					
						
							|  |  |  |                 return True | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.output( | 
					
						
							|  |  |  |                     "Sorry, client forfeiting requires you to have beaten the game on this server." | 
					
						
							|  |  |  |                     " You can ask the server admin for a /forfeit") | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _cmd_remaining(self) -> bool: | 
					
						
							|  |  |  |         """List remaining items in your game, but not their location or recipient""" | 
					
						
							|  |  |  |         if self.ctx.remaining_mode == "enabled": | 
					
						
							|  |  |  |             remaining_item_ids = get_remaining(self.ctx, self.client.team, self.client.slot) | 
					
						
							|  |  |  |             if remaining_item_ids: | 
					
						
							|  |  |  |                 self.output("Remaining items: " + ", ".join(Items.lookup_id_to_name.get(item_id, "unknown item") | 
					
						
							|  |  |  |                                                             for item_id in remaining_item_ids)) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.output("No remaining items found.") | 
					
						
							|  |  |  |             return True | 
					
						
							|  |  |  |         elif self.ctx.remaining_mode == "disabled": | 
					
						
							|  |  |  |             self.output( | 
					
						
							|  |  |  |                 "Sorry, !remaining has been disabled on this server.") | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         else:  # is goal | 
					
						
							|  |  |  |             if self.ctx.client_game_state[self.client.team, self.client.slot] == CLIENT_GOAL: | 
					
						
							|  |  |  |                 remaining_item_ids = get_remaining(self.ctx, self.client.team, self.client.slot) | 
					
						
							|  |  |  |                 if remaining_item_ids: | 
					
						
							|  |  |  |                     self.output("Remaining items: " + ", ".join(Items.lookup_id_to_name.get(item_id, "unknown item") | 
					
						
							|  |  |  |                                                                 for item_id in remaining_item_ids)) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     self.output("No remaining items found.") | 
					
						
							|  |  |  |                 return True | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.output( | 
					
						
							|  |  |  |                     "Sorry, !remaining requires you to have beaten the game on this server") | 
					
						
							|  |  |  |                 return False | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_countdown(self, seconds: str = "10") -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """Start a countdown in seconds""" | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2020-04-21 21:46:16 +02:00
										 |  |  |             timer = int(seconds, 10) | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         except ValueError: | 
					
						
							|  |  |  |             timer = 10 | 
					
						
							|  |  |  |         asyncio.create_task(countdown(self.ctx, timer)) | 
					
						
							| 
									
										
										
										
											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 the server's perspective""" | 
					
						
							|  |  |  |         buffer = ""  # try not to spam small packets over network | 
					
						
							|  |  |  |         count = 0 | 
					
						
							|  |  |  |         for location_id, location_name in Regions.lookup_id_to_name.items():  # cheat console is -1, keep in mind | 
					
						
							|  |  |  |             if location_id != -1 and location_id not in self.ctx.location_checks[self.client.team, self.client.slot]: | 
					
						
							|  |  |  |                 buffer += f'Missing: {location_name}\n' | 
					
						
							|  |  |  |                 count += 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if buffer: | 
					
						
							|  |  |  |             self.output(buffer + 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-20 11:47:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |     @mark_raw | 
					
						
							|  |  |  |     def _cmd_alias(self, alias_name: str = ""): | 
					
						
							|  |  |  |         if alias_name: | 
					
						
							|  |  |  |             alias_name = alias_name[:15] | 
					
						
							|  |  |  |             self.ctx.name_aliases[self.client.team, self.client.slot] = alias_name | 
					
						
							|  |  |  |             self.output(f"Hello, {alias_name}") | 
					
						
							|  |  |  |             update_aliases(self.ctx, self.client.team) | 
					
						
							|  |  |  |             save(self.ctx) | 
					
						
							|  |  |  |             return True | 
					
						
							|  |  |  |         elif (self.client.team, self.client.slot) in self.ctx.name_aliases: | 
					
						
							|  |  |  |             del (self.ctx.name_aliases[self.client.team, self.client.slot]) | 
					
						
							|  |  |  |             self.output("Removed Alias") | 
					
						
							|  |  |  |             update_aliases(self.ctx, self.client.team) | 
					
						
							|  |  |  |             save(self.ctx) | 
					
						
							|  |  |  |             return True | 
					
						
							|  |  |  |         return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 03:24:27 +02:00
										 |  |  |     @mark_raw | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_getitem(self, item_name: str) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """Cheat in an item""" | 
					
						
							|  |  |  |         if self.ctx.item_cheat: | 
					
						
							|  |  |  |             item_name, usable, response = get_intended_text(item_name, Items.item_table.keys()) | 
					
						
							|  |  |  |             if usable: | 
					
						
							|  |  |  |                 new_item = ReceivedItem(Items.item_table[item_name][3], -1, self.client.slot) | 
					
						
							|  |  |  |                 get_received_items(self.ctx, self.client.team, self.client.slot).append(new_item) | 
					
						
							|  |  |  |                 notify_all(self.ctx, 'Cheat console: sending "' + item_name + '" to ' + self.client.name) | 
					
						
							|  |  |  |                 send_new_items(self.ctx) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                 return True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 self.output(response) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                 return False | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         else: | 
					
						
							|  |  |  |             self.output("Cheating is disabled.") | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 03:24:27 +02:00
										 |  |  |     @mark_raw | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_hint(self, item_or_location: str = "") -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         """Use !hint {item_name/location_name}, for example !hint Lamp or !hint Link's House. """ | 
					
						
							|  |  |  |         points_available = self.ctx.location_check_points * len( | 
					
						
							|  |  |  |             self.ctx.location_checks[self.client.team, self.client.slot]) - \ | 
					
						
							|  |  |  |                            self.ctx.hint_cost * self.ctx.hints_used[self.client.team, self.client.slot] | 
					
						
							|  |  |  |         if not item_or_location: | 
					
						
							|  |  |  |             self.output(f"A hint costs {self.ctx.hint_cost} points. " | 
					
						
							|  |  |  |                         f"You have {points_available} points.") | 
					
						
							| 
									
										
										
										
											2020-04-22 05:09:46 +02:00
										 |  |  |             hints = {hint.re_check(self.ctx, self.client.team) for hint in | 
					
						
							|  |  |  |                      self.ctx.hints[self.client.team, self.client.slot]} | 
					
						
							|  |  |  |             self.ctx.hints[self.client.team, self.client.slot] = hints | 
					
						
							|  |  |  |             notify_hints(self.ctx, self.client.team, list(hints)) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |             return True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         else: | 
					
						
							|  |  |  |             item_name, usable, response = get_intended_text(item_or_location) | 
					
						
							|  |  |  |             if usable: | 
					
						
							|  |  |  |                 if item_name in Items.hint_blacklist: | 
					
						
							|  |  |  |                     self.output(f"Sorry, \"{item_name}\" is marked as non-hintable.") | 
					
						
							|  |  |  |                     hints = [] | 
					
						
							|  |  |  |                 elif item_name in Items.item_table:  # item name | 
					
						
							|  |  |  |                     hints = collect_hints(self.ctx, self.client.team, self.client.slot, item_name) | 
					
						
							|  |  |  |                 else:  # location name | 
					
						
							|  |  |  |                     hints = collect_hints_location(self.ctx, self.client.team, self.client.slot, item_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if hints: | 
					
						
							| 
									
										
										
										
											2020-04-22 15:50:14 +02:00
										 |  |  |                     new_hints = set(hints) - self.ctx.hints[self.client.team, self.client.slot] | 
					
						
							|  |  |  |                     old_hints = set(hints) - new_hints | 
					
						
							|  |  |  |                     if old_hints: | 
					
						
							|  |  |  |                         notify_hints(self.ctx, self.client.team, list(old_hints)) | 
					
						
							|  |  |  |                         if not new_hints: | 
					
						
							|  |  |  |                             self.output("Hint was previously used, no points deducted.") | 
					
						
							|  |  |  |                     if new_hints: | 
					
						
							|  |  |  |                         found_hints = sum(not hint.found for hint in new_hints) | 
					
						
							|  |  |  |                         if not found_hints:  # everything's been found, no need to pay | 
					
						
							|  |  |  |                             can_pay = True | 
					
						
							|  |  |  |                         elif self.ctx.hint_cost: | 
					
						
							|  |  |  |                             can_pay = points_available // (self.ctx.hint_cost * found_hints) >= 1 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |                         else: | 
					
						
							| 
									
										
										
										
											2020-04-22 15:50:14 +02:00
										 |  |  |                             can_pay = True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-22 15:50:14 +02:00
										 |  |  |                         if can_pay: | 
					
						
							|  |  |  |                             self.ctx.hints_used[self.client.team, self.client.slot] += found_hints | 
					
						
							| 
									
										
										
										
											2020-04-22 05:09:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-22 15:50:14 +02:00
										 |  |  |                             for hint in new_hints: | 
					
						
							|  |  |  |                                 if not hint.found: | 
					
						
							| 
									
										
										
										
											2020-04-22 05:09:46 +02:00
										 |  |  |                                     self.ctx.hints[self.client.team, hint.finding_player].add(hint) | 
					
						
							|  |  |  |                                     self.ctx.hints[self.client.team, hint.receiving_player].add(hint) | 
					
						
							| 
									
										
										
										
											2020-04-22 15:50:14 +02:00
										 |  |  |                             notify_hints(self.ctx, self.client.team, list(new_hints)) | 
					
						
							|  |  |  |                             save(self.ctx) | 
					
						
							|  |  |  |                         else: | 
					
						
							|  |  |  |                             notify_client(self.client, f"You can't afford the hint. " | 
					
						
							|  |  |  |                                                        f"You have {points_available} points and need at least " | 
					
						
							|  |  |  |                                                        f"{self.ctx.hint_cost}, " | 
					
						
							|  |  |  |                                                        f"more if multiple items are still to be found.") | 
					
						
							|  |  |  |                         return True | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |                 else: | 
					
						
							|  |  |  |                     self.output("Nothing found. Item/Location may not exist.") | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                     return False | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 self.output(response) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                 return False | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  | async def process_client_cmd(ctx: Context, client: Client, cmd, args): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     if type(cmd) is not str: | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |         await send_msgs(client, [['InvalidCmd']]) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if cmd == 'Connect': | 
					
						
							|  |  |  |         if not args or type(args) is not dict or \ | 
					
						
							|  |  |  |                 'password' not in args or type(args['password']) not in [str, type(None)] or \ | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |                 'rom' not in args or type(args['rom']) is not list: | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |             await send_msgs(client, [['InvalidArguments', 'Connect']]) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         errors = set() | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         if ctx.password is not None and args['password'] != ctx.password: | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             errors.add('InvalidPassword') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         if tuple(args['rom']) not in ctx.rom_names: | 
					
						
							|  |  |  |             errors.add('InvalidRom') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |             team, slot = ctx.rom_names[tuple(args['rom'])] | 
					
						
							|  |  |  |             if any([c.slot == slot and c.team == team for c in ctx.clients if c.auth]): | 
					
						
							|  |  |  |                 errors.add('SlotAlreadyTaken') | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 client.name = ctx.player_names[(team, slot)] | 
					
						
							|  |  |  |                 client.team = team | 
					
						
							|  |  |  |                 client.slot = slot | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if errors: | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |             await send_msgs(client, [['ConnectionRefused', list(errors)]]) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         else: | 
					
						
							|  |  |  |             client.auth = True | 
					
						
							| 
									
										
										
										
											2020-02-16 15:32:40 +01:00
										 |  |  |             client.version = args.get('version', Client.version) | 
					
						
							|  |  |  |             client.tags = args.get('tags', Client.tags) | 
					
						
							|  |  |  |             reply = [['Connected', [(client.team, client.slot), | 
					
						
							|  |  |  |                                     [(p, n) for (t, p), n in ctx.player_names.items() if t == client.team]]]] | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             items = get_received_items(ctx, client.team, client.slot) | 
					
						
							|  |  |  |             if items: | 
					
						
							|  |  |  |                 reply.append(['ReceivedItems', (0, tuplize_received_items(items))]) | 
					
						
							|  |  |  |                 client.send_index = len(items) | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |             await send_msgs(client, reply) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |             await on_client_joined(ctx, client) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |     if client.auth: | 
					
						
							|  |  |  |         if cmd == 'Sync': | 
					
						
							|  |  |  |             items = get_received_items(ctx, client.team, client.slot) | 
					
						
							|  |  |  |             if items: | 
					
						
							|  |  |  |                 client.send_index = len(items) | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |                 await send_msgs(client, [['ReceivedItems', (0, tuplize_received_items(items))]]) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         elif cmd == 'LocationChecks': | 
					
						
							|  |  |  |             if type(args) is not list: | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |                 await send_msgs(client, [['InvalidArguments', 'LocationChecks']]) | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |                 return | 
					
						
							|  |  |  |             register_location_checks(ctx, client.team, client.slot, args) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         elif cmd == 'LocationScouts': | 
					
						
							|  |  |  |             if type(args) is not list: | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |                 await send_msgs(client, [['InvalidArguments', 'LocationScouts']]) | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |                 return | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |             locs = [] | 
					
						
							|  |  |  |             for location in args: | 
					
						
							|  |  |  |                 if type(location) is not int or 0 >= location > len(Regions.location_table): | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |                     await send_msgs(client, [['InvalidArguments', 'LocationScouts']]) | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |                     return | 
					
						
							|  |  |  |                 loc_name = list(Regions.location_table.keys())[location - 1] | 
					
						
							|  |  |  |                 target_item, target_player = ctx.locations[(Regions.location_table[loc_name][0], client.slot)] | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |                 replacements = {'SmallKey': 0xA2, 'BigKey': 0x9D, 'Compass': 0x8D, 'Map': 0x7D} | 
					
						
							|  |  |  |                 item_type = [i[2] for i in Items.item_table.values() if type(i[3]) is int and i[3] == target_item] | 
					
						
							|  |  |  |                 if item_type: | 
					
						
							|  |  |  |                     target_item = replacements.get(item_type[0], target_item) | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |                 locs.append([loc_name, location, target_item, target_player]) | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |             # logging.info(f"{client.name} in team {client.team+1} scouted {', '.join([l[0] for l in locs])}") | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |             await send_msgs(client, [['LocationInfo', [l[1:] for l in locs]]]) | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         elif cmd == 'UpdateTags': | 
					
						
							|  |  |  |             if not args or type(args) is not list: | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |                 await send_msgs(client, [['InvalidArguments', 'UpdateTags']]) | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |                 return | 
					
						
							|  |  |  |             client.tags = args | 
					
						
							| 
									
										
										
										
											2020-03-23 02:47:07 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |         elif cmd == 'GameFinished': | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |             if (client.team, client.slot) not in ctx.client_game_state: | 
					
						
							| 
									
										
										
										
											2020-04-24 21:42:13 -07:00
										 |  |  |                 finished_msg = f'{client.name} (Team #{client.team + 1}) has found the triforce.' | 
					
						
							|  |  |  |                 notify_all(ctx, finished_msg) | 
					
						
							|  |  |  |                 print(finished_msg) | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |                 ctx.client_game_state[client.team, client.slot] = CLIENT_GOAL | 
					
						
							| 
									
										
										
										
											2020-04-24 20:07:28 -07:00
										 |  |  |             # TODO: Add auto-forfeit code here | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |         if cmd == 'Say': | 
					
						
							|  |  |  |             if type(args) is not str or not args.isprintable(): | 
					
						
							| 
									
										
										
										
											2020-04-19 14:05:58 +02:00
										 |  |  |                 await send_msgs(client, [['InvalidArguments', 'Say']]) | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |                 return | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |             notify_all(ctx, ctx.get_aliased_name(client.team, client.slot) + ': ' + args) | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |             print(args) | 
					
						
							|  |  |  |             client.messageprocessor(args) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 13:57:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-16 04:06:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  | def set_password(ctx: Context, password): | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     ctx.password = password | 
					
						
							| 
									
										
										
										
											2020-02-18 09:14:31 +01:00
										 |  |  |     logging.warning('Password set to ' + password if password else 'Password disabled') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  | class ServerCommandProcessor(CommandProcessor): | 
					
						
							|  |  |  |     ctx: Context | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, ctx: Context): | 
					
						
							|  |  |  |         self.ctx = ctx | 
					
						
							|  |  |  |         super(ServerCommandProcessor, self).__init__() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def default(self, raw: str): | 
					
						
							|  |  |  |         notify_all(self.ctx, '[Server]: ' + raw) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 03:24:27 +02:00
										 |  |  |     @mark_raw | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_kick(self, player_name: str) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |         """Kick specified player from the server""" | 
					
						
							|  |  |  |         for client in self.ctx.clients: | 
					
						
							|  |  |  |             if client.auth and client.name.lower() == player_name.lower() and client.socket and not client.socket.closed: | 
					
						
							|  |  |  |                 asyncio.create_task(client.socket.close()) | 
					
						
							|  |  |  |                 self.output(f"Kicked {client.name}") | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                 return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.output(f"Could not find player {player_name} to kick") | 
					
						
							|  |  |  |         return False | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |     def _cmd_save(self) -> bool: | 
					
						
							|  |  |  |         """Save current state to multidata""" | 
					
						
							|  |  |  |         save(self.ctx) | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |         self.output("Game saved") | 
					
						
							| 
									
										
										
										
											2020-04-23 06:16:54 +02:00
										 |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_players(self) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |         """Get information about connected players""" | 
					
						
							| 
									
										
										
										
											2020-04-19 14:51:48 +02:00
										 |  |  |         self.output(get_players_string(self.ctx)) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |         return True | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_exit(self) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |         """Shutdown the server""" | 
					
						
							|  |  |  |         asyncio.create_task(self.ctx.server.ws_server._close()) | 
					
						
							|  |  |  |         self.ctx.running = False | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |         return True | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 03:24:27 +02:00
										 |  |  |     @mark_raw | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_password(self, new_password: str = "") -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |         """Set the server password. Leave the password text empty to remove the password""" | 
					
						
							| 
									
										
										
										
											2020-04-19 03:24:27 +02:00
										 |  |  |         set_password(self.ctx, new_password if new_password else None) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |         return True | 
					
						
							| 
									
										
										
										
											2020-04-15 10:31:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 03:24:27 +02:00
										 |  |  |     @mark_raw | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_forfeit(self, player_name: str) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |         """Send out the remaining items from a player's game to their intended recipients""" | 
					
						
							|  |  |  |         seeked_player = player_name.lower() | 
					
						
							|  |  |  |         for (team, slot), name in self.ctx.player_names.items(): | 
					
						
							|  |  |  |             if name.lower() == seeked_player: | 
					
						
							|  |  |  |                 forfeit_player(self.ctx, team, slot) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                 return True | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |         self.output(f"Could not find player {player_name} to forfeit") | 
					
						
							|  |  |  |         return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _cmd_send(self, player_name: str, *item_name: str) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |         """Sends an item to the specified player""" | 
					
						
							|  |  |  |         seeked_player, usable, response = get_intended_text(player_name, self.ctx.player_names.values()) | 
					
						
							|  |  |  |         if usable: | 
					
						
							|  |  |  |             item = " ".join(item_name) | 
					
						
							|  |  |  |             item, usable, response = get_intended_text(item, Items.item_table.keys()) | 
					
						
							|  |  |  |             if usable: | 
					
						
							|  |  |  |                 for client in self.ctx.clients: | 
					
						
							|  |  |  |                     if client.name == seeked_player: | 
					
						
							|  |  |  |                         new_item = ReceivedItem(Items.item_table[item][3], -1, client.slot) | 
					
						
							|  |  |  |                         get_received_items(self.ctx, client.team, client.slot).append(new_item) | 
					
						
							|  |  |  |                         notify_all(self.ctx, 'Cheat console: sending "' + item + '" to ' + client.name) | 
					
						
							|  |  |  |                         send_new_items(self.ctx) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                         return True | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 self.output(response) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                 return False | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |         else: | 
					
						
							|  |  |  |             self.output(response) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |     def _cmd_hint(self, player_name: str, *item_or_location: str) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |         """Send out a hint for a player's item or location to their team""" | 
					
						
							|  |  |  |         seeked_player, usable, response = get_intended_text(player_name, self.ctx.player_names.values()) | 
					
						
							|  |  |  |         if usable: | 
					
						
							|  |  |  |             for (team, slot), name in self.ctx.player_names.items(): | 
					
						
							|  |  |  |                 if name == seeked_player: | 
					
						
							|  |  |  |                     item = " ".join(item_or_location) | 
					
						
							|  |  |  |                     item, usable, response = get_intended_text(item) | 
					
						
							|  |  |  |                     if usable: | 
					
						
							|  |  |  |                         if item in Items.item_table:  # item name | 
					
						
							|  |  |  |                             hints = collect_hints(self.ctx, team, slot, item) | 
					
						
							|  |  |  |                             notify_hints(self.ctx, team, hints) | 
					
						
							|  |  |  |                         else:  # location name | 
					
						
							|  |  |  |                             hints = collect_hints_location(self.ctx, team, slot, item) | 
					
						
							|  |  |  |                             notify_hints(self.ctx, team, hints) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                         return True | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |                     else: | 
					
						
							|  |  |  |                         self.output(response) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |                         return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |         else: | 
					
						
							|  |  |  |             self.output(response) | 
					
						
							| 
									
										
										
										
											2020-04-21 06:26:51 +02:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 17:24:51 +01:00
										 |  |  | async def console(ctx: Context): | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  |     session = prompt_toolkit.PromptSession() | 
					
						
							| 
									
										
										
										
											2020-04-13 11:26:50 +02:00
										 |  |  |     while ctx.running: | 
					
						
							| 
									
										
										
										
											2020-03-13 03:53:20 +01:00
										 |  |  |         with patch_stdout(): | 
					
						
							|  |  |  |             input_text = await session.prompt_async() | 
					
						
							| 
									
										
										
										
											2020-01-15 00:34:12 +01:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2020-04-14 20:22:42 +02:00
										 |  |  |             ctx.commandprocessor(input_text) | 
					
						
							| 
									
										
										
										
											2020-01-15 00:34:12 +01:00
										 |  |  |         except: | 
					
						
							|  |  |  |             import traceback | 
					
						
							|  |  |  |             traceback.print_exc() | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | async def forward_port(port: int): | 
					
						
							| 
									
										
										
										
											2020-03-07 16:22:04 +01:00
										 |  |  |     import upnpy | 
					
						
							|  |  |  |     import socket | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     upnp = upnpy.UPnP() | 
					
						
							|  |  |  |     upnp.discover() | 
					
						
							|  |  |  |     device = upnp.get_igd() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     service = device['WANPPPConnection.1'] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  |     # get own lan IP | 
					
						
							| 
									
										
										
										
											2020-03-07 16:22:04 +01:00
										 |  |  |     ip = socket.gethostbyname(socket.gethostname()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # This specific action returns an empty dict: {} | 
					
						
							|  |  |  |     service.AddPortMapping( | 
					
						
							|  |  |  |         NewRemoteHost='', | 
					
						
							|  |  |  |         NewExternalPort=port, | 
					
						
							|  |  |  |         NewProtocol='TCP', | 
					
						
							|  |  |  |         NewInternalPort=port, | 
					
						
							|  |  |  |         NewInternalClient=ip, | 
					
						
							|  |  |  |         NewEnabled=1, | 
					
						
							| 
									
										
										
										
											2020-03-07 17:38:49 +01:00
										 |  |  |         NewPortMappingDescription='Berserker\'s Multiworld', | 
					
						
							|  |  |  |         NewLeaseDuration=60 * 60 * 24  # 24 hours | 
					
						
							| 
									
										
										
										
											2020-03-07 16:22:04 +01:00
										 |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     logging.info(f"Attempted to forward port {port} to {ip}, your local ip address.") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 18:09:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-22 21:23:39 +01:00
										 |  |  | def parse_args() -> argparse.Namespace: | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     parser = argparse.ArgumentParser() | 
					
						
							| 
									
										
										
										
											2020-04-02 11:21:33 +02:00
										 |  |  |     defaults = Utils.get_options()["server_options"] | 
					
						
							|  |  |  |     parser.add_argument('--host', default=defaults["host"]) | 
					
						
							|  |  |  |     parser.add_argument('--port', default=defaults["port"], type=int) | 
					
						
							|  |  |  |     parser.add_argument('--password', default=defaults["password"]) | 
					
						
							|  |  |  |     parser.add_argument('--multidata', default=defaults["multidata"]) | 
					
						
							|  |  |  |     parser.add_argument('--savefile', default=defaults["savefile"]) | 
					
						
							|  |  |  |     parser.add_argument('--disable_save', default=defaults["disable_save"], action='store_true') | 
					
						
							|  |  |  |     parser.add_argument('--loglevel', default=defaults["loglevel"], | 
					
						
							|  |  |  |                         choices=['debug', 'info', 'warning', 'error', 'critical']) | 
					
						
							|  |  |  |     parser.add_argument('--location_check_points', default=defaults["location_check_points"], type=int) | 
					
						
							|  |  |  |     parser.add_argument('--hint_cost', default=defaults["hint_cost"], type=int) | 
					
						
							|  |  |  |     parser.add_argument('--disable_item_cheat', default=defaults["disable_item_cheat"], action='store_true') | 
					
						
							|  |  |  |     parser.add_argument('--port_forward', default=defaults["port_forward"], action='store_true') | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |     parser.add_argument('--forfeit_mode', default=defaults["forfeit_mode"], nargs='?', | 
					
						
							|  |  |  |                         choices=['auto', 'enabled', 'disabled', "goal"], help='''\
 | 
					
						
							|  |  |  |                              Select !forfeit Accessibility. (default: %(default)s) | 
					
						
							|  |  |  |                              auto:     Automatic "forfeit" on goal completion | 
					
						
							|  |  |  |                              enabled:  !forfeit is always available | 
					
						
							|  |  |  |                              disabled: !forfeit is never available | 
					
						
							|  |  |  |                              goal:     !forfeit can be used after goal completion | 
					
						
							|  |  |  |                              ''')
 | 
					
						
							|  |  |  |     parser.add_argument('--remaining_mode', default=defaults["remaining_mode"], nargs='?', | 
					
						
							|  |  |  |                         choices=['enabled', 'disabled', "goal"], help='''\
 | 
					
						
							|  |  |  |                              Select !remaining Accessibility. (default: %(default)s) | 
					
						
							|  |  |  |                              enabled:  !remaining is always available | 
					
						
							|  |  |  |                              disabled: !remaining is never available | 
					
						
							|  |  |  |                              goal:     !remaining can be used after goal completion | 
					
						
							|  |  |  |                              ''')
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     args = parser.parse_args() | 
					
						
							| 
									
										
										
										
											2020-03-22 21:23:39 +01:00
										 |  |  |     return args | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async def main(args: argparse.Namespace): | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |     logging.basicConfig(format='[%(asctime)s] %(message)s', level=getattr(logging, args.loglevel.upper(), logging.INFO)) | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  |     portforwardtask = None | 
					
						
							| 
									
										
										
										
											2020-03-11 09:16:07 +01:00
										 |  |  |     if args.port_forward: | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  |         portforwardtask = asyncio.create_task(forward_port(args.port)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-22 19:42:44 +01:00
										 |  |  |     ctx = Context(args.host, args.port, args.password, args.location_check_points, args.hint_cost, | 
					
						
							| 
									
										
										
										
											2020-04-25 15:11:58 +02:00
										 |  |  |                   not args.disable_item_cheat, args.forfeit_mode, args.remaining_mode) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ctx.data_filename = args.multidata | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         if not ctx.data_filename: | 
					
						
							|  |  |  |             import tkinter | 
					
						
							|  |  |  |             import tkinter.filedialog | 
					
						
							|  |  |  |             root = tkinter.Tk() | 
					
						
							|  |  |  |             root.withdraw() | 
					
						
							|  |  |  |             ctx.data_filename = tkinter.filedialog.askopenfilename(filetypes=(("Multiworld data","*multidata"),)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with open(ctx.data_filename, 'rb') as f: | 
					
						
							| 
									
										
										
										
											2020-01-04 22:08:13 +01:00
										 |  |  |             jsonobj = json.loads(zlib.decompress(f.read()).decode("utf-8")) | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |             for team, names in enumerate(jsonobj['names']): | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |                 for player, name in enumerate(names, 1): | 
					
						
							|  |  |  |                     ctx.player_names[(team, player)] = name | 
					
						
							| 
									
										
										
										
											2020-01-18 09:50:12 +01:00
										 |  |  |             ctx.rom_names = {tuple(rom): (team, slot) for slot, team, rom in jsonobj['roms']} | 
					
						
							|  |  |  |             ctx.remote_items = set(jsonobj['remote_items']) | 
					
						
							|  |  |  |             ctx.locations = {tuple(k): tuple(v) for k, v in jsonobj['locations']} | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     except Exception as e: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |         logging.error('Failed to read multiworld data (%s)' % e) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         return | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-02 10:40:38 -07:00
										 |  |  |     ip = args.host if args.host else Utils.get_public_ipv4() | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ctx.disable_save = args.disable_save | 
					
						
							|  |  |  |     if not ctx.disable_save: | 
					
						
							|  |  |  |         if not ctx.save_filename: | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  |             ctx.save_filename = (ctx.data_filename[:-9] if ctx.data_filename[-9:] == 'multidata' else ( | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  |                     ctx.data_filename + '_')) + 'multisave' | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         try: | 
					
						
							|  |  |  |             with open(ctx.save_filename, 'rb') as f: | 
					
						
							| 
									
										
										
										
											2020-01-04 22:08:13 +01:00
										 |  |  |                 jsonobj = json.loads(zlib.decompress(f.read()).decode("utf-8")) | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  |                 ctx.set_save(jsonobj) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         except FileNotFoundError: | 
					
						
							| 
									
										
										
										
											2020-01-18 12:21:57 +01:00
										 |  |  |             logging.error('No save data found, starting a new game') | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |         except Exception as e: | 
					
						
							| 
									
										
										
										
											2020-02-09 05:28:48 +01:00
										 |  |  |             logging.exception(e) | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  |     if portforwardtask: | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             await portforwardtask | 
					
						
							|  |  |  |         except: | 
					
						
							|  |  |  |             logging.exception("Automatic port forwarding failed with:") | 
					
						
							|  |  |  |     ctx.server = websockets.serve(functools.partial(server, ctx=ctx), ctx.host, ctx.port, ping_timeout=None, | 
					
						
							|  |  |  |                                   ping_interval=None) | 
					
						
							|  |  |  |     logging.info('Hosting game at %s:%d (%s)' % (ip, ctx.port, | 
					
						
							|  |  |  |                                                  'No password' if not ctx.password else 'Password: %s' % ctx.password)) | 
					
						
							| 
									
										
										
										
											2019-12-09 19:27:56 +01:00
										 |  |  |     await ctx.server | 
					
						
							|  |  |  |     await console(ctx) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     loop = asyncio.get_event_loop() | 
					
						
							| 
									
										
										
										
											2020-03-22 21:23:39 +01:00
										 |  |  |     loop.run_until_complete(main(parse_args())) |