summaryrefslogtreecommitdiff
path: root/src/SMAPI/Framework/Networking/SLidgrenServer.cs
blob: ff871e64182b2135e72b742038f3c54660addaca (plain)
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
using System;
using System.Diagnostics.CodeAnalysis;
using System.IO;
using Lidgren.Network;
using StardewValley.Network;

namespace StardewModdingAPI.Framework.Networking
{
    /// <summary>A multiplayer server used to connect to an incoming player. This is an implementation of <see cref="LidgrenServer"/> that adds support for SMAPI's metadata context exchange.</summary>
    internal class SLidgrenServer : LidgrenServer
    {
        /*********
        ** Fields
        *********/
        /// <summary>SMAPI's implementation of the game's core multiplayer logic.</summary>
        private readonly SMultiplayer Multiplayer;

        /// <summary>A callback to raise when receiving a message. This receives the incoming message, a method to send a message, and a callback to run the default logic.</summary>
        private readonly Action<IncomingMessage, Action<OutgoingMessage>, Action> OnProcessingMessage;


        /*********
        ** Public methods
        *********/
        /// <summary>Construct an instance.</summary>
        /// <param name="multiplayer">SMAPI's implementation of the game's core multiplayer logic.</param>
        /// <param name="gameServer">The underlying game server.</param>
        /// <param name="onProcessingMessage">A callback to raise when receiving a message. This receives the incoming message, a method to send a message, and a callback to run the default logic.</param>
        public SLidgrenServer(IGameServer gameServer, SMultiplayer multiplayer, Action<IncomingMessage, Action<OutgoingMessage>, Action> onProcessingMessage)
            : base(gameServer)
        {
            this.Multiplayer = multiplayer;
            this.OnProcessingMessage = onProcessingMessage;
        }


        /*********
        ** Protected methods
        *********/
        /// <summary>Parse a data message from a client.</summary>
        /// <param name="rawMessage">The raw network message to parse.</param>
        [SuppressMessage("ReSharper", "AccessToDisposedClosure", Justification = "The callback is invoked synchronously.")]
        protected override void parseDataMessageFromClient(NetIncomingMessage rawMessage)
        {
            // add hook to call multiplayer core
            NetConnection peer = rawMessage.SenderConnection;
            using IncomingMessage message = new();
            using Stream readStream = new NetBufferReadStream(rawMessage);
            using BinaryReader reader = new(readStream);

            while (rawMessage.LengthBits - rawMessage.Position >= 8)
            {
                message.Read(reader);
                NetConnection connection = rawMessage.SenderConnection; // don't pass rawMessage into context because it gets reused
                this.OnProcessingMessage(message, outgoing => this.sendMessage(connection, outgoing), () =>
                {
                    if (this.peers.ContainsLeft(message.FarmerID) && this.peers[message.FarmerID] == peer)
                        this.gameServer.processIncomingMessage(message);
                    else if (message.MessageType == StardewValley.Multiplayer.playerIntroduction)
                    {
                        NetFarmerRoot farmer = this.Multiplayer.readFarmer(message.Reader);
                        this.gameServer.checkFarmhandRequest("", this.getConnectionId(rawMessage.SenderConnection), farmer, msg => this.sendMessage(peer, msg), () => this.peers[farmer.Value.UniqueMultiplayerID] = peer);
                    }
                });
            }
        }
    }
}