-
Notifications
You must be signed in to change notification settings - Fork 5.7k
BIP40: Stratum wire protocol #1557
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Draft
ben221199
wants to merge
34
commits into
bitcoin:master
Choose a base branch
from
ben221199:add-bip40
base: master
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Draft
Changes from all commits
Commits
Show all changes
34 commits
Select commit
Hold shift + click to select a range
47ca202
Write initial draft of BIP 40
ben221199 914fb7c
Fix inline code
ben221199 72bfc4d
Allow public domain
ben221199 abe08e4
Improve specification of transport protocols
ben221199 a5aee5b
Add letters and default ports
ben221199 8658be8
Add more information about WebSockets
ben221199 e13def4
Add information about WebSocket subprotocol
ben221199 e280771
Add information about HTTP polling
ben221199 83647b1
Improve text
ben221199 3d217b0
Add note
ben221199 c82a2a1
Add information about HTTP Push
ben221199 805559f
Add two methods
ben221199 ec52ae1
Improve commands
ben221199 84f8519
Process editorial feedback
ben221199 b50e9e5
Merge branch 'master' into add-bip40
ben221199 0cdfcb5
Add IANA considerations
ben221199 19a39e2
Fix links
ben221199 1cf5983
Add information about service vendor
ben221199 2f620f5
Fix layer
ben221199 48adb0f
Merge branch 'master' into add-bip40
ben221199 b5a9b10
Add IANA consideration for HTTP fields
ben221199 19cf447
Add example commands
ben221199 6e26691
Add example pub/sub commands
ben221199 73f8ad1
Move services to seperate file
ben221199 2787db0
Add information about discovery service
ben221199 40bc721
Improve service documentation
ben221199 05e1194
Improve blockchain service documentation
ben221199 16b6d5c
Apply suggestions
ben221199 4cf8985
Improve sentences
ben221199 1936b49
Improve documents of Stratum methods
ben221199 bbad8a0
Fix type and description
ben221199 387af0f
Add examples to Firstbits methods
ben221199 db5ed75
Improve documentation of server commands
ben221199 996724f
Add remaining Stratum services
ben221199 File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,182 @@ | ||
<pre> | ||
BIP: 40 | ||
Layer: API/RPC | ||
Title: Stratum wire protocol | ||
Author: Marek Palatinus <[email protected]> | ||
Ben van Hartingsveldt <[email protected]> | ||
Comments-Summary: No comments yet. | ||
Comments-URI: https://github.com/bitcoin/bips/wiki/Comments:BIP-0040 | ||
Status: Draft | ||
Type: Standards Track | ||
Created: 2024-03-07 | ||
License: PD | ||
</pre> | ||
|
||
==Abstract== | ||
|
||
This document describes the Stratum wire protocol used in clients such as Electrum, and which also laid the foundation for the Stratum mining protocol, used in many Bitcoin pools. | ||
|
||
The specification is intended to set a standard for connections between clients and servers in which the client is able to request data without having to download the full blockchain. | ||
|
||
The specification consists of three parts. In the first part, the data format is defined. In the second part, the possible transport protocols (including encapsulation) are described. In the third part, the available methods are documented. | ||
|
||
==Motivation== | ||
|
||
Many clients want to give users access to the Bitcoin ecosystem. However, for specific reasons not every user is able to download the full blockchain to their machine. | ||
|
||
The Stratum wire protocol defines a way to access the blockchain without having it downloaded. For example, a client can request both the genesis block and the latest block with the same latency, because the server synchronizes and indexes all the blocks for us. | ||
|
||
==History== | ||
|
||
The Stratum wire protocol was introduced by Marek Palatinus in late 2011 and early 2012. It was a language-independent alternative to the Python-based protocol in early versions of Electrum, created by Thomas Voegtlin. The Stratum wire protocol made it possible to create compatible servers that Electrum could connect to, but it also made it possible to write alternative clients. | ||
|
||
Later in 2012, Marek Palatinus introduced Stratum for mining pools as well: The Stratum mining protocol, as defined in [[bip-0041.mediawiki|BIP 41]]. | ||
|
||
==Extension<ref name="extended">This is extended specification information of the Stratum wire protocol to make it more complete. This information will not be found in any code or specification before this BIP.</ref>== | ||
|
||
This BIP has some extended specification information that isn't found in the older, legacy specifications. For backwards compatibility, it is recommended to implement the legacy specification information too, which is also described in this specification. | ||
|
||
==Specification: Data Format== | ||
|
||
Stratum leverages [https://www.jsonrpc.org/ JSON-RPC]. Both versions 1.0 and 2.0 are allowed, according to their respective rules. | ||
|
||
Additionally, it is a convention to give all Stratum methods in the <code>method</code> property a name that meets the following EBNF notation: <code>method ::= serviceType ("[" serviceVendor "]")? "." methodName</code>, where <code>serviceType</code> and <code>serviceVendor</code> are allowed to have dots and <code>methodName</code> isn't. For example, the values <code>blockchain.block.subscribe</code> and <code>blockchain.block[Electrum].subscribe</code>. | ||
|
||
The service vendor is optional. However, multiple vendors can define the same service type, so the service vendor is introduced to distinguish them. If the service vendor is absent, the default service type is executed. It is up to the server implementer which service vendor is the default. | ||
|
||
''Note: In JSON-RPC 1.0, the <code>param</code> property is an array, so everything should be passed in the right order. In JSON-RPC 2.0, named parameters are also allowed. In that case, only the parameter names that are documented should be used; otherwise, the method can fail. It is maybe also possible that a specific method (or server) only supports <code>params</code> as an array, because it doesn't know how to handle named params, even if it supports JSON-RPC 2.0.'' | ||
|
||
==Specification: Transport Protocols== | ||
|
||
It is possible to send JSON-RPC messages over different transport protocols, like TCP and HTTP. It is also possible to protect these protocols with SSL/TLS. | ||
|
||
''Note: Implementers may write servers that support multiple coins, assuming that those coins all use the same port (e.g. 50001). Because HTTP and WebSockets support virtual hosting (<code>Host</code> header or <code>:authority</code> pseudo-header), it is possible to create a server on a single IP address that supports multiple coins by checking the domain name. The same is the case for the transport protocols with SSL/TLS when SNI is used. On plain TCP, virtual hosting is not possible. A similar experience can be reached by using multiple IP addresses. For IPv6, a network of /96 would be enough to hypothetically support all coins ever listed in SLIP-44. For IPv4, you would need all IPv4 addresses that exist to do the same, even the invalid ones, so a suggestion is to only use the server for coins you really need. Also, other ports could possibly be used, and in that case, a /16 IP block with all 65536 TCP ports in use will be enough to support all coins. However, because using non-standard and non-default ports is not user-friendly, this is also not recommended.'' | ||
|
||
===TCP=== | ||
|
||
Stratum over a TCP connection. Every JSON-RPC message (including batch messages) is sent on a single line, ending with a line-feed (<code>\n</code>), so <code>\r\n</code> is also allowed. Line-feeds inside the JSON should be encoded as usual. Both client and server can initiate a request on which the other side could respond with a result or an error. Below, the default ports of Stratum for Bitcoin are defined. Other coins can define other default ports. | ||
|
||
* Default port: <code>50001</code> | ||
* Letter: <code>t</code> | ||
|
||
===TCP over SSL/TLS=== | ||
|
||
Stratum over a TCP connection with SSL/TLS: the same as normal TCP, but with SSL/TLS enabled. | ||
|
||
* Default port: <code>50002</code> | ||
* Letter: <code>s</code> | ||
|
||
===HTTP=== | ||
|
||
Stratum over an HTTP connection. When communicating over HTTP, there 2 possible options: polling and pushing. In both cases, the request and response have a <code>Content-Type</code> header with the value <code>application/stratum</code>. Like in the TCP transport, the messages have to end with a line-feed (<code>\n</code>). It is possible to send multiple messages in one HTTP request or response. | ||
|
||
A request or response may contain a <code>Content-MD5</code> header to allow the receiver to detect any modification to the content. It is up to clients and servers how to handle a mismatch. A suggestion for a client would be to retry the command. A suggestion for a server would be to return a 400 error (or return a 200 error with a JSON-RPC error message). | ||
|
||
* Default port: <code>8081</code> | ||
* Letter: <code>h</code> | ||
|
||
====Session==== | ||
|
||
Because it is not sure if an HTTP connection will stay open, it is highly possible that messages will be sent over multiple connections. To keep track of the same "session", a cookie with the name <code>STRATUM_SESSION</code> is used. If a client sends a request with a cookie, the servers knows exactly which session is used and which notifications to send. | ||
|
||
Servers could send the <code>Stratum-Session-Timeout</code><ref name="extended"/> header to tell the client when a session with the same session id will be seen as a new session by the server. The header contains a number that indicates how many seconds are left. This is different from <code>Expires</code> or <code>Max-Age</code> parameters of the <code>Set-Cookie</code> header that indicate when a cookie (not necessarily a session) should have been expired by the client. For backwards compatibility, a <code>X-Session-Timeout</code> header with the same value should be sent in those requests, too. | ||
|
||
====HTTP Poll==== | ||
|
||
When polling, an HTTP POST request is sent with <code>application/stratum</code> in the <code>Content-Type</code> header. In the body, there could be one or more messages. If the body is empty, the client only wants to check for notifications. | ||
|
||
If no error occurred, the server will respond with a 200 status code and with <code>application/stratum</code> in the <code>Content-Type</code> header. If there are one or more notifications or other messages, the body will contain those messages. Otherwise, the body will be empty. | ||
|
||
====HTTP Push==== | ||
|
||
HTTP Poll is the default mode when connecting with a server over HTTP. To get the session in HTTP Push mode, a client should send a regular HTTP Poll request with an additional <code>Stratum-Callback-URL</code><ref name="extended"/> header. This will let the server send notifications using the callback URL without the client needing to send HTTP poll requests with a very small interval. To disable HTTP Push mode, a regular HTTP poll request should be sent with an additional empty <code>Stratum-Callback-URL</code><ref name="extended"/> header. For backwards compatibility, a <code>X-Callback-URL</code> header with the same value should be sent in those requests too. | ||
|
||
The callback URL will be called when there are notifications. This request is an HTTP POST request with a <code>Content-Type</code> header and a <code>Stratum-Session-ID</code><ref name="extended"/> header. The <code>Stratum-Session-ID</code><ref name="extended"/> contains the same id as the <code>STRATUM_SESSION</code> cookie. This makes it possible to use the callback URL for multiple sessions. The response of the callback URL is expected to be empty, so any response body will likely be ignored. For backwards compatibility, a <code>X-Session-ID</code> header with the same value should be sent in those requests too. | ||
|
||
===HTTP over SSL/TLS=== | ||
|
||
Stratum over an HTTP connection with SSL/TLS: the same as normal HTTP, but with SSL/TLS enabled. | ||
|
||
* Default port: <code>8082</code> | ||
* Letter: <code>g</code> | ||
|
||
===WebSocket=== | ||
|
||
Stratum over a WebSocket connection. When using WebSockets, the JSON-RPC messages (including batch messages) are encapsulated in a WebSocket message. It is also possible to send multiple JSON-RPC messages in one WebSocket message. Every JSON-RPC message should end with a line-feed (<code>\n</code>). Both client and server can initiate a message request on which the other side could respond with a result or an error, like Stratum over TCP. | ||
|
||
* Default port: <code>8083</code><ref name="extended"/> | ||
* Letter: <code>w</code><ref name="extended"/> | ||
|
||
====Subprotocol<ref name="extended"/>==== | ||
|
||
In a WebSocket upgrade request, it is possible to use the <code>Sec-WebSocket-Protocol</code> header to let the WebSocket server know which subprotocol is desired to send over the connection. For Stratum, the value <code>stratum</code> is registered. The use of this header is optional. If the server supports the use of this subprotocol too, it will let the client know by sending a <code>Sec-WebSocket-Protocol</code> header back. If the server doesn't send a subprotocol back, the connection will continue without using one. Use of this feature is fully backwards compatible. | ||
|
||
====Example==== | ||
|
||
<pre> | ||
// Open WebSocket with using a subprotocol | ||
new WebSocket('ws://stratum.example.com:8083',['stratum']); | ||
|
||
// Open WebSocket without using a subprotocol | ||
new WebSocket('ws://stratum.example.com:8083'); | ||
</pre> | ||
|
||
===WebSocket over SSL/TLS=== | ||
|
||
Stratum over a WebSocket connection with SSL/TLS: the same as normal WebSocket, but with SSL/TLS enabled. | ||
|
||
* Default port: <code>8084</code><ref name="extended"/> | ||
* Letter: <code>u</code><ref name="extended"/> | ||
|
||
==Specification: Services== | ||
|
||
On top of Stratum, some services are defined. They are listed below, grouped by vendor. | ||
|
||
===Vendor: Stratum=== | ||
|
||
* [[bip-0040/service-discovery.mediawiki|Discovery]] | ||
* [[bip-0040/service-example.mediawiki|Example]] | ||
* [[bip-0040/service-example.pubsub.mediawiki|Example Publish-Subscribe]] | ||
* [[bip-0040/service-server.mediawiki|Server]] | ||
* [[bip-0040/service-server.peers.mediawiki|Server Peers]] | ||
|
||
===Vendor: Electrum=== | ||
|
||
* [[bip-0040/service-blockchain.mediawiki|Blockchain]] | ||
* [[bip-0040/service-blockchain.address.mediawiki|Blockchain Address]] | ||
* [[bip-0040/service-blockchain.block.mediawiki|Blockchain Block]] | ||
* [[bip-0040/service-blockchain.headers.mediawiki|Blockchain Headers]] | ||
* [[bip-0040/service-blockchain.numblocks.mediawiki|Blockchain Number of Blocks]] | ||
* [[bip-0040/service-blockchain.transaction.mediawiki|Blockchain Transaction]] | ||
* [[bip-0040/service-blockchain.utxo.mediawiki|Blockchain Unspent Transaction Output]] | ||
|
||
===Vendor: blockchain.info=== | ||
|
||
* [[bip-0040/service-firstbits.mediawiki|Firstbits]] | ||
|
||
===Vendor: firstbits.com=== | ||
|
||
* [[bip-0040/service-firstbits.mediawiki|Firstbits]] | ||
|
||
==IANA Considerations== | ||
|
||
===HTTP Field=== | ||
|
||
IANA maintains the registry of HTTP Fields at https://www.iana.org/assignments/http-fields. | ||
ben221199 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
This document serves as the specification for the HTTP fields "Stratum-Session-ID", "Stratum-Session-Timeout" and "Stratum-Callback-URL". | ||
|
||
===Media Type=== | ||
|
||
IANA maintains the registry of Media Types [https://www.rfc-editor.org/info/bcp13 BCP13] at http://www.iana.org/assignments/media-types. | ||
ben221199 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
This document serves as the specification for the media type "application/stratum". It has been registered with IANA. | ||
ben221199 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
===WebSocket Subprotocol Name=== | ||
|
||
IANA maintains the registry of WebSocket Subprotocol Names at https://www.iana.org/assignments/websocket. | ||
ben221199 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
This document serves as the specification for the WebSocket subprotocol name "stratum". It has been registered with IANA. | ||
|
||
==References== | ||
<references/> |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,72 @@ | ||
=Service: Blockchain Address= | ||
|
||
* Service type: <code>blockchain.address</code> | ||
* Service vendor: <code>Electrum</code> | ||
|
||
==Method: Get History== | ||
|
||
* Method name: <code>get_history</code> | ||
|
||
{| | ||
! <ins>[Request]</ins> | ||
! Type | ||
! Example | ||
! Description | ||
|- | ||
! Method | ||
| colspan="3" | <code>blockchain.address.get_history</code> | ||
|- | ||
! Params [0] | ||
| <code>string</code> | ||
| <code>1NS17iag9jJgTHD1VXjvLCEnZuQ3rJDE9L</code> | ||
| The address to get the history of. | ||
|- | ||
! <ins>[Response]</ins> | ||
! Type | ||
! Example | ||
! Description | ||
|- | ||
! Result | ||
| <code>array</code> | ||
| | ||
| The history of the blockchain address. | ||
|} | ||
|
||
==Method: Get Balance== | ||
|
||
* Method name: <code>get_balance</code> | ||
|
||
==Method: Subscribe== | ||
|
||
* Method name: <code>subscribe</code> | ||
|
||
{| | ||
! <ins>[Request]</ins> | ||
! Type | ||
! Example | ||
! Description | ||
|- | ||
! Method | ||
| colspan="3" | <code>blockchain.address.subscribe</code> | ||
|- | ||
! Params [0] | ||
| <code>string</code> | ||
| <code>1NS17iag9jJgTHD1VXjvLCEnZuQ3rJDE9L</code> | ||
| The address to subscribe to. | ||
|} | ||
|
||
==Method: Unsubscribe== | ||
|
||
* Method name: <code>unsubscribe</code> | ||
|
||
==Method: Get Memory Pool== | ||
|
||
* Method name: <code>get_mempool</code> | ||
|
||
==Method: Get Proof== | ||
|
||
* Method name: <code>get_proof</code> | ||
|
||
==Method: List Unspent== | ||
|
||
* Method name: <code>listunspent</code> |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
=Service: Blockchain Block= | ||
|
||
* Service type: <code>blockchain.block</code> | ||
* Service vendor: <code>Electrum</code> | ||
|
||
==Method: Subscribe== | ||
|
||
* Method name: <code>subscribe</code> | ||
|
||
==Method: Unsubscribe== | ||
|
||
* Method name: <code>unsubscribe</code> | ||
|
||
==Method: Get Block Number== | ||
|
||
* Method name: <code>get_blocknum</code> | ||
|
||
==Method: Get Chunk== | ||
|
||
* Method name: <code>get_chunk</code> | ||
|
||
==Method: Get Header== | ||
|
||
* Method name: <code>get_header</code> | ||
|
||
==Method: Ping== | ||
|
||
* Method name: <code>ping</code> |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
=Service: Blockchain Headers= | ||
|
||
* Service type: <code>blockchain.headers</code> | ||
* Service vendor: <code>Electrum</code> | ||
|
||
==Method: Subscribe== | ||
|
||
* Method name: <code>subscribe</code> |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
=Service: Blockchain= | ||
|
||
* Service type: <code>blockchain</code> | ||
* Service vendor: <code>Electrum</code> | ||
|
||
==Method: Estimate Fee== | ||
|
||
* Method name: <code>estimatefee</code> | ||
|
||
==Method: Relay Fee== | ||
|
||
* Method name: <code>relayfee</code> |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
=Service: Blockchain Number of Blocks= | ||
|
||
* Service type: <code>blockchain.numblocks</code> | ||
* Service vendor: <code>Electrum</code> | ||
|
||
==Method: Subscribe== | ||
|
||
* Method name: <code>subscribe</code> |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
=Service: Blockchain Transaction= | ||
|
||
* Service type: <code>blockchain.transaction</code> | ||
* Service vendor: <code>Electrum</code> | ||
|
||
==Method: Subscribe== | ||
|
||
* Method name: <code>subscribe</code> | ||
|
||
==Method: Unsubscribe== | ||
|
||
* Method name: <code>unsubscribe</code> | ||
|
||
==Method: Guess Fee== | ||
|
||
* Method name: <code>guess_fee</code> | ||
|
||
==Method: Broadcast== | ||
|
||
* Method name: <code>broadcast</code> | ||
|
||
==Method: Get== | ||
|
||
* Method name: <code>get</code> | ||
|
||
==Method: Get Merkle== | ||
|
||
* Method name: <code>get_merkle</code> |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
=Service: Blockchain Unspent Transaction Output= | ||
|
||
* Service type: <code>blockchain.utxo</code> | ||
* Service vendor: <code>Electrum</code> | ||
|
||
==Method: Get Address== | ||
|
||
* Method name: <code>get_address</code> |
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Uh oh!
There was an error while loading. Please reload this page.