hotel-madness/scripts/utils/networking.gd

239 lines
9.9 KiB
GDScript3
Raw Normal View History

extends Node
signal playerlist_changed()
## Connection Code
# Connect the signals.
func _ready() -> void:
if multiplayer.connected_to_server.connect(_on_connected_ok) != OK:
Log.error("FAILED to connect the connected_to_server signal to _on_connected_ok!", "Internal Error: Failed to connect signal.")
if multiplayer.connection_failed.connect(close_network) != OK:
Log.error("FAILED to connect the connection_failed signal to _on_connected_fail!", "Internal Error: Failed to connect signal.")
if multiplayer.server_disconnected.connect(close_network) != OK:
Log.error("FAILED to connect the server_disconnected signal to _on_server_disconnected!", "Internal Error: Failed to connect signal.")
2024-12-05 10:34:01 +01:00
if multiplayer.peer_disconnected.connect(_on_peer_disconnected) != OK:
Log.error("FAILED to connect the peer_disconnected signal to _on_peer_disconnected!", "Internal Error: Failed to connect signal.")
if playerlist_changed.connect(_on_playerlist_changed) != OK:
Log.error("FAILED to connect the playerlist_changed signal to _on_playerlist_changed!", "Internal Error: Failed to connect signal.")
2024-12-06 11:11:51 +01:00
func _on_playerlist_changed() -> void:
pass
# Start the network listener.
func start_server() -> Error:
var peer: ENetMultiplayerPeer = ENetMultiplayerPeer.new()
if peer.create_server(Game.port, Game.max_clients) != OK:
Log.warning("Couldn't create the server at port %d!" % Game.port)
return FAILED
multiplayer.multiplayer_peer = peer
Log.info("Created the server at port %d." % Game.port)
return OK
# Connect to a server.
func join_server() -> Error:
var peer: ENetMultiplayerPeer = ENetMultiplayerPeer.new()
if peer.create_client(Game.ip, Game.port) != OK:
Log.warning("Couldn't connect to the server at %s:%d!" % [Game.ip, Game.port])
return FAILED
multiplayer.multiplayer_peer = peer
Log.info("Connected to the server at %s:%d." % [Game.ip, Game.port])
return OK
# Close all network connections.
func close_network() -> void:
multiplayer.multiplayer_peer = null
Log.info("Closed all network connections.")
## Matchmaking Code
var mutex: Mutex = Mutex.new()
var rooms: Dictionary = {}
2024-12-06 11:11:51 +01:00
var server_peers: Dictionary = {}
var peers: Dictionary = {}
var isManager: bool = false
var managerID: int
func join_room() -> void:
if join_server() != OK:
Log.warning("Failed to connect to sever.")
func _on_connected_ok() -> void:
if rpc_id(1, "send_roomname", Game.roomname) != OK:
Log.warning("Failed to send roomname!")
2024-12-05 10:34:01 +01:00
func _on_peer_disconnected(id: int) -> void:
mutex.lock()
if multiplayer.get_unique_id() == 1:
if rooms.has(id):
var roomname: String = rooms[id]
if !rooms.erase(roomname):
Log.warning("The room %s associated with %d doesn't exist in rooms, even though it should!" % [roomname, id])
2024-12-06 11:11:51 +01:00
mutex.unlock()
return
2024-12-05 10:34:01 +01:00
if !rooms.erase(id):
Log.warning("The peer %d doesn't exist in rooms, even though it should!" % id)
2024-12-06 11:11:51 +01:00
mutex.unlock()
return
for peer: int in server_peers[roomname]:
2024-12-05 10:34:01 +01:00
if peer == id:
continue
if rpc_id(peer, "host_disconnected_message"):
Log.warning("Peer %d couldn't be disconnected!" % peer)
2024-12-06 11:11:51 +01:00
mutex.unlock()
return
if !server_peers.erase(roomname):
2024-12-05 10:34:01 +01:00
Log.warning("Room %s couldn't be found, even though it should exist!" % roomname)
2024-12-06 11:11:51 +01:00
mutex.unlock()
return
2024-12-05 10:34:01 +01:00
Log.info("Room %s terminated." % roomname)
else:
2024-12-06 11:11:51 +01:00
for room: String in server_peers:
for peer: int in server_peers[room]:
2024-12-05 10:34:01 +01:00
if peer == id:
2024-12-06 11:11:51 +01:00
var peer_array: Array = server_peers[room]
2024-12-05 10:34:01 +01:00
peer_array.erase(id)
2024-12-06 11:11:51 +01:00
server_peers[room] = peer_array
for found_peer: int in server_peers[room]:
if rpc_id(found_peer, "peer_disconnected_message", id) != OK:
2024-12-05 10:34:01 +01:00
Log.warning("Failed to send disconnect message to %d!" % found_peer)
2024-12-06 11:11:51 +01:00
mutex.unlock()
return
2024-12-05 10:34:01 +01:00
mutex.unlock()
# The following function sends the roomname selected by the client.
@rpc("any_peer", "call_remote", "reliable")
func send_roomname(roomname: String) -> void:
mutex.lock()
if multiplayer.get_unique_id() == 1:
if !rooms.has(roomname) and !rooms.has(multiplayer.get_remote_sender_id()):
rooms[roomname] = multiplayer.get_remote_sender_id()
rooms[multiplayer.get_remote_sender_id()] = roomname
2024-12-06 11:11:51 +01:00
server_peers[roomname] = [multiplayer.get_remote_sender_id()]
Log.info("Room %s registered." % roomname)
if rpc_id(multiplayer.get_remote_sender_id(), "server_response", 0, rooms[roomname]) != OK:
2024-12-05 10:34:01 +01:00
Log.warning("Failed to send elevation reply to %d!")
2024-12-06 11:11:51 +01:00
elif server_peers[roomname] is Array:
# For some odd reason, Godot doesn't allow casting server_peers[roomname] to an Array, but this works?
var peer_array: Array = server_peers[roomname]
peer_array.append(multiplayer.get_remote_sender_id())
2024-12-06 11:11:51 +01:00
server_peers[roomname] = peer_array
if rpc_id(multiplayer.get_remote_sender_id(), "server_response", 1, rooms[roomname]) != OK:
2024-12-05 10:34:01 +01:00
Log.warning("Failed to send join reply to %d!")
2024-12-06 11:11:51 +01:00
mutex.unlock()
return
Log.debug("Room %s joined by peer %d." % [roomname, multiplayer.get_remote_sender_id()])
else:
Log.warning("Peer %d tried to send this client a request meant for the authority!" % multiplayer.get_remote_sender_id())
if rpc_id(multiplayer.get_remote_sender_id(), "server_response", 2, 0) != OK:
2024-12-05 10:34:01 +01:00
Log.warning("Failed to send fail reply to %d!")
mutex.unlock()
2024-12-05 10:34:01 +01:00
# This is the message sent out if the host of the room has left.
@rpc("authority", "call_remote", "reliable")
func host_disconnected_message() -> void:
2024-12-06 11:11:51 +01:00
Log.info("Host disconnected from the session.")
managerID = 1
close_network()
if get_tree().change_scene_to_file("res://start.tscn"):
Log.warning("Failed to change back to the main scene!")
2024-12-05 10:34:01 +01:00
# This is the message sent out if a peer in the room has left.
@rpc("authority", "call_remote", "reliable")
2024-12-06 11:11:51 +01:00
func peer_disconnected_message(id: int) -> void:
for peer: String in peers:
if peers[peer] == id:
if !peers.erase(peer):
Log.warning("Peer %d never existed, but was tasked to remove it!" % id)
return
if emit_signal("playerlist_changed") != OK:
Log.warning("Couldn't emit playerlist_changed signal.")
return
Log.info("Peer %d successfully removed." % id)
2024-12-05 10:34:01 +01:00
# This is the initial server response, which tells the client what its role is.
@rpc("authority", "call_remote", "reliable")
func server_response(status: int, manager: int) -> void:
2024-12-05 10:34:01 +01:00
match status:
0:
# create new room host
isManager = true
if get_tree().change_scene_to_file("res://scenes/ui/lobby.tscn") != OK:
Log.error("Couldn't change to the lobby scene! Closing application.", "Couldn't change to the lobby scene!")
peers[Game.username] = manager
managerID = manager
2024-12-05 10:34:01 +01:00
1:
# create new room member
managerID = manager
if rpc_id(managerID, "request_playerlist", Game.username) != OK:
Log.warning("Failed to send username to room host %d." % multiplayer.get_remote_sender_id())
2024-12-05 10:34:01 +01:00
2:
Log.warning("Sent a join request to a regular peer, not the server!")
# This informs the server of the recently joined player.
@rpc("any_peer", "call_remote", "reliable")
func inform_server(roomname: String) -> void:
mutex.lock()
if multiplayer.get_unique_id() == 1:
if multiplayer.get_remote_sender_id() == rooms[roomname]:
2024-12-06 11:11:51 +01:00
var peer_array: Array = server_peers[roomname]
peer_array.append(multiplayer.get_remote_sender_id())
2024-12-06 11:11:51 +01:00
server_peers[roomname] = peer_array
else:
Log.warning("Peer %d attempted to trick the server into adding it to the room!" % multiplayer.get_remote_sender_id())
else:
Log.warning("Peer %d tried to send this client a request meant for the authority!" % multiplayer.get_remote_sender_id())
mutex.unlock()
# Get the playerlist from the room host.
@rpc("any_peer", "call_remote", "reliable")
func request_playerlist(peername: String) -> void:
if isManager:
2024-12-06 11:11:51 +01:00
if !peers.has(peername) and !peers.has(multiplayer.get_remote_sender_id()):
peers[peername] = multiplayer.get_remote_sender_id()
if rpc_id(multiplayer.get_remote_sender_id(), "send_playerlist", 0, peers) != OK:
Log.warning("Failed to send join status to %d." % multiplayer.get_remote_sender_id())
2024-12-06 11:11:51 +01:00
if !peers.erase(peername):
Log.warning("Couldn't reverse adding %d to the playerlist." % peername)
return
if rpc_id(1, "inform_server", Game.roomname) != OK:
Log.warning("Failed to send playerinfo to server.")
2024-12-06 11:11:51 +01:00
return
for peer: String in peers:
var remote_id: int = peers[peer]
if remote_id != multiplayer.get_unique_id():
if rpc_id(remote_id, "send_playerlist", 1, peers) != OK:
Log.warning("Failed to send playerlist to %d." % multiplayer.get_remote_sender_id())
if emit_signal("playerlist_changed") != OK:
Log.warning("Couldn't emit playerlist_changed signal.")
else:
if rpc_id(multiplayer.get_remote_sender_id(), "send_playerlist", 2, {}) != OK:
Log.warning("Couldn't send failure response to peer %d!" % multiplayer.get_remote_sender_id())
Log.warning("Peer %d tried to register with name %s, which already exists!" % [multiplayer.get_remote_sender_id(), peername])
else:
if rpc_id(multiplayer.get_remote_sender_id(), "send_playerlist", 3, {}) != OK:
Log.warning("Couldn't send failure response to peer %d!" % multiplayer.get_remote_sender_id())
Log.warning("Peer %d tried to send this client a request meant for the room host!" % multiplayer.get_remote_sender_id())
# Recive the playerlist on the client.
@rpc("any_peer", "call_remote", "reliable")
func send_playerlist(status: int, newPeers: Dictionary) -> void:
if multiplayer.get_remote_sender_id() == managerID:
match status:
0:
peers = newPeers
Log.info("Room %s successfully joined." % Game.roomname)
if get_tree().change_scene_to_file("res://scenes/ui/lobby.tscn") != OK:
Log.error("Couldn't change to the lobby scene! Closing application.", "Couldn't change to the lobby scene!")
1:
peers = newPeers
Log.debug("Playerlist recived.")
if emit_signal("playerlist_changed") != OK:
Log.warning("Couldn't emit playerlist_changed signal.")
2:
Log.warning("Name %s is already taken in the room!" % Game.username)
3:
Log.warning("Sent a playerlist request to a regular peer, not the host!")