This document focuses on the Node.js implementation of the Sidetree protocol.
There will exist several Sidetree node configurations, which offer a variety of modes that support different features and trade-offs. The choice to run one type or another largely depends on the type of user, machine, and intent the operator has in mind.
A full node offers the largest set of features and highest resolution performance of DIDs, but also requires more significant bandwidth, hardware, storage, and system resource consumption to operate. A full node will attempt to fetch and retain all data associated with the Sidetree operations present in the target system. As such, full nodes are able to quickly resolve DID lookup requests and may feature more aggressive caching of DID state than other node configurations.
A light node is a node that retains the ability to independently resolve DIDs without relying on a trusted party or trusted assertions by other nodes, while minimizing the amount of bandwidth and data required to do so. Light nodes run a copy of the target system's node (e.g. a blockchain) and fetch all minimal anchor file data required to create an independent mapping that enables just-in-time resolution of DIDs.
NOTE: Light node support is in development, with release of a supporting node implementation in May 2019.
The Operation Processor holds most of the state of a Sidetree node. It is a singleton class with the following methods for DID Document state update and retrieval.
This is the core method to update the state of a DID Document:
public process (transactionNumber: number, operationIndex: number, operation: Operation)The operation is a JSON object representing a create, update, or a delete operation. Recall from the protocol description that the hash of this object is the operation hash, which represents the version of the document produced as the result of this operation.
The transactionNumber and operationIndex parameters together provides a deterministic ordering of all operations. The transactionNumber is a monotonically increasing number (need NOT be by 1) that identifies a Sidetree transaction. The operationIndex is the index of this operation amongst all the operations batched within the same Sidetree transaction.
Note:
transactionNumberandoperationIndexare explicitly called out as parameters for theprocessmethod for clarity. They may be embedded within theoperationparameter in actual implementation.
Under normal processing, the observer would process operations in chronological order. However the implementation accepts process calls with out-of-ordered operations. This is used to handle delays and out-of-orderedness introduced by the CAS layer.
It is useful to view the operations as producing a collection of version chains one per DID. Each create operation introduces a new chain and each update operation adds an edge to an existing chain. There could be holes in the chain if some historical update is missing - as noted above, this could be caused due to CAS delays.
When two update operations reference the same (prior) version of a DID Document, the cache removes from consideration the later of the two operations and all operations directly and indirectly referencing the removed operation. This ensures that the document versions of a particular DID form a chain without any forks. For illustration, assume we have recorded four operations for a particular DID producing the following chain:
v0 -> v1 -> v2 -> v3
If we find an earlier update operation v0 -> v4, the new chain for the DID would be:
v0 -> v4
In the above description, earlier and later refer to the logical time of the operation derived from the position of the operation in the blockchain.
This method is used to handle rollbacks (forks) in the blockchain.
public rollback (transactionNumber: number)The effect of this method is to delete the effects of any operation included in a transaction with a transaction number greater than or equal to the transactionNumber provided.
The resolve method returns the latest document for a given DID.
The Batch Writer batches pending (Create, Update, Delete and Recover) operations and anchors them on the blockchain at a periodic interval.
The batching interval can specified by the batchingIntervalInSeconds configuration parameter.
The Observer watches the public blockchain to identify Sidetree operations, then parses the operations into data structures that can be used for efficient DID resolutions. The primary goals for the Observer are to:
- Maximize ingestion processing rate.
- Allow horizontal scaling for high DID resolution throughput.
- Allow sharing of the processed data structure by multiple Sidetree nodes to minimize redundant computation.
The above goals lead to a design where minimal processing of the operations at the time of ingestion and defers the heavy processing such as signature validation and JSON patch to the time of DID resolution.
The blockchain REST API interface aims to abstract the underlying blockchain away from the main protocol logic. This allows the underlying blockchain to be replaced without affecting the core protocol logic. The interface also allows the protocol logic to be implemented in an entirely different language while interfacing with the same blockchain.
| HTTP status code | Description |
|---|---|
| 200 | Everything went well. |
| 400 | Bad client request. |
| 401 | Unauthenticated or unauthorized request. |
| 404 | Resource not found. |
| 500 | Server error. |
Gets the latest logical blockchain time. This API allows the Observer and Batch Writer to determine protocol version to be used.
A blockchain time hash must not be predictable/pre-computable, a canonical implementation would be to use the block number as the time and the block hash as the time hash. It is intentional that the concepts related to blockchain blocks are hidden from the layers above.
| Minimum API version | v1.0 |
GET /<api-version>/time
None.
None.
Get /v1.0/time
{
"time": "The logical blockchain time.",
"hash": "The hash associated with the blockchain time."
}{
"time": 545236,
"hash": "0000000000000000002443210198839565f8d40a6b897beac8669cf7ba629051"
}Gets the time identified by the time hash.
| Minimum API version | v1.0 |
GET /<api-version>/time/<time-hash>
None.
None.
Get /v1.0/time/0000000000000000001bfd6c48a6c3e81902cac688e12c2d87ca3aca50e03fb5
{
"time": "The logical blockchain time.",
"hash": "The hash associated with the blockchain time, must be the same as the value given in query path."
}{
"time": 545236,
"hash": "0000000000000000002443210198839565f8d40a6b897beac8669cf7ba629051"
}Fetches Sidetree transactions in chronological order.
Note: The call may not to return all Sidetree transactions in one batch, in which case the caller can use the transaction number of the last transaction in the returned batch to fetch subsequent transactions.
| Minimum API version | v1.0 |
GET /<api-version>/transactions?since=<transaction-number>&transaction-time-hash=<transaction-time-hash>
None.
-
sinceOptional. A transaction number. When not given, all Sidetree transactions since inception will be returned. When given, only Sidetree transactions after the specified transaction will be returned.
-
transaction-time-hashOptional, but MUST BE given if
sinceparameter is specified.This is the hash associated with the time the transaction specified by the
sinceparameter is anchored on blockchain. Multiple transactions can have the same transaction time and thus the same transaction time hash.The transaction time hash helps the blockchain layer detect block reorganizations (temporary forks);
HTTP 400 Bad Requestwithinvalid_transaction_number_or_time_hashas thecodeparameter value in a JSON body is returned on such events.
GET /v1.0/transactions?since=170&transaction-time-hash=00000000000000000000100158f474719e5a319933856f7f464fcc65a3cb2253
{
"moreTransactions": "True if there are more transactions beyond the returned batch. False otherwise.",
"transactions": [
{
"transactionNumber": "A monotonically increasing number (need NOT be by 1) that identifies a Sidtree transaction.",
"transactionTime": "The logical blockchain time this transaction is anchored. Used for protocol version selection.",
"transactionTimeHash": "The hash associated with the transaction time.",
"anchorFileHash": "Hash of the anchor file of this transaction."
},
...
]
}HTTP/1.1 200 OK
{
"moreTransactions": false,
"transactions": [
{
"transactionNumber": 89,
"transactionTime": 545236,
"transactionTimeHash": "0000000000000000002352597f8ec45c56ad19994808e982f5868c5ff6cfef2e",
"anchorFileHash": "QmWd5PH6vyRH5kMdzZRPBnf952dbR4av3Bd7B2wBqMaAcf"
},
{
"transactionNumber": 100,
"transactionTime": 545236,
"transactionTimeHash": "00000000000000000000100158f474719e5a319933856f7f464fcc65a3cb2253",
"anchorFileHash": "QmbJGU4wNti6vNMGMosXaHbeMHGu9PkAUZtVBb2s2Vyq5d"
}
]
}HTTP/1.1 400 Bad Request
{
"code": "invalid_transaction_number_or_time_hash"
}Given a list of Sidetree transactions, returns the first transaction in the list that is valid. Returns 404 NOT FOUND if none of the given transactions are valid. This API is primarily used by the Sidetree core library to determine a transaction that can be used as a marker in time to reprocess transactions in the event of a block reorganization (temporary fork).
| Minimum API version | v1.0 |
POST /<api-version>/transactions/firstValid HTTP/1.1| Name | Value |
|---|---|
Content-Type |
application/json |
{
"transactions": [
{
"transactionNumber": "The transaction to be validated.",
"transactionTime": "The logical blockchain time this transaction is anchored. Used for protocol version selection.",
"transactionTimeHash": "The hash associated with the transaction time.",
"anchorFileHash": "Hash of the anchor file of this transaction."
},
...
]
}POST /v1.0/transactions/firstValid HTTP/1.1
Content-Type: application/json
{
"transactions": [
{
"transactionNumber": 19,
"transactionTime": 545236,
"transactionTimeHash": "0000000000000000002352597f8ec45c56ad19994808e982f5868c5ff6cfef2e",
"anchorFileHash": "Qm28BKV9iiM1ZNzMsi3HbDRHDPK5U2DEhKpCYhKk83UPEg"
},
{
"transactionNumber": 18,
"transactionTime": 545236,
"transactionTimeHash": "0000000000000000000054f9719ef6ca646e2503a9c5caac1c6ea95ffb4af587",
"anchorFileHash": "Qmb2wxUwvEpspKXU4QNxwYQLGS2gfsAuAE9LPcn5LprS1nb"
},
{
"transactionNumber": 16,
"transactionTime": 545200,
"transactionTimeHash": "0000000000000000000f32c84291a3305ad9e5e162d8cc363420831ecd0e2800",
"anchorFileHash": "QmbBPdjWSdJoQGHbZDvPqHxWqqeKUdzBwMTMjJGeWyUkEzK"
},
{
"transactionNumber": 12,
"transactionTime": 545003,
"transactionTimeHash": "0000000000000000001e002080595267fe034d370897b7b506d119ad29da1541",
"anchorFileHash": "Qmss3gKdm9uU9YLx3MPRHQTcUq1CR1Xv9Zpdu7EBG9Pk9Y"
},
{
"transactionNumber": 4,
"transactionTime": 544939,
"transactionTimeHash": "00000000000000000000100158f474719e5a319933856f7f464fcc65a3cb2253",
"anchorFileHash": "QmdcDrVPWy3ZXoZcuvFq7fDVqatks22MMqPAxDqXsZzGhy"
}
]
}{
"transactionNumber": "The transaction number of the first valid transaction in the given list",
"transactionTime": "The logical blockchain time this transaction is anchored. Used for protocol version selection.",
"transactionTimeHash": "The hash associated with the transaction time.",
"anchorFileHash": "Hash of the anchor file of this transaction."
}HTTP/1.1 200 OK
{
"transactionNumber": 16,
"transactionTime": 545200,
"transactionTimeHash": "0000000000000000000f32c84291a3305ad9e5e162d8cc363420831ecd0e2800",
"anchorFileHash": "QmbBPdjWSdJoQGHbZDvPqHxWqqeKUdzBwMTMjJGeWyUkEzK"
}HTTP/1.1 404 NOT FOUNDWrites a Sidetree transaction to the underlying blockchain.
| Minimum API version | v1.0 |
POST /<api-version>/transactions
| Name | Value |
|---|---|
Content-Type |
application/json |
{
"anchorFileHash": "The hash of a Sidetree anchor file."
}POST /v1.0/transactions HTTP/1.1
{
"anchorFileHash": "QmbJGU4wNti6vNMGMosXaHbeMHGu9PkAUZtVBb2s2Vyq5d"
}None.
The CAS (content addressable storage) REST API interface aims to abstract the underlying Sidetree storage away from the main protocol logic. This allows the CAS to be updated or even replaced if needed without affecting the core protocol logic. Conversely, the interface also allows the protocol logic to be implemented in an entirely different language while interfacing with the same CAS.
All hashes used in the API are encoded multihash as specified by the Sidetree protocol.
| HTTP status code | Description |
|---|---|
| 200 | Everything went well. |
| 400 | Bad client request. |
| 401 | Unauthenticated or unauthorized request. |
| 404 | Resource not found. |
| 500 | Server error. |
Read the content of a given address and return it in the response body as octet-stream.
| Minimum API version | v1.0 |
GET /<api-version>/<hash>?max-size=<maximum-allowed-size>
-
max-sizeRequired.
If the content exceeds the specified maximum allowed size,
HTTP 400 Bad Requestwithcontent_exceeds_maximum_allowed_sizeas the value for thecodeparameter in a JSON body is returned.
GET /v1.0/QmWd5PH6vyRH5kMdzZRPBnf952dbR4av3Bd7B2wBqMaAcf
| Name | Value |
|---|---|
Content-Type |
application/octet-stream |
HTTP/1.1 404 Not FoundHTTP/1.1 400 Bad Request
{
"code": "content_exceeds_maximum_allowed_size"
}Write content to CAS.
| Minimum API version | v1.0 |
POST /<api-version>/
| Name | Value |
|---|---|
Content-Type |
application/octet-stream |
| Name | Value |
|---|---|
Content-Type |
application/json |
{
"hash": "Hash of data written to CAS"
}{
"hash": "QmWd5PH6vyRH5kMdzZRPBnf952dbR4av3Bd7B2wBqMaAcf"
}