forked from CLSFramework/sample-playmaker-server-python-grpc
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathserver.py
229 lines (191 loc) · 10.7 KB
/
server.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
from concurrent import futures
from time import sleep
import service_pb2_grpc as pb2_grpc
import service_pb2 as pb2
from typing import Union
from multiprocessing import Manager, Lock
from utils.logger_utils import setup_logger
import logging
import grpc
import argparse
import datetime
from src.interfaces.IAgent import IAgent
from src.sample_coach_agent import SampleCoachAgent
from src.sample_player_agent import SamplePlayerAgent
from src.sample_trainer_agent import SampleTrainerAgent
import traceback
console_logging_level = logging.INFO
file_logging_level = logging.INFO
player_console_logging_level = logging.INFO
player_file_logging_level = logging.DEBUG
main_logger = None
log_dir = None
class GrpcAgent:
def __init__(self, agent_type, uniform_number, logger, debug) -> None:
self.agent_type: pb2.AgentType = agent_type
self.uniform_number: int = uniform_number
self.agent: IAgent = None
self.logger: logging.Logger = logger
if self.agent_type == pb2.AgentType.PlayerT:
self.agent = SamplePlayerAgent(self.logger)
elif self.agent_type == pb2.AgentType.CoachT:
self.agent = SampleCoachAgent(self.logger)
elif self.agent_type == pb2.AgentType.TrainerT:
self.agent = SampleTrainerAgent(self.logger)
self.agent.set_debug_mode(debug)
self.debug_mode: bool = False
def GetAction(self, state: pb2.State):
self.logger.debug(f"================================= cycle={state.world_model.cycle}.{state.world_model.stoped_cycle} =================================")
# self.logger.debug(f"State: {state}")
try:
if self.agent_type == pb2.AgentType.PlayerT:
return self.GetPlayerActions(state)
elif self.agent_type == pb2.AgentType.CoachT:
return self.GetCoachActions(state)
elif self.agent_type == pb2.AgentType.TrainerT:
return self.GetTrainerActions(state)
except Exception as e:
self.logger.error(f"Error in GetAction: {e}")
self.logger.error(traceback.format_exc())
return pb2.PlayerActions()
def GetPlayerActions(self, state: pb2.State):
self.agent.update_actions(state.world_model)
return self.agent.get_actions()
def GetBestPlannerAction(self, request: pb2.BestPlannerActionRequest) -> int:
self.logger.debug(f"GetBestPlannerAction cycle:{request.state.world_model.cycle} pairs:{len(request.pairs)} unum:{request.state.register_response.uniform_number}")
best_index = max(request.pairs.items(), key=lambda x: x[1].evaluation)[0]
best_action = request.pairs[best_index].action
while best_action.parent_index and best_action.parent_index > 0:
best_action = request.pairs[best_action.parent_index].action
res = pb2.BestPlannerActionResponse(index=best_action.index)
return res
def GetCoachActions(self, state: pb2.State):
self.agent.update_actions(state.world_model)
return self.agent.get_actions()
def GetTrainerActions(self, state: pb2.State):
self.agent.update_actions(state.world_model)
return self.agent.get_actions()
def SetServerParams(self, server_params: pb2.ServerParam):
try:
self.logger.debug(f"Server params received unum {server_params.register_response.uniform_number}")
self.agent.set_server_params(server_params)
except Exception as e:
self.logger.error(f"Error in GetAction: {e}")
self.logger.error(traceback.format_exc())
return pb2.PlayerActions()
def SetPlayerParams(self, player_params: pb2.PlayerParam):
try:
self.logger.debug(f"Player params received unum {player_params.register_response.uniform_number}")
self.agent.set_player_params(player_params)
except Exception as e:
self.logger.error(f"Error in GetAction: {e}")
self.logger.error(traceback.format_exc())
return pb2.PlayerActions()
def SetPlayerType(self, player_type: pb2.PlayerType):
try:
self.logger.debug(f"Player type received unum {player_type.register_response.uniform_number}")
self.agent.set_player_types(player_type)
except Exception as e:
self.logger.error(f"Error in GetAction: {e}")
self.logger.error(traceback.format_exc())
return pb2.PlayerActions()
class GameHandler(pb2_grpc.GameServicer):
def __init__(self, shared_lock, shared_number_of_connections, debug) -> None:
self.agents: dict[int, GrpcAgent] = {}
self.shared_lock = shared_lock
self.shared_number_of_connections = shared_number_of_connections
self.debug = debug
def GetPlayerActions(self, state: pb2.State, context):
main_logger.debug(f"GetPlayerActions unum {state.register_response.uniform_number} at {state.world_model.cycle}")
res = self.agents[state.register_response.client_id].GetAction(state)
return res
def GetCoachActions(self, state: pb2.State, context):
main_logger.debug(f"GetCoachActions coach at {state.world_model.cycle}")
res = self.agents[state.register_response.client_id].GetAction(state)
return res
def GetTrainerActions(self, state: pb2.State, context):
main_logger.debug(f"GetTrainerActions trainer at {state.world_model.cycle}")
res = self.agents[state.register_response.client_id].GetAction(state)
return res
def SendServerParams(self, serverParams: pb2.ServerParam, context):
main_logger.debug(f"Server params received unum {serverParams.register_response.uniform_number}")
self.agents[serverParams.register_response.client_id].SetServerParams(serverParams)
res = pb2.Empty()
return res
def SendPlayerParams(self, playerParams: pb2.PlayerParam, context):
main_logger.debug(f"Player params received unum {playerParams.register_response.uniform_number}")
self.agents[playerParams.register_response.client_id].SetPlayerParams(playerParams)
res = pb2.Empty()
return res
def SendPlayerType(self, playerType: pb2.PlayerType, context):
main_logger.debug(f"Player type received unum {playerType.register_response.uniform_number}")
self.agents[playerType.register_response.client_id].SetPlayerType(playerType)
res = pb2.Empty()
return res
def SendInitMessage(self, initMessage: pb2.InitMessage, context):
main_logger.debug(f"Init message received unum {initMessage.register_response.uniform_number}")
self.agents[initMessage.register_response.client_id].debug_mode = initMessage.debug_mode
res = pb2.Empty()
return res
def Register(self, register_request: pb2.RegisterRequest, context):
try:
with self.shared_lock:
main_logger.info(f"received register request from team_name: {register_request.team_name} "
f"unum: {register_request.uniform_number} "
f"agent_type: {register_request.agent_type}")
self.shared_number_of_connections.value += 1
main_logger.info(f"Number of connections {self.shared_number_of_connections.value}")
team_name = register_request.team_name
uniform_number = register_request.uniform_number
agent_type = register_request.agent_type
register_response = pb2.RegisterResponse(client_id=self.shared_number_of_connections.value,
team_name=team_name,
uniform_number=uniform_number,
agent_type=agent_type)
logger = setup_logger(f"agent{register_response.uniform_number}_{register_response.client_id}", log_dir,
console_level=player_console_logging_level, file_level=player_file_logging_level)
self.agents[self.shared_number_of_connections.value] = GrpcAgent(agent_type, uniform_number, logger, self.debug)
return register_response
except Exception as e:
main_logger.error(f"Error in Register: {e}")
main_logger.error(traceback.format_exc())
return pb2.RegisterResponse()
def SendByeCommand(self, register_response: pb2.RegisterResponse, context):
main_logger.debug(f"Bye command received unum {register_response.uniform_number}")
# with shared_lock:
self.agents.pop(register_response.client_id)
res = pb2.Empty()
return res
def GetBestPlannerAction(self, pairs: pb2.BestPlannerActionRequest, context):
main_logger.debug(f"GetBestPlannerAction cycle:{pairs.state.world_model.cycle} pairs:{len(pairs.pairs)} unum:{pairs.register_response.uniform_number}")
res = self.agents[pairs.register_response.client_id].GetBestPlannerAction(pairs)
return res
def serve(port, shared_lock, shared_number_of_connections, debug):
server = grpc.server(futures.ThreadPoolExecutor(max_workers=22))
game_service = GameHandler(shared_lock, shared_number_of_connections, debug)
pb2_grpc.add_GameServicer_to_server(game_service, server)
server.add_insecure_port(f'[::]:{port}')
server.start()
main_logger.info(f"Starting server on port {port}")
server.wait_for_termination()
def main():
global main_logger, log_dir, file_logging_level, player_file_logging_level
parser = argparse.ArgumentParser(description='Run play maker server')
parser.add_argument('-p', '--rpc-port', required=False, help='The port of the server', default=50051)
parser.add_argument('-l', '--log-dir', required=False, help='The directory of the log file',
default=f'logs/{datetime.datetime.now().strftime("%Y-%m-%d_%H-%M-%S")}')
parser.add_argument('--disable-log-file', required=False, help='Disable logging to a file', default=False, action='store_true')
parser.add_argument('-d', '--debug', required=False, help='Enable debug mode for agents', default=False, action='store_true')
args = parser.parse_args()
log_dir = args.log_dir
if args.disable_log_file:
file_logging_level = None
player_file_logging_level = None
main_logger = setup_logger("pmservice", log_dir, console_level=console_logging_level, file_level=file_logging_level)
main_logger.info("Starting server")
manager = Manager()
shared_lock = Lock() # Create a Lock for synchronization
shared_number_of_connections = manager.Value('i', 0)
serve(args.rpc_port, shared_lock, shared_number_of_connections, args.debug)
if __name__ == '__main__':
main()