Warp D.R.E.
A Delegated Resolution Environment for Warp Contracts.
D.R.E. nodes are a special kind of nodes that are responsible for evaluating and serving contracts' states -
in a decentralized manner - taking the burden of the state evaluation from the end users.
The reasoning
The purpose of implementing Warp D.R.E. is to address the following issues.
Evaluation of high-interaction contracts - evaluating contracts with thousands of interactions is a hassle for the user's CPU. Response-time and UX can be improved significantly when computation is delegated to D.R.E.
Interaction with high-risk contracts - some contracts may perform risky/unsafe operations. Contract interaction via Warp D.R.E. ensures the safety of users' devices.
Insights into PST tokens' - Providing aggregate information about PST tokens' status and count has always been challenging. With D.R.E.'s aggregation tool, one can check global address data with just a few clicks.
Avoid centralised, closed-sourced solutions - private, unknown processing logic, and tightly coupled to a specific cloud vendor (such as GCP or AWS) belong to the web2 era. Warp D.R.E. is built on the principle of "Don't trust, verify". Once initial testing is complete, we will open up D.R.E nodes for public participation.
Nodes
Currently, there are two nodes available
- The East Side - https://dre-1.warp.cc
- The West Side - https://dre-2.warp.cc
To verify if your contract is available - check the /contract
endpoint,
eg: https://dre-1.warp.cc/contract?id=-8A6RexFkpfWwuyVO98wzSFZh0d6VJuI-buTJvlwOJQ&events=true
How it works?
1. Messages processing
- The Warp Gateway and Warp Sequencer are publishing messages to a pub/sub (currently Redis, Streamr in the
future) - to a
contracts
channel. - Two kind of messages are being sent:
- contract deployment notification (whenever a new contract is being deployed, either via Warp or directly to
Arweave L1).
Message format:
{
"contractTxId": "<contract_txId>",
"test": <true|false>,
"source": "warp-gw"
"initialState": <contract_initial_state>
} - contract interaction notification (whenever a new interaction is being registered, either via Warp Sequencer
or directly to Arweave L1).
Message format:{
"contractTxId": "<contract_txId>",
"test": <true|false>,
"source": "warp-gw"
"interaction": <new_interaction>
}
- contract deployment notification (whenever a new contract is being deployed, either via Warp or directly to
Arweave L1).
Message format:
- D.R.E. is subscribing for messages on the
contracts
channel. - D.R.E. maintains two internal queues. The queues are implemented
with BullMQ.
- register queue for registering new contracts (for messages with an initial state)
- update queue for processing contracts' interactions (for messages with a new interaction)
- Each of the queues have its own set of workers. Each worker runs as a separate, sandboxed processor that is isolated from the rest of the code.
- Each message that comes to D.R.E. is first validated (message format, tx id format, etc.).
- If it is a
contract deployment
notification AND contract is not yet registered - a new job is added to the register queue.
The processor that picks up such job simply sets theinitialState
from the incoming message in the D.R.E. cache. - If it is a
contract update
notification AND contract is not yet registered - a new job is added to the register queue.
The processor that picks up such job evaluates the contract's state from scratch. - If it is a
contract update
notification AND contract is already registered - a new job is added to the update queue. The processor that picks up such job either- evaluates the state only for the interaction from the message - if D.R.E. has a contract cached
at
message.interaction.lastSortKey
- evaluates the state for all the interactions from the lastly cached - if D.R.E has a contract cached at a lower
sort key than
message.interaction.lastSortKey
.
- evaluates the state only for the interaction from the message - if D.R.E. has a contract cached
at
2. Caching
D.R.E. is currently using two kinds of caches
- Warp Contracts SDK's internal cache (with the LMDB plugin)
- better-sqlite3 based cache. This cache is used for
serving data via D.R.E. endpoints - in order not to interfere
with the Warp Contracts SDK internal cache.
It also serves as a from of a backup.
WAL mode is being used for increased performance.
The evaluated state (apart from being automatically cached by the Warp Contracts SDK) is additionally:
Signed by the D.R.E.'s wallet. The data for the signature consists of:
// state is stringified with 'safe-stable-stringify', not JSON.stringify.
// JSON.stringify is non-deterministic.
const stringifiedState = stringify(state);
const hash = crypto.createHash('sha256');
hash.update(stringifiedState);
const stateHash = hash.digest('hex');
const dataToSign = await deepHash([
arweave.utils.stringToBuffer(owner), // a jwk.n of the D.R.E's wallet
arweave.utils.stringToBuffer(sortKey), // a sort key at which the state has been evaluated
arweave.utils.stringToBuffer(contractTxId), // what could it be....
arweave.utils.stringToBuffer(stateHash), // a state hash
arweave.utils.stringToBuffer(stringify(manifest)) // a full node's manifest
]);
The manifest
contains all the data that was used for the evaluation of the state, including
- the libraries version (warp-contracts and all warp plugins)
- the evaluation options
- git commit hash at which the node was running during state evaluation.
A manifest may look like this:
{
"gitCommitHash": "19a327e141300772985ef1b9e44c01c17de4f668",
"warpSdkConfig": {
"warp-contracts": "1.2.23",
"warp-contracts-lmdb": "1.1.1",
"warp-contracts-nlp-plugin": "1.0.5"
},
"evaluationOptions": {
"useVM2": true,
"maxCallDepth": 5,
"maxInteractionEvaluationTimeSeconds": 10,
"allowBigInt": true,
"internalWrites": true
},
"owner": "<very_long_string>",
"walletAddress": "uOImfFuq_KVTHZfsKdC-3WriZwtInyEkxe2MlU_le9Q"
}
- After signing - the state is stored in the
better-sqlite3
(including the validity, error messages, state hash, node's manifest at the time of evaluation, signature).
NOTE: This data allows to recreate the exact environment that was used to evaluate the state (and verify it locally). - As a last step - a new state is being published on pub/sub
states
channel. The messages on this channel are being listened by the Warp Aggregate Node, which combines the data from all the D.R.E.s.
3. Events
To give a better understanding of that is going on, the D.R.E. registers events at certain points of processing. Each event consists of:
- event type
- timestamp
- optional message
The event type is one of:
REQUEST_REGISTER
- a registration request has been received for a contractREQUEST_UPDATE
- an update request has been received for a contractREJECT
- an update or registration request has been rejected either because of incoming message validation errors or because contract is blacklisted.FAILURE
- an error have occurred during contract evaluationEVALUATED
- contract has been successfully evaluated (but it required loading interactions from Warp GW)PROGRESS
- added after evaluating each 500 interactions (useful for tracking the evaluation progress of a registered contract that has thousands of interactions)UPDATED
- contract has been successfully updated using the interaction sent in the message
4. Blacklisting
Whenever a contract reaches a certain amount of failures (3 for both dre-1 and dre-2) - it is blacklisted and ignored.
5. Endpoints
/
,/status
- contains information about node manifest, workers configuration, queues status, etc./contract
- returns all data about a given contract. Parameters:id
- the contract tx idstate
-true|false
- whether state should be returned.true
by defaultvalidity
-true|false
- whether validity should be returned.false
by defaulterrorMessages
-true|false
- whether error messages should be returned.false
by defaulterrors
- whether errors thrown during the evaluation process should be returned (e.g. the errors thrown by Warp Gateway, Arweave gateway).false
by defaultevents
true|false
- whether events for this contract should be returned..false
by defaultquery
- a jsonpath-plus expression that allows to query the current state (for example query for the balance of a concrete wallet address). Whenever aquery
parameter is passed, aresult
is returned instead ofstate
.
Example query:
https://dre-1.warp.cc/contract?id=5Yt1IujBmOm1LSux9KDUTjCE7rJqepzP7gZKf_DyzWI&query=$.balances.7FljANNIG0FfumNxShdt3ELtL33HMoxnoHHct2TQdvE
/cached
- returns a list of all cached contracts/blacklist
- returns all contracts that failed at least once. If contract failed 3 times - it is blacklisted./errors
- returns all errors for all contracts/sync
- schedule force synchronization of a given contract. This endpoint can't be called more frequent then every 10 seconds. Parameters:id
- text contract tx id
Example:
http ":8080/sync?id=KT45jaf8n9UwgkEareWxPgLJk4oMWpI5NODgYVIF1fY"
HTTP/1.1 200 OK
Scheduled for updateError when too frequent requests per one contract:
http ":8080/sync?id=KT45jaf8n9UwgkEareWxPgLJk4oMWpI5NODgYVIF1fY"
HTTP/1.1 500 Internal Server Error
Chill out and wait 10s
Future work
- Sync the local state with D.R.E. inside the Warp Contract SDK while connecting to a contract
(i.e. while calling
warp.contract(<contract_tx_id)
). - Replace Redis with Streamr - in progress
- More nodes
- One-command deployment via Docker
unsafeClient
compatible nodes- Bundle all the evaluated states
- A decentralized validation layer built on of the D.R.E. nodes
- Stats/dashboard