Geek7 commited on
Commit
db98c95
·
verified ·
1 Parent(s): 3161a5d

Upload server.py

Browse files
Files changed (1) hide show
  1. server.py +199 -0
server.py ADDED
@@ -0,0 +1,199 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import random
2
+ import json
3
+ import socketio
4
+ from flask import Flask, jsonify
5
+ import eventlet
6
+ import threading
7
+ from nats.aio.client import Client as NATS
8
+ import asyncpg
9
+ import asyncio
10
+
11
+ eventlet.monkey_patch()
12
+
13
+ sio = socketio.Server(cors_allowed_origins='*')
14
+ app = Flask(__name__)
15
+ app.wsgi_app = socketio.WSGIApp(sio, app.wsgi_app)
16
+
17
+ DB_DSN = "postgresql://postgres:postgres@pgbouncer:6432/postgres"
18
+ NATS_URL = "nats://nats:4222"
19
+
20
+ WAITING_PLAYERS = []
21
+ PLAYER_DATA = {}
22
+ GAME_STATES = {}
23
+
24
+ nats_client = NATS()
25
+ loop = asyncio.new_event_loop()
26
+ asyncio.set_event_loop(loop)
27
+
28
+ async def init_db():
29
+ return await asyncpg.create_pool(dsn=DB_DSN)
30
+
31
+ db_pool = loop.run_until_complete(init_db())
32
+
33
+ async def publish_event(event_type, data):
34
+ message = json.dumps({'type': event_type, 'data': data})
35
+ await nats_client.publish("checkers", message.encode())
36
+
37
+ async def subscribe():
38
+ async def message_handler(msg):
39
+ event = json.loads(msg.data.decode())
40
+ if event['type'] == 'opponent_move':
41
+ sio.emit('opponent_move', event['data']['move'], room=event['data']['room'], skip_sid=event['data']['sid'])
42
+ elif event['type'] == 'game_over':
43
+ sio.emit('game_over', event['data'], room=event['data']['room'])
44
+ elif event['type'] == 'opponent_disconnected':
45
+ sio.emit('opponent_disconnected', room=event['data']['room'])
46
+
47
+ await nats_client.connect(servers=[NATS_URL], loop=loop)
48
+ await nats_client.subscribe("checkers", cb=message_handler)
49
+
50
+ def init_board():
51
+ board = [[None for _ in range(8)] for _ in range(8)]
52
+ for row in range(3):
53
+ for col in range(8):
54
+ if (row + col) % 2 == 1:
55
+ board[row][col] = 'b'
56
+ for row in range(5, 8):
57
+ for col in range(8):
58
+ if (row + col) % 2 == 1:
59
+ board[row][col] = 'r'
60
+ return board
61
+
62
+ def get_valid_moves(board, row, col, color, is_king):
63
+ directions = [(-1, -1), (-1, 1)] if color == 'red' else [(1, -1), (1, 1)]
64
+ if is_king:
65
+ directions += [(-d[0], -d[1]) for d in directions]
66
+
67
+ moves, captures = [], []
68
+ for dr, dc in directions:
69
+ r, c = row + dr, col + dc
70
+ if 0 <= r < 8 and 0 <= c < 8 and board[r][c] is None:
71
+ moves.append((r, c))
72
+ elif 0 <= r < 8 and 0 <= c < 8:
73
+ mid_piece = board[r][c]
74
+ if mid_piece and mid_piece[0].lower() != color[0]:
75
+ r2, c2 = r + dr, c + dc
76
+ if 0 <= r2 < 8 and 0 <= c2 < 8 and board[r2][c2] is None:
77
+ captures.append(((r2, c2), (r, c)))
78
+ return moves, captures
79
+
80
+ @app.route('/')
81
+ def index():
82
+ return "Checkers server running."
83
+
84
+ @app.route('/state')
85
+ def game_state():
86
+ for sid, pdata in PLAYER_DATA.items():
87
+ if 'room' in pdata:
88
+ room = pdata['room']
89
+ if room in GAME_STATES:
90
+ return jsonify({'board': GAME_STATES[room]['board']})
91
+ return jsonify({'board': [[None]*8 for _ in range(8)]})
92
+
93
+ @sio.event
94
+ def connect(sid, environ):
95
+ print(f"{sid} connected")
96
+
97
+ @sio.event
98
+ def set_name(sid, data):
99
+ name = data.get('name', 'Player')
100
+ PLAYER_DATA[sid] = {'name': name}
101
+
102
+ if WAITING_PLAYERS:
103
+ opponent_sid = WAITING_PLAYERS.pop(0)
104
+ room = f"room_{sid[:4]}_{opponent_sid[:4]}"
105
+ for psid in [sid, opponent_sid]:
106
+ pdata = PLAYER_DATA[psid]
107
+ pdata['room'] = room
108
+ sio.enter_room(psid, room)
109
+
110
+ players = [(sid, 'red'), (opponent_sid, 'black')]
111
+ random.shuffle(players)
112
+ for psid, color in players:
113
+ PLAYER_DATA[psid]['color'] = color
114
+ opp_sid = opponent_sid if psid == sid else sid
115
+ opp_name = PLAYER_DATA[opp_sid]['name']
116
+ sio.emit("match_found", {'color': color, 'opponent': opp_name}, to=psid)
117
+
118
+ GAME_STATES[room] = {'board': init_board(), 'turn': 'red'}
119
+ else:
120
+ WAITING_PLAYERS.append(sid)
121
+
122
+ @sio.event
123
+ def move(sid, data):
124
+ pdata = PLAYER_DATA.get(sid)
125
+ if not pdata:
126
+ return
127
+ room = pdata.get('room')
128
+ state = GAME_STATES.get(room)
129
+ if not state:
130
+ return
131
+ board = state['board']
132
+ turn = state['turn']
133
+ color = pdata['color']
134
+
135
+ if color != turn:
136
+ return
137
+
138
+ from_row, from_col = data['from']
139
+ to_row, to_col = data['to']
140
+ piece = board[from_row][from_col]
141
+ if not piece or piece[0].lower() != color[0]:
142
+ return
143
+
144
+ is_king = piece.isupper()
145
+ _, captures = get_valid_moves(board, from_row, from_col, color, is_king)
146
+ move_is_capture = abs(to_row - from_row) == 2
147
+ if captures and not move_is_capture:
148
+ return
149
+
150
+ board[to_row][to_col] = piece
151
+ board[from_row][from_col] = None
152
+
153
+ if move_is_capture:
154
+ cap_row = (from_row + to_row) // 2
155
+ cap_col = (from_col + to_col) // 2
156
+ board[cap_row][cap_col] = None
157
+ _, new_captures = get_valid_moves(board, to_row, to_col, color, is_king or to_row in (0, 7))
158
+ if new_captures:
159
+ state['board'] = board
160
+ GAME_STATES[room] = state
161
+ sio.emit('multi_jump', {'from': [to_row, to_col]}, to=sid)
162
+ return
163
+
164
+ if (color == 'red' and to_row == 0) or (color == 'black' and to_row == 7):
165
+ board[to_row][to_col] = piece.upper()
166
+
167
+ flat = sum(board, [])
168
+ if all(p not in flat for p in ['r', 'R']):
169
+ loop.create_task(publish_event('game_over', {'winner': 'black', 'room': room}))
170
+ del GAME_STATES[room]
171
+ return
172
+ if all(p not in flat for p in ['b', 'B']):
173
+ loop.create_task(publish_event('game_over', {'winner': 'red', 'room': room}))
174
+ del GAME_STATES[room]
175
+ return
176
+
177
+ state['board'] = board
178
+ state['turn'] = 'black' if turn == 'red' else 'red'
179
+ GAME_STATES[room] = state
180
+ loop.create_task(publish_event('opponent_move', {'move': data, 'room': room, 'sid': sid}))
181
+
182
+ @sio.event
183
+ def disconnect(sid):
184
+ print(f"{sid} disconnected")
185
+ if sid in WAITING_PLAYERS:
186
+ WAITING_PLAYERS.remove(sid)
187
+ pdata = PLAYER_DATA.pop(sid, None)
188
+ if pdata and 'room' in pdata:
189
+ room = pdata['room']
190
+ for psid in list(PLAYER_DATA.keys()):
191
+ if PLAYER_DATA[psid].get('room') == room:
192
+ del PLAYER_DATA[psid]
193
+ if room in GAME_STATES:
194
+ del GAME_STATES[room]
195
+ loop.create_task(publish_event('opponent_disconnected', {'room': room}))
196
+
197
+ if __name__ == '__main__':
198
+ threading.Thread(target=lambda: loop.run_until_complete(subscribe()), daemon=True).start()
199
+ eventlet.wsgi.server(eventlet.listen(('0.0.0.0', 5000)), app)