# Multiplayer missions
Multiplayer missions are missions where every player is connected to a common server. That server accepts commands from players and also send notifications to the players. This enables player-to-player communication, as well as adding web operators to interact with the players.
Missions are built on shared data which is mutated and events are generated as a result of those mutations. All clients may subscribe to areas of the data and then be advised of all changes. Consistency is assured by the client and server buffering which enables in-order guarantee on message delivery. Clients disconnecting and reconnecting will have their messages automatically delivered.
# Server Termination Commands (terminationCommands
)
The server will interpret the terminationCommands
shared data. This section is specifically for clients to register commands which will be run when they are disconnected and then purged from the server after not reconnecting for some time. terminationCommands
are very important in that they assure server data is not stale even in the face of player disconnects.
Generally, you'll want to add yourself (with an ID) to a connectedAircraft
. Given that, you'll want to also set up a terminationCommands.{id}
which goes and removes connectedAircraft.{id}
. This way, when you disconnect the server is able to automatically clean up for you and clients will be notified of the delete operation, in case they need to respond accordingly.
# Shared Data
Shared data is the foundation of the multiplayer platform. The server will store arbitrary values and the clients may subscribe to updates on those values. Each client carefully issues commands to update
and delete
data, using a policy
to avoid conflicts and enable merging of commands.
Both on the aircraft and on the web you may use set_shared_data
to issue these commands. In the aircraft, MultiplayerClient
is your gateway to multiplayer data.
# MultiplayerClient
MultiplayerClient
is the type of object returned by fn.create_multiplayer_connection
. This object represents the interface to the multiplayer server and it has various functions to call and state to access.
Function | Parameters | Remarks |
---|---|---|
Connect | url , userId , roomId , roomPassword | Establish a connection to the server. |
Subscribe | path , callback<object> | Subscribe to a path and retrieve the current data via callback |
Get | path , callback<object> | Get a pat and retrieve the current data via callback |
Send | message | Send a message object |
Close | None. | Disconnect and destroy the connection |
Message Types:
Message Type | Parameters | Remarks |
---|---|---|
read | path , value | The server has returned an error regarding a recent command you sent. |
update | value , value , policy | The server has returned an error regarding a recent command you sent. |
delete | path | The server is sending you data about shared data changes. |
Policy | Remarks |
---|---|
delta | Value is relative. |
no_overwrite | Ignore update if path exists. |
Property | Remarks |
---|---|
Status | Connection status. |
Event Handler | Parameters | Remarks |
---|---|---|
OnError | error (string) | The server has returned an error regarding a recent command you sent. |
OnMessage | data (object) | The server is sending you data about shared data changes. |
Connection status values:
Status | Remarks |
---|---|
Unknown | Default state, this is the status before calling connect. |
Disconnected | The connection is disconnected, retry will be automatically attempted. |
Connecting | Connecting to the server. |
LoggingIn | Server is connected, handshake in progress. |
LoginFailed | Fatal. Login was not successful. |
Connected | Currently connected. |
# Multiplayer simple scoring example
This mission creates a score table and each aircraft has a button to set the score for that player.
Additional features:
- Show flight plans on the web client map
- Clients will clean up their
connectedAircraft
andterminationCommands
entries, but not their score.
{
"title": "Multiplayer Score Mission Test Program",
"author":"davux3",
"api_version": 0.1,
"aircraft": ["H145"],
"data":{
"server_url": "wss://5ed547d.online-server.cloud/mpserver/ws",
"create_room_url": "https://davux.com/dispatcher/",
"webConfig": {
"fligihtPlans": {
"type":"map_line",
"source":{"static":"flightPlans"},
"name":"Flight Plan",
"stroke":{"no_resolve":{"color": "#d303fc", "width":2}},
"icon":{"static":"icons.wp_blue"}
},
"connectedAircraftIcons": {
"type":"map_point",
"source":{"static":"connectedAircraft"},
"name":"Connected Aircraft",
"text":"{UserName}",
"icon":{"static":"icons.h160_icon"}
},
"scoreList": {
"type":"list",
"source":{"static":"gameScores"},
"title":"Game Scores",
"emptyText":"No players have connected yet.",
"rows":{
"row0":{
"1": {"text":"{UserName}"},
"2": {"text":"Total Score: {0}", "params": [ {"round":{"param":"Score"}} ]},
"3": {"text":""}
}
}
},
"connectedAircraftList": {
"type":"list",
"source":{"static":"connectedAircraft"},
"title":"Connected Aircraft",
"emptyText":"No aircraft are connected right now",
"rows":{
"row0":{
"1": {"icon":{"static":"icons.h160_icon"}},
"2": {"text":"{UserName}"},
"4": {"button":"View","commands": [ {"set_map_center": {"param": "location"}, "zoom": 16} ]}
}
}
}
}
},
"briefing":[
{"#comment":[
"MP_MODE ... 0: not set, 1: offline, 2: online"
]},
{"title":"Mission Initial Setup", "show_condition": {"require":{"local":"MP_MODE"}, "eq": 0}},
{"buttonbar":[
{"title":"Offline (Single player)", "commands": [ {"set":{"local":"MP_MODE"}, "value":1} ]},
{"title":"Online (Multiplayer)", "commands": [ {"call_macro":"mp_open_login_dialog"} ]}
], "show_condition": {"require":{"local":"MP_MODE"}, "eq": 0}},
{"title":"Multiplayer (Online)", "show_condition": {"require":{"local":"MP_MODE"}, "eq": 2}},
{"buttonbar":[
{"title":"View Multiplayer Status", "commands": [ {"call_macro":"mp_open_login_dialog"} ]}
], "show_condition": {"require":{"local":"MP_MODE"}, "eq": 2}},
{"title":"Game Score", "show_condition": {"require":{"local":"MP_MODE"},"ne":0}},
{"text":"My score: {local:MY_SCORE}", "show_condition": {"require":{"local":"MP_MODE"},"ne":0}},
{"buttonbar":[
{
"title":"Increment My Score",
"commands":[
{"set":{"local":"MY_SCORE"},"value":{"add":[ {"local":"MY_SCORE"}, 1 ]}},
{"set_shared_data":"update", "path":"gameScores.{service_auth}.Score", "value": {"local":"MY_SCORE"} }
]
}
],
"show_condition": {"require":{"local":"MP_MODE"},"ne":0}}
],
"events": {
"ON_MISSION_ABORTING": {
"commands": [ {"call_macro":"mp_aborting_mission"} ]
}
},
"macros":{
"mp_open_login_dialog":[
{"#comment": "Show the login dialog dispatch (or multiplayer status"},
{"set_dispatch":[
{"buttonbar":[ {"title":"<- Back to briefing", "commands": [{"set_briefing_dialog":1} ]} ]},
{"title":"Log in", "show_condition": {"require":{"local":"MP_MODE"}, "eq": 0}},
{"text":"You are playing offline.", "show_condition": {"require":{"local":"MP_MODE"}, "eq": 1}},
{"text":{"text":"User Id: {0}", "params":[{"local":"service_auth"}]}, "show_condition": {"require":{"local":"MP_MODE"}, "eq": 0}},
{"text":"User Name:", "show_condition": {"require":{"local":"MP_MODE"}, "eq": 0}},
{"textbox":"mp_userName", "show_condition": {"require":{"local":"MP_MODE"}, "eq": 0}},
{"text":"Room:", "show_condition": {"require":{"local":"MP_MODE"}, "eq": 0}},
{"textbox":"mp_room", "show_condition": {"require":{"local":"MP_MODE"}, "eq": 0}},
{"text":"Password:", "show_condition": {"require":{"local":"MP_MODE"}, "eq": 0}},
{"textbox":"mp_password", "show_condition": {"require":{"local":"MP_MODE"}, "eq": 0}},
{"buttonbar":[
{"title":"Create Room (Opens on PC)", "commands": [ {"open_url":"{static:create_room_url}?room={local:mp_room}"} ]},
{"title":"Log In", "commands": [ {"call_macro":"mp_login"} ]}
],
"disabled_condition":{"require":{"struct":{"local":"MP_CONN"}, "path":"Status"},"eq":"Connected"},
"show_condition": {"require":{"local":"MP_MODE"}, "eq": 0}},
{"text":{"text":"MP Connection Status: {0}", "params":[
{"struct":{"local":"MP_CONN"}, "path":"Status"}
]}, "show_condition": {"require":{"local":"MP_MODE"}, "ne": 1}},
{"text":{"text":"MP Server Last Error: {local:MP_LAST_ERROR}"}, "show_condition": {"require":{"local":"MP_MODE"}, "ne": 1}},
{"title":"Debug Info"},
{"text":{"text":"Multiplayer Mode: {0}", "params":[
{"switch":{"local":"MP_MODE"}, "case":{
"0": "Undecided",
"1": "Offline, Singleplayer",
"2": "Multiplayer"
}}
]}},
{"#comment":{"text":"Debug MP Message: {local:MP_MSG}"}, "show_condition": {"require":{"local":"MP_MODE"}, "ne": 1}}
]},
{"set_dispatch_dialog":1}
],
"mp_login":[
{"#comment":"try to make the actual connection to the server"},
{"set":{"param":"service_auth"},"value":{"local":"service_auth"}},
{"set":{"local":"MP_LAST_ERROR"},"value":""},
{"set":{"local":"MP_CONN"}, "value": {"fn": "create_multiplayer_connection"}},
{"set":{"local":"MP_CONN", "path":"OnError"}, "value":{"js:create_async_function":[
{"set":{"local":"MP_LAST_ERROR"},"value":{"struct": {"param":"$args"}, "index": 0}}
]}},
{"set":{"local":"MP_CONN", "path":"OnMessage"}, "value":{"js:create_async_function":[
{"set":{"param":"arg0"},"value":{"struct": {"param":"$args"}, "index": 0}},
{"call_macro":"mp_on_message","params":{"msg": {"param":"arg0"}}}
]}},
{"set":{"param":"unused"},"value":{"struct":{"local":"MP_CONN"}, "function":"Connect", "params":[
{"static":"server_url"}, {"param":"service_auth"}, {"local":"mp_room"}, {"local":"mp_password"}
]}},
{"create_thread":{"commands":[
{"wait_for":{"struct":{"local":"MP_CONN"}, "path":"Status"},"eq":"Connected"},
{"#comment":"once we log in once, we're committed to muoltiplayer"},
{"set":{"local":"MP_MODE"}, "value": 2},
{"set_briefing_dialog":1},
{"#comment":"First create terminationCommands with no_overwrite, then add an entry for us, and then populate with commands to clear us from connectedAircraft and terminationCommands when we become stale on the server"},
{"set_shared_data":"update",
"path":"terminationCommands",
"policy":"no_overwrite",
"value": {"create_struct":{}}
},
{"set_shared_data":"update",
"path":"terminationCommands.{service_auth}",
"value": {"create_struct":{
"removeFromConnectedAircraft":{"create_struct":{
"type":"delete",
"path":"connectedAircraft.{service_auth}"
}},
"removeFromFlightPlans":{"create_struct":{
"type":"delete",
"path":"flightPlans.{service_auth}"
}},
"removeFromTerminationCommands":{"create_struct":{
"type":"delete",
"path":"terminationCommands.{service_auth}"
}}
}}},
{"#comment":"make sure we have connectedAircraft table. all players must use no_overwrite when ensuring the table exists to prevent anybody from destroying the table."},
{"set_shared_data":"update", "path":"connectedAircraft", "policy":"no_overwrite", "value": {"create_struct":{}} },
{"set_shared_data":"update", "path":"icons", "policy":"no_overwrite", "value": {"fn":"get_mission_icons"} },
{"set_shared_data":"update", "path":"flightPlans", "policy":"no_overwrite", "value": {"create_struct":{}} },
{"set_shared_data":"update", "path":"webConfig", "policy":"no_overwrite", "value": {"static":"webConfig"} },
{"set_shared_data":"update", "path":"gameScores", "policy":"no_overwrite", "value": {"create_struct":{}} },
{"set_shared_data":"update",
"path":"connectedAircraft.{service_auth}",
"value": {"create_struct":{
"location":{"resolve_location":"$USER"},
"UserName": {"local":"mp_userName"}
}}},
{"set_shared_data":"update",
"path":"gameScores.{service_auth}",
"value": {"create_struct":{
"UserName": {"local":"mp_userName"},
"Score": 0
}}
},
{"#comment":"update our location, score and flightplan (if changed) forever"},
{"while":1,"eq":1,"do":[
{"sleep":5},
{"set_shared_data":"update", "path":"connectedAircraft.{service_auth}.location", "value": {"resolve_location":"$USER"} },
{"set_shared_data":"update", "path":"gameScores.{service_auth}.Score", "value": {"local":"MY_SCORE"} },
{"if":{"json:stringify": {"local":"$FLIGHTPLAN"}}, "ne": {"param":"FPL"},"then":[
{"set":{"param":"FPL"},"value":{"json:stringify": {"local":"$FLIGHTPLAN"}}},
{"set_shared_data":"update", "path":"flightPlans.{service_auth}", "value": {"create_struct":{
"points":{"local":"$FLIGHTPLAN"}
}}}
]}
]}
]}}
],
"mp_initialize":[
{"#comment":"setup for multiplayer operations later"},
{"set":{"local":"MP_LAST_ERROR"},"value":""},
{"set":{"local":"MP_MODE"},"value":0},
{"#comment":"MP_MODE 0: undecided, 1: offline, 2:online"},
{"#comment":"these are for debugging only"},
{"set":{"local":"MP_MSG"},"value":""},
{"set":{"local":"mp_room"},"value":""},
{"set":{"local":"mp_password"},"value":""},
{"set":{"local":"mp_userName"},"value":{"var":["ATC AIRLINE","string"]}},
{"#comment": "Create or access a unique ID to identify you on the server irrespective of callsign"},
{"set":{"local":"service_auth"}, "value":{"fn":"create_guid"}},
{"create_thread":{"commands":[
{"wait_for":{"local":"MP_MODE"},"ne":0},
{"call_macro":"mp_begin"}
]}}
],
"mp_on_message":[
{"#comment":"param - msg"},
{"#comment":"handle READ, UPDATE and DELETE operations below"},
{"set":{"param":"json"},"value":{"json:stringify":{"param":"msg"}}},
{"switch":{"struct": {"param":"msg"}, "path": "type"}, "case": {
"read":[
{"set":{"local":"MP_MSG"},"value": "we got an read: {json}"}
],
"update": [
{"set":{"local":"MP_MSG"},"value": "we got an update: {json}"}
],
"delete": [
{"set":{"local":"MP_MSG"},"value": "we got an delete: {json}"}
]
}}
],
"mp_begin":[
{"#comment":"called once we decided if we are single or muliplayer. MP_MODE 1:offline, 2:online"},
{"#comment":"offline case, manually run the logic and complete logic"},
{"set_objective_title":"Ready to play the game!"}
],
"mp_aborting_mission":[
{"#comment":"we want to clean up our multiplayer connection if it was created"},
{"if":{"local":"MP_CONN"},"ne":null, "then":[
{"set":{"param":"unused"},"value":{"struct":{"local":"MP_CONN"}, "function":"Close", "params":[]}}
]}
]
},
"objectives": [
{
"title": "Setup required",
"commands": [
{"set":{"local":"MY_SCORE"},"value":0},
{"call_macro":"mp_initialize"},
{"sleep": "forever"}
]
}
],
"icons":{
"wp_blue":"",
"h160_icon": ""
}
}