Runtime protocol reference
Introduction
Marlowe Runtime is the application backend for managing Marlowe contracts on the Cardano blockchain. The term "Runtime" refers to the whole system, its components, and sub-components. Runtime and its protocols and sub-protocols communicate with one another to enable Runtime core functions, which are discovering and querying on-chain Marlowe contracts, and creating Marlowe transactions.
The Marlowe Runtime protocol enables communicating with Runtime directly.
These protocols, and specifically this root-level marlowe-runtime-cli protocol, is the primary API for Runtime.
If you want to communicate with Runtime, ultimately, you will be using the marlowe-runtime-cli protocol and its sub-protocols. 
In the larger context of working with Runtime, you don't necessarily need to be aware of the protocols to be able to communicate with Runtime. For example, if you are using the REST API or Runtime CLI, under the hood, they all communicate with Runtime using this protocol, as does everything that communicates with Runtime.
Intended audience
This document is intended for developers who are writing client applications that interact with Runtime. You may be writing scripts, command-line tools, servers for DApps that communicate with Runtime, or something similar. If you don't want to use the REST API, Runtime CLI, or Marlowe CLI, or any other intermediary, but instead want to connect directly to the TCP socket that Runtime exposes, this document is a useful reference because it describes the behavior, syntax, states, and messages of these protocols.
Key concepts
| Concepts | Description | 
|---|---|
| Peer role | The peer role is either server or client. All protocol sessions take place between two peers -- one server and one client. | 
| Protocol state | A description of the current state of a protocol session. | 
| Peer agency | Describes which peer role is able to send messages to the other in a given protocol state. Agency is exclusive (either server or client has agency, never both). When one peer is able to send a message to the other peer, it has agency. When a peer is in a state of only being able to receive a message, we say it does not have agency. States in which neither peer has agency are terminal states. | 
| Messages | Packets of data that a peer can send to another peer when it has agency. When a message is sent, that event transitions the protocol state from one state to another. | 
About messages and agency
Certain message types are only available from certain states, and when a peer sends a message to another, it changes the state of the protocol, potentially changing the peer agency also. In the majority of cases, agency alternates when a peer sends a message to the other peer. For example, the client will send a message to the server, then the server will gain agency. Then the server sends a message back to the client and the client will gain agency again. There are some cases in which one peer will send multiple messages and keep agency between them, but for the most part it is a back and forth process.
Marlowe Runtime protocol
Below is a state diagram for the Marlowe Runtime protocol:
The protocols are all defined using the typed-protocols library.
The top-level protocol is the Marlowe Runtime protocol. It is defined here:
Four sub-protocols
The Marlowe Runtime protocol consists of four sub-protocols:
- Marlowe Sync
- Marlowe Header Sync
- Marlowe Bulk Sync
- Marlowe Query
- Tx Job
- Marlowe Load
- Contract Query
- Marlowe Transfer
Client starts session
The client starts a session by sending a message to the server saying which sub-protocol it intends to communicate with, then a protocol session for the sub-protocol begins.
Marlowe Runtime protocol states
At any given time, the Marlowe Runtime protocol can be in one of five states:
- Init
- MarloweSync
- MarloweHeaderSync
- MarloweBulkSync
- MarloweQuery
- TxJob
- MarloweLoad
- ConractQuery
- MarloweTransfer
There is an init state, then there is one protocol state per sub-protocol.
These states transition in a fairly straightforward manner. 
The protocol starts in the Init state, and then via one of the message types, it will transition into one of the other states.
Once in that protocol state, the protocol stays there for the rest of the session, communicating in that protocol.
Descriptions of the Marlowe Runtime protocol states
| Protocol state | Agency | Parameter | Description | 
|---|---|---|---|
| 1. Init | Client | The initial state. | |
| 2. MarloweSync st | Determined by st | The peers are communicating via the MarloweSyncsub-protocol. | |
| st | A protocol state from MarloweSync | ||
| 3. MarloweHeaderSync st | Determined by st | The peers are communicating via the MarloweHeaderSyncsub-protocol. | |
| st | A protocol state from MarloweHeaderSync | ||
| 4. MarloweBulkSync st | Determined by st | The peers are communicating via the MarloweBulkSyncsub-protocol. | |
| st | A protocol state from MarloweBulkSync | ||
| 5. MarloweQuery st | Determined by st | The peers are communicating via the Query MarloweSyncRequestsub-protocol. | |
| st | A protocol state from Query MarloweSyncRequest | ||
| 6. TxJob st | Determined by st | The peers are communicating via the Job MarloweTxCommandsub-protocol. | |
| st | A protocol state from Job MarloweTxCommand | ||
| 7. MarloweLoad st | Determined by st | The peers are communicating via the MarloweLoadsub-protocol. | |
| st | A protocol state from MarloweLoad | ||
| 8. ContractQuery st | Determined by st | The peers are communicating via the Query ContractRequestsub-protocol. | |
| st | A protocol state from Query ContractRequest | ||
| 9. MarloweTransfer st | Determined by st | The peers are communicating via the MarloweTransfersub-protocol. | |
| st | A protocol state from MarloweTransfer | 
Message types
There are eight message types in the Marlowe Runtime protocol.
The first seven initiate sub-protocol sessions, while the final seven are carriers for sub-protocol messages.
For example, the MarloweSync message type embeds a message from the sub-protocol in the Marlowe Runtime protocol. 
| Message | Begin state | End state | Parameter | Description | 
|---|---|---|---|---|
| 1. RunMarloweSync | Init | MarloweSync Init | Start a MarloweSyncsession. | |
| 2. RunMarloweHeaderSync | Init | MarloweHeaderSync Idle | Start a MarloweHeaderSyncsession. | |
| 3. RunMarloweBulkSync | Init | MarloweBulkSync Idle | Start a MarloweBulkSyncsession. | |
| 4. RunMarloweQuery | Init | MarloweQuery Req | Start a MarloweQuerysession. | |
| 5. RunTxJob | Init | TxJob Init | Start a TxJobsession. | |
| 6. RunMarloweLoad | Init | MarloweLoad (Processing RootNode) | Start a MarloweLoadsession. | |
| 7. RunContractQuery | Init | ContractQuery Req | Start a ContractQuerysession. | |
| 8. RunMarloweTransfer | Init | MarloweTransfer Idle | Start a MarloweTransfersession. | |
| 9. MarloweSync msg | MarloweSync st | MarloweSync st' | Wrap a MarloweSyncmessage. | |
| msg | A MarloweSyncmessage with begin statestand end statest' | |||
| 10. MarloweHeaderSync msg | MarloweHeaderSync st | MarloweHeaderSync st' | Wrap a MarloweHeaderSyncmessage. | |
| msg | A MarloweHeaderSyncmessage with begin statestand end statest' | |||
| 11. MarloweBulkSync msg | MarloweBulkSync st | MarloweBulkSync st' | Wrap a MarloweBulkSyncmessage. | |
| msg | A MarloweBulkSyncmessage with begin statestand end statest' | |||
| 12. MarloweQuery msg | MarloweQuery st | MarloweQuery st' | Wrap a MarloweQuerymessage. | |
| msg | A MarloweQuerymessage with begin statestand end statest' | |||
| 13. TxJob msg | TxJob st | TxJob st' | Wrap a Job MarloweTxCommandmessage. | |
| msg | A TxJobmessage with begin statestand end statest' | |||
| 14. MarloweLoad msg | MarloweLoad st | MarloweLoad st' | Wrap a Job MarloweTxCommandmessage. | |
| msg | A MarloweLoadmessage with begin statestand end statest' | |||
| 15. ContractQuery msg | ContractQuery st | ContractQuery st' | Wrap a Job MarloweTxCommandmessage. | |
| msg | A ContractQuerymessage with begin statestand end statest' | |||
| 16. MarloweTransfer msg | MarloweTransfer st | MarloweTransfer st' | Wrap a Job MarloweTxCommandmessage. | |
| msg | A MarloweTransfermessage with begin statestand end statest' | 
Binary format for sending messages over TCP
Use the binary format for the Marlowe Runtime protocol to send messages over TCP. The binary format describes how each message type is converted into binary data, then read and decoded.
At this time, Haskell is the only supported language that has the functions for this.
If you are using Haskell, use the Marlowe client library, one of our libraries in our marlowe-cardano repo. It exports a monad transformer called MarloweT that enables you to run a client of the Marlowe Runtime protocol. Run that monad transformer by providing the port number and host address of Runtime and it will connect to it and handle the binary format for you. 
Messaging behavior
On a functional level, the Marlowe Runtime protocol multiplexes the four sub-protocols into one.
The client always sends one of these eight message types (RunMarloweSync, RunMarloweHeaderSync, RunMarloweBulkSync, RunMarloweQuery, RunTxJob, RunMarloweLoad, RunContractQuery, RunMarloweTransfer) to start.
Depending on which one it started the session with, it will then continuously send that message type between client and server.
If it starts with RunMarloweSync, the client and server will then just exchange MarloweSync messages back and forth.
Inside each of those is a message from the underlying MarloweSync protocol.
When finished, it disconnects and the session is over. 
About the Haskell source files
There is a data structure in the Haskell source files that describes the different protocol states.
There is a message type that shows what all the available messages are for that protocol.
Each message indicates the initial state of the message.
For example, RunMarloweSync starts in the state StInit, then transitions into the MarloweSync.StInit state.