using System; using System.Collections.Generic; using System.Linq; using Lidgren.Network; using StardewValley.Network; namespace StardewModdingAPI.Framework.Networking { /// Metadata about a connected player. internal class MultiplayerPeer : IMultiplayerPeer { /********* ** Properties *********/ /// The server through which to send messages, if this is an incoming farmhand. private readonly SLidgrenServer Server; /// The client through which to send messages, if this is the host player. private readonly SLidgrenClient Client; /// The network connection to the player. private readonly NetConnection ServerConnection; /********* ** Accessors *********/ /// The player's unique ID. public long PlayerID { get; } /// Whether this is a connection to the host player. public bool IsHost { get; } /// Whether the player has SMAPI installed. public bool HasSmapi => this.ApiVersion != null; /// The player's OS platform, if is true. public GamePlatform? Platform { get; } /// The installed version of Stardew Valley, if is true. public ISemanticVersion GameVersion { get; } /// The installed version of SMAPI, if is true. public ISemanticVersion ApiVersion { get; } /// The installed mods, if is true. public IEnumerable Mods { get; } /********* ** Public methods *********/ /// Construct an instance. /// The player's unique ID. /// The metadata to copy. /// The server through which to send messages. /// The server connection through which to send messages. /// The client through which to send messages. /// Whether this is a connection to the host player. public MultiplayerPeer(long playerID, RemoteContextModel model, SLidgrenServer server, NetConnection serverConnection, SLidgrenClient client, bool isHost) { this.PlayerID = playerID; this.IsHost = isHost; if (model != null) { this.Platform = model.Platform; this.GameVersion = model.GameVersion; this.ApiVersion = model.ApiVersion; this.Mods = model.Mods.Select(mod => new MultiplayerPeerMod(mod)).ToArray(); } this.Server = server; this.ServerConnection = serverConnection; this.Client = client; } /// Construct an instance for a connection to an incoming farmhand. /// The player's unique ID. /// The metadata to copy, if available. /// The server through which to send messages. /// The server connection through which to send messages. public static MultiplayerPeer ForConnectionToFarmhand(long playerID, RemoteContextModel model, SLidgrenServer server, NetConnection serverConnection) { return new MultiplayerPeer( playerID: playerID, model: model, server: server, serverConnection: serverConnection, client: null, isHost: false ); } /// Construct an instance for a connection to the host player. /// The player's unique ID. /// The metadata to copy. /// The client through which to send messages. /// Whether this connection is for the host player. public static MultiplayerPeer ForConnectionToHost(long playerID, RemoteContextModel model, SLidgrenClient client, bool isHost) { return new MultiplayerPeer( playerID: playerID, model: model, server: null, serverConnection: null, client: client, isHost: isHost ); } /// Get metadata for a mod installed by the player. /// The unique mod ID. /// Returns the mod info, or null if the player doesn't have that mod. public IMultiplayerPeerMod GetMod(string id) { if (string.IsNullOrWhiteSpace(id) || this.Mods == null || !this.Mods.Any()) return null; id = id.Trim(); return this.Mods.FirstOrDefault(mod => mod.ID != null && mod.ID.Equals(id, StringComparison.InvariantCultureIgnoreCase)); } /// Send a message to the given peer, bypassing the game's normal validation to allow messages before the connection is approved. /// The message to send. public void SendMessage(OutgoingMessage message) { if (this.IsHost) this.Client.sendMessage(message); else this.Server.SendMessage(this.ServerConnection, message); } } }