Clash messages are passed between client and server in PDUs containing an integer message identifier and a binary payload. The files in this package are used to map the identifier to a message structure definition so that the binary payload can be parsed.
Each message structure is defined in an individual JSON file. The name of the file is not significant, but generally it corresponds to the type name of the message.
Each file defines a type. Types are referred to by their name attribute or, for top level messages, by their id attribute (the identifier passed in the PDU).
Using just the name and id attributes we can define simple structures such as the KeepAlive message:
{
"id": 10108,
"name": "KeepAlive"
}
To read more complex messages we need to define the field list that makes up the payload. For example, the EndClientTurn message comprises three fields; tick (the time that the command was issued), checksum (the game state checksum), and commands, an array of CommandComponent structures:
{
"id": 14102,
"name": "EndClientTurn",
"fields": [
{"name": "tick", "type": "INT"},
{"name": "checksum", "type": "INT"},
{"name": "commands", "type": "CommandComponent[]"}
]
}
Each field has a name attribute that is used to refer to it and a type attribute that defines its serialized structure. The type can refer to another structure or a built in primitive, or an array of structures or primitives.
Five primitive types are defined.
-
BOOLEAN: A single bit, read in little endian order. All other primitives are read from byte boundaries.
-
BYTE: One byte.
-
INT: Four bytes, big-endian
-
LONG: Eight bytes, big-endian
-
STRING: An integer length, then a UTF-8 encoded string.
-
ZIP_STRING: An integer length, then a little-endian integer unzipped length, then a zlib compressed UTF-8 encoded string.
Arrays types are defined by appending square brackets to a type name, e.g. "INT[5]" defines an array of five ints. If the size specifier is omitted the size of the array will be read/written as an integer before the array contents. E.g. a field of type "CommandComponent[]" would be written as an INT (the array size), followed by the contents of the array, a list of command components.
There is one other specialized structure, the optional field. Optionals are defined by prefixing a type name with "?", e.g. "?LONG". They are serialized as a BOOLEAN (indicating the presence of the value), followed by the value if it is present.
Some structures have additional trailing fields that are conditionally included. Extension fields are read when their id attribute matches the id field of the structure that they extend (note: the field with the name "id" is matched against, not the id attribute of the structure). For example, the CommandComponent has extensions that are conditional on the command id:
{
"name": "CommandComponent",
"fields": [
{"name": "id", "type": "INT"}
],
"extensions": [
{
"id": 4,
"comment": "Donate unit in response to troop request",
"fields": [
{"type": "INT"},
{"name": "messageId", "type": "INT"},
{"name": "unitId", "type": "INT"},
{"type": "INT"},
{"name": "tick", "type": "INT"}
]
},
{
"id": 10,
"comment": "Donate unit to clan war castle",
"fields": [
{"type": "INT"},
{"name": "homeId", "type": "INT"},
{"name": "unitId", "type": "INT"},
{"name": "index", "type": "INT"},
{"name": "tick", "type": "INT"}
]
},
Fields with primitive values may have a default attribute to indicate the value that should be used if the field is not set.
Messages and Fields may have a comment attribute to document their use.