This page documents a proposal to add versioning to the WireProtocol and to deploy a new wire protocol version that supports additional features.
The immediate goal of this proposal is to add bi-directional exchange of arbitrary key-value metadata. The intended use case is to allow extensions to extend existing commands in new and creative ways. But, this functionality may also be used by Mercurial core and core-shipped extensions to add features to Mercurial in the future. For example, this feature could facilitate the transmission of messages for client-side display on every command - not just commands that support it today.
Version Negotiation
A capability will be exposed stating which wire protocol versions are supported. e.g.
protocols=1,2
At the beginning of interaction with a remote, the client will perform a capabilities command like it has done for years.
If the remote does not expose the protocols capability, the client will assume wireproto version 1 is the only version supported.
If the remote exposes the protocols capability, the client may select any supported version for subsequent commands issued during that session. Selecting the highest mutually-supported version is highly recommended.
Versioned SSH Wire Protocol
The SSH peer will need to identify the version of the wire protocol in use. We have a number of options here.
Commands are prefixed with a version identifier
In this solution, each command string is prefixed by a version identifier. The version identifier is likely something binary so it can't be confused for a command name. e.g.
\x01\x02getbundle\n
Here, the the version identifier is \x01\x02 and the command is getbundle. \x01 says a version identifier will follow in the next byte. \x02 is the protocol version. In this case, version 2. The prefix could be any byte sequence we want.
In this solution, command receivers will examine the first few bytes of every supposed command line and look for the special version identifying bytes. If found, it will interpret the remaining payload using the requested wire protocol version. If not found, it will assume version 1 (the original) wire protocol is in use. Therefore, this method is backwards compatible with old clients.
Client issues version switch command
In this solution, the client will issue a special command indicating an upgrade of the current channel to a new wire protocol version. e.g.
setwireproto\n 2\n
All subsequent commands will use the specified version of the wire protocol.
Clients should not send this command unless both they and the remote support the specified version (via capabilities negotiation). If this command is not received, the remote will assume version 1 (the original) wire protocol is in use.