forked from fogleman/Minecraft
-
Notifications
You must be signed in to change notification settings - Fork 33
/
client.py
196 lines (180 loc) · 9.33 KB
/
client.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
# Python packages
from _socket import SHUT_RDWR
import socket
from threading import Thread, Event, Lock
import struct
from warnings import warn
# Third-party packages
# Modules from this project
import pyglet
from blocks import BlockID
import globals as G
from globals import BLOCKS_DIR, SECTOR_SIZE
from items import ItemStack
from player import Player
from savingsystem import null2, structuchar2, sector_to_blockpos
from utils import extract_string_packet
from biome import BiomeGenerator
class PacketReceiver(Thread):
def __init__ (self, world, controller, sock):
Thread.__init__(self)
self.world = world
self.controller = controller
self._stop = Event()
self.lock = Lock()
self.sock = sock
def run(self):
try:
self.loop()
except socket.error as e:
if e[0] in (10053, 10054):
#TODO: GUI tell the client they were disconnected
print("Disconnected from server.")
else:
raise e
self.controller.back_to_main_menu.set()
# loop() is run once in its own thread
def loop(self):
packetcache, packetsize = b"", 0
append_to_sector_packets = self.world.sector_packets.append
while 1:
resp = self.sock.recv(16384)
if self._stop.isSet() or not resp:
print("Client PacketReceiver:",self._stop.isSet() and "Shutting down" or "We've been disconnected by the server")
self.sock.shutdown(SHUT_RDWR)
return
# Sometimes data is sent in multiple pieces, just because sock.recv returned data doesn't mean its complete
# So we write the datalength in the beginning of all messages, and don't look at the packet until we have enough data
packetcache += resp
if not packetsize and len(packetcache) >= 4:
packetsize = struct.unpack("i", packetcache[:4])[0]
# Sometimes we get multiple packets in a single burst, so loop through packetcache until we lack the data
while packetsize and len(packetcache) >= packetsize:
#Once we've obtained the whole packet
packetid = struct.unpack_from("B",packetcache, 4)[0] # Server Packet Type
packet = packetcache[5:packetsize]
with self.lock:
append_to_sector_packets((packetid, packet))
packetcache = packetcache[packetsize:] # Cut off the part we just read
packetsize = struct.unpack("i", packetcache[:4])[0] if len(packetcache) >= 4 else 0 # Get the next packet's size
#=== The following functions are run by the Main Thread ===#
# Process a packet off the stack that was received by loop()
def dequeue_packet(self):
with self.lock:
packetid, packet = self.world.sector_packets.popleft()
if packetid == 1: # Entire Sector
blocks, sectors = self.world, self.world.sectors
secpos = struct.unpack("iii", packet[:12])
sector = sectors[secpos]
cx, cy, cz = sector_to_blockpos(secpos)
fpos = 12
exposed_pos = fpos + 1024
for x in range(cx, cx+8):
for y in range(cy, cy+8):
for z in range(cz, cz+8):
read = packet[fpos:fpos+2]
fpos += 2
unpacked = structuchar2.unpack(read)
if read != null2 and unpacked in BLOCKS_DIR:
position = x,y,z
try:
blocks[position] = BLOCKS_DIR[unpacked]
if blocks[position].sub_id_as_metadata:
blocks[position] = type(BLOCKS_DIR[unpacked])()
blocks[position].set_metadata(0)
except KeyError:
main_blk = BLOCKS_DIR[(unpacked[0], 0)]
if main_blk.sub_id_as_metadata: # sub id is metadata
blocks[position] = type(main_blk)()
blocks[position].set_metadata(unpacked[-1])
sector.append(position)
if packet[exposed_pos:exposed_pos+1] == b"1":
blocks.show_block(position)
exposed_pos += 1
if secpos in self.world.sector_queue:
del self.world.sector_queue[secpos] #Delete any hide sector orders
elif packetid == 2: # Blank Sector
self.world.sectors[struct.unpack("iii", packet)] = []
elif packetid == 3: # Add Block
self.world._add_block(struct.unpack("iii", packet[:12]),
BLOCKS_DIR[struct.unpack("BB", packet[12:])])
elif packetid == 4: # Remove Block
self.world._remove_block(struct.unpack("iii", packet))
elif packetid == 5: # Chat Print
self.controller.write_line(packet[:-4].decode('utf-8'), color=struct.unpack("BBBB", packet[-4:]))
if not self.controller.text_input.visible:
self.controller.chat_box.visible = True
pyglet.clock.unschedule(self.controller.hide_chat_box)
pyglet.clock.schedule_once(self.controller.hide_chat_box, G.CHAT_FADE_TIME)
elif packetid == 6: # Inventory
player = self.controller.player
caret = 0
for inventory in (player.quick_slots.slots, player.inventory.slots, player.armor.slots):
for i in range(len(inventory)):
id_main, id_sub, amount = struct.unpack("HBB", packet[caret:caret+4])
caret += 4
if id_main == 0: continue
durability = -1
if id_main >= G.ITEM_ID_MIN and (id_main, id_sub) not in G.ITEMS_DIR:
#The subid must be durability
durability = id_sub * G.ITEMS_DIR[(id_main, 0)].durability // 255
id_sub = 0
inventory[i] = ItemStack(type=BlockID(id_main, id_sub), amount=amount, durability=durability)
self.controller.item_list.update_items()
self.controller.inventory_list.update_items()
elif packetid == 7: # New player connected
plyid, name = struct.unpack("H", packet[:2])[0], packet[2:].decode('utf-8')
if plyid not in self.controller.player_ids:
self.controller.player_ids[plyid] = Player(username=name, local_player=False)
elif name == '\0':
del self.controller.player_ids[plyid]
elif packetid == 8: # Player Movement
ply = self.controller.player_ids[struct.unpack("H", packet[:2])[0]]
ply.momentum = struct.unpack("fff", packet[2:14])
ply.position = struct.unpack("ddd", packet[14:])
elif packetid == 9: # Player Jump
self.controller.player_ids[struct.unpack("H", packet)[0]].dy = 0.016
elif packetid == 10: # Update Tile Entity
self.world[struct.unpack("iii", packet[:12])].update_tile_entity(packet[12:])
elif packetid == 255: # Spawn Position
self.controller.player.position = struct.unpack("iii", packet[:12])
packet = packet[12:]
packet, seed = extract_string_packet(packet)
self.world.biome_generator = BiomeGenerator(seed)
#Now that we know where the player should be, we can enable .update again
self.controller.update = self.controller.update_disabled
else:
warn("Received unknown packetid %s, there's probably a version mismatch between client and server!" % packetid)
# Helper Functions for sending data to the Server
def request_sector(self, sector):
self.sock.sendall(b"\1"+struct.pack("iii", *sector))
def add_block(self, position, block):
self.sock.sendall(b"\3"+struct.pack("iiiBB", *(position+(block.id.main, block.id.sub))))
def remove_block(self, position):
self.sock.sendall(b"\4"+struct.pack("iii", *position))
def send_chat(self, msg):
msg = msg.encode('utf-8')
self.sock.sendall(b"\5"+struct.pack("i", len(msg))+msg)
def request_spawnpos(self):
name = G.USERNAME.encode('utf-8')
self.sock.sendall(struct.pack("B", 255)+struct.pack("i",len(name)) + name)
def send_player_inventory(self):
packet = b""
for item in (self.controller.player.quick_slots.slots + self.controller.player.inventory.slots + self.controller.player.armor.slots):
if item:
packet += struct.pack("HBB", item.type.main, item.type.sub if item.max_durability == -1 else item.durability * 255 // item.max_durability, item.amount)
else:
packet += b"\0\0\0\0"
self.sock.sendall(b"\6"+packet)
def send_movement(self, momentum, position):
self.sock.sendall(b"\x08"+struct.pack("fff", *momentum) + struct.pack("ddd", *position))
def send_jump(self):
self.sock.sendall(b"\x09")
def update_tile_entity(self, position, value):
self.sock.sendall(b"\x0A" + struct.pack("iii", *position) + struct.pack("i", len(value)) + value)
def stop(self):
self._stop.set()
try:
self.sock.shutdown(SHUT_RDWR)
except:
pass