# 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 and terminationCommands 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": ""
  }
}