Node.js Lockstep Multiplayer Architecture
Posted
by
Wakaka
on Game Development
See other posts from Game Development
or by Wakaka
Published on 2013-10-26T04:44:50Z
Indexed on
2013/10/26
10:15 UTC
Read the original article
Hit count: 353
Background
I'm using the lockstep model for a multiplayer Node.js/Socket.IO game in a client-server architecture. User input (mouse or keypress) is parsed into commands like 'attack'
and 'move'
on the client, which are sent to the server and scheduled to be executed on a certain tick. This is in contrast to sending state data to clients, which I don't wish to use due to bandwidth issues.
Each tick, the server will send the list of commands on that tick (possibly empty) to each client. The server and all clients will then process the commands and simulate that tick in exactly the same way. With Node.js this is actually quite simple due to possibility of code sharing between server and client. I'll just put the deterministic simulator in the /shared folder which can be run by both server and client. The server simulation is required so that there is an authoritative version of the simulation which clients cannot alter.
Problem
Now, the game has many entity classes, like Unit
, Item
, Tree
etc. Entities are created in the simulator. However, for each class, it has some methods that are shared and some that are client-specific. For instance, the Unit
class has addHp
method which is shared. It also has methods like getSprite
(gets the image of the entity), isVisible
(checks if unit can be seen by the client), onDeathInClient
(does a bunch of stuff when it dies only on the client like adding announcements) and isMyUnit
(quick function to check if the client owns the unit). Up till now, I have been piling all the client functions into the shared Unit
class, and adding a this.game.isServer()
check when necessary. For instance, when the unit dies, it will call if (!this.game.isServer()) { this.onDeathInClient(); }
.
This approach has worked pretty fine so far, in terms of functionality. But as the codebase grew bigger, this style of coding seems a little strange. Firstly, the client code is clearly not shared, and yet is placed under the /shared folder. Secondly, client-specific variables for each entity are also instantiated on the server entity (like unit.sprite
) and can run into problems when the server cannot instantiate the variable (it doesn't have Image
class like on browsers).
So my question is, is there a better way to organize the client code, or is this a common way of doing things for lockstep multiplayer games? I can think of a possible workaround, but it does have its own problems.
Possible workaround (with problems)
I could use Javascript mixins that are only added when in a browser. Thus, in the /shared/unit.js file in the /shared folder, I would have this code at the end:
if (typeof exports !== 'undefined') module.exports = Unit;
else mixin(Unit, LocalUnit);
Then I would have /client/localunit.js store an object LocalUnit
of client-side methods for Unit
.
Now, I already have a publish-subscribe system in place for events in the simulator. To remove the this.game.isServer()
checks, I could publish entity-specific events whenever I want the client to do something. For instance, I would do this.publish('Death')
in /shared/unit.js and do this.subscribe('Death', this.onDeathInClient)
in /client/localunit.js. But this would make the simulator's event listeners list on the server and the client different. Now if I want to clear all subscribed events only from the shared simulator, I can't. Of course, it is possible to create two event subscription systems - one client-specific and one shared - but now the publish()
method would have to do if (!this.game.isServer()) { this.publishOnClient(event); }
.
All in all, the workaround off the top of my head seems pretty complicated for something as simple as separating the client and shared code. Thus, I wonder if there is an established and simpler method for better code organization, hopefully specific to Node.js games.
© Game Development or respective owner