-
Notifications
You must be signed in to change notification settings - Fork 1.7k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
1 changed file
with
350 additions
and
0 deletions.
There are no files selected for viewing
This file contains 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,350 @@ | ||
# Mercury Documentation | ||
|
||
## Contracts | ||
|
||
Use this tool to configure contracts: | ||
|
||
https://github.com/smartcontractkit/the-most-amazing-mercury-contract-configuration-tool | ||
|
||
TODO: updated process here @Austin Born | ||
|
||
[Reference contract](https://github.com/smartcontractkit/reference-data-directory/blob/master/ethereum-testnet-goerli-arbitrum-1/contracts/0x535051166466D159da8742167c9CA1eFe9e82613.json) | ||
|
||
[OCR2 config documentation](https://github.com/smartcontractkit/libocr/blob/master/offchainreporting2/internal/config/public_config.go) | ||
|
||
**🚨 Important config** | ||
|
||
`s` - transmission schedule. This should be set to the number of oracles on the feed - meaning that every oracle will attempt to transmit to the mercury server in the first stage of transmission. eg `[4]` if there are 4 node in the DON, excluding the bootstrap node. | ||
|
||
`f` - set this to `n//3` (where `//` denotes integer division), e.g. if you have 16 oracles, set `f` to 5. | ||
|
||
`deltaRound` - report generation frequency. This determines how frequently a new round should be started at most (if rounds take longer than this due to network latency, there will be fewer rounds per second than this parameter would suggest). `100ms` is a good starting point (10 rounds/s). | ||
|
||
`reportingPluginConfig.alphaAccept` - set this to `0`, because our mercury ContractTransmitter doesn't know the latest report that's been sent to the mercury server and we therefore always have a "pending" report which we compare against before accepting a report for transmission. | ||
|
||
`reportingPluginConfig.deltaC` - set this to `0` so every round will result in a report. | ||
|
||
<details><summary>Example `verifier/<0xaddress>.json`</summary> | ||
|
||
```json | ||
{ | ||
"contractVersion": 1001, | ||
"digests": { | ||
"0x0006c67c0374ab0dcfa45c63b37df2ea8d16fb903c043caa98065033e9c15666": { | ||
"feedId": "0x14e044f932bb959cc2aa8dc1ba110c09224e639aae00264c1ffc2a0830904a3c", | ||
"proxyEnabled": true, | ||
"status": "active" | ||
} | ||
}, | ||
"feeds": { | ||
"0x14e044f932bb959cc2aa8dc1ba110c09224e639aae00264c1ffc2a0830904a3c": { | ||
"digests": [ | ||
"0x0006c67c0374ab0dcfa45c63b37df2ea8d16fb903c043caa98065033e9c15666" | ||
], | ||
"docs": { | ||
"assetName": "Chainlink", | ||
"feedCategory": "verified", | ||
"feedType": "Crypto", | ||
"hidden": true | ||
}, | ||
"externalAdapterRequestParams": { | ||
"endpoint": "cryptolwba", | ||
"from": "LINK", | ||
"to": "USD" | ||
}, | ||
"feedId": "0x14e044f932bb959cc2aa8dc1ba110c09224e639aae00264c1ffc2a0830904a3c", | ||
"latestConfig": { | ||
"offchainConfig": { | ||
"deltaGrace": "0", | ||
"deltaProgress": "2s", | ||
"deltaResend": "20s", | ||
"deltaRound": "250ms", | ||
"deltaStage": "60s", | ||
"f": 1, | ||
"maxDurationObservation": "250ms", | ||
"maxDurationQuery": "0s", | ||
"maxDurationReport": "250ms", | ||
"maxDurationShouldAcceptFinalizedReport": "250ms", | ||
"maxDurationShouldTransmitAcceptedReport": "250ms", | ||
"rMax": 100, | ||
"reportingPluginConfig": { | ||
"alphaAcceptInfinite": false, | ||
"alphaAcceptPpb": "0", | ||
"alphaReportInfinite": false, | ||
"alphaReportPpb": "0", | ||
"deltaC": "0s" | ||
}, | ||
"s": [ | ||
4 | ||
] | ||
}, | ||
"offchainConfigVersion": 30, | ||
"onchainConfig": { | ||
"max": "99999999999999999999999999999", | ||
"min": "1" | ||
}, | ||
"onchainConfigVersion": 1, | ||
"oracles": [ | ||
{ | ||
"api": [ | ||
"coinmetrics", | ||
"ncfx", | ||
"tiingo-test" | ||
], | ||
"operator": "clc-ocr-mercury-arbitrum-goerli-nodes-0" | ||
}, | ||
{ | ||
"api": [ | ||
"coinmetrics", | ||
"ncfx", | ||
"tiingo-test" | ||
], | ||
"operator": "clc-ocr-mercury-arbitrum-goerli-nodes-1" | ||
}, | ||
{ | ||
"api": [ | ||
"coinmetrics", | ||
"ncfx", | ||
"tiingo-test" | ||
], | ||
"operator": "clc-ocr-mercury-arbitrum-goerli-nodes-2" | ||
}, | ||
{ | ||
"api": [ | ||
"coinmetrics", | ||
"ncfx", | ||
"tiingo-test" | ||
], | ||
"operator": "clc-ocr-mercury-arbitrum-goerli-nodes-3" | ||
} | ||
] | ||
}, | ||
"marketing": { | ||
"category": "crypto", | ||
"history": true, | ||
"pair": [ | ||
"LINK", | ||
"USD" | ||
], | ||
"path": "link-usd-verifier" | ||
}, | ||
"name": "LINK/USD-RefPricePlus-ArbitrumGoerli-002", | ||
"reportFields": { | ||
"ask": { | ||
"decimals": 8, | ||
"maxSubmissionValue": "99999999999999999999999999999", | ||
"minSubmissionValue": "1", | ||
"resultPath": "data,ask" | ||
}, | ||
"bid": { | ||
"decimals": 8, | ||
"maxSubmissionValue": "99999999999999999999999999999", | ||
"minSubmissionValue": "1", | ||
"resultPath": "data,bid" | ||
}, | ||
"median": { | ||
"decimals": 8, | ||
"maxSubmissionValue": "99999999999999999999999999999", | ||
"minSubmissionValue": "1", | ||
"resultPath": "data,mid" | ||
} | ||
}, | ||
"status": "testing" | ||
} | ||
}, | ||
"name": "Mercury v0.2 - Production Testnet Verifier (v1.0.0)", | ||
"status": "testing" | ||
} | ||
``` | ||
</details> | ||
|
||
## Jobs | ||
|
||
### Bootstrap | ||
|
||
**🚨 Important config** | ||
|
||
`relayConfig.chainID` - target chain id. (the chain we pull block numbers from) | ||
|
||
`contractID` - the contract address of the verifier contract. | ||
|
||
<details><summary>Example bootstrap TOML</summary> | ||
|
||
```toml | ||
type = "bootstrap" | ||
relay = "evm" | ||
schemaVersion = 1 | ||
name = "$feed_name" | ||
contractID = "$verifier_contract_address" | ||
feedID = "$feed_id" # IMPORTANT - DON'T FORGET THIS OR IT WON'T WORK | ||
contractConfigTrackerPollInterval = "15s" | ||
|
||
[relayConfig] | ||
chainID = $evm_chain_id | ||
fromBlock = $from_block | ||
``` | ||
</details> | ||
|
||
### OCR2 | ||
|
||
<details><summary>Example OCR2 Mercury TOML</summary> | ||
|
||
```toml | ||
type = "offchainreporting2" | ||
schemaVersion = 1 | ||
name = "$feed_name" | ||
forwardingAllowed = false | ||
maxTaskDuration = "1s" | ||
contractID = "$verifier_contract_address" | ||
feedID = "$feed_id" | ||
contractConfigTrackerPollInterval = "15s" | ||
ocrKeyBundleID = "$key_bundle_id" | ||
p2pv2Bootstrappers = [ | ||
"$bootstrapper_address>" | ||
] | ||
relay = "evm" | ||
pluginType = "mercury" | ||
transmitterID = "$csa_public_key" | ||
|
||
observationSource = """ | ||
// ncfx | ||
ds1_payload [type=bridge name="ncfx" timeout="50ms" requestData="{\\"data\\":{\\"endpoint\\":\\"crypto-lwba\\",\\"from\\":\\"ETH\\",\\"to\\":\\"USD\\"}}"]; | ||
ds1_median [type=jsonparse path="data,mid"]; | ||
ds1_bid [type=jsonparse path="data,bid"]; | ||
ds1_ask [type=jsonparse path="data,ask"]; | ||
ds1_median_multiply [type=multiply times=100000000]; | ||
ds1_bid_multiply [type=multiply times=100000000]; | ||
ds1_ask_multiply [type=multiply times=100000000]; | ||
// tiingo | ||
ds2_payload [type=bridge name="tiingo" timeout="50ms" requestData="{\\"data\\":{\\"endpoint\\":\\"crypto-lwba\\",\\"from\\":\\"ETH\\",\\"to\\":\\"USD\\"}}"]; | ||
ds2_median [type=jsonparse path="data,mid"]; | ||
ds2_bid [type=jsonparse path="data,bid"]; | ||
ds2_ask [type=jsonparse path="data,ask"]; | ||
ds2_median_multiply [type=multiply times=100000000]; | ||
ds2_bid_multiply [type=multiply times=100000000]; | ||
ds2_ask_multiply [type=multiply times=100000000]; | ||
// coinmetrics | ||
ds3_payload [type=bridge name="coinmetrics" timeout="50ms" requestData="{\\"data\\":{\\"endpoint\\":\\"crypto-lwba\\",\\"from\\":\\"ETH\\",\\"to\\":\\"USD\\"}}"]; | ||
ds3_median [type=jsonparse path="data,mid"]; | ||
ds3_bid [type=jsonparse path="data,bid"]; | ||
ds3_ask [type=jsonparse path="data,ask"]; | ||
ds3_median_multiply [type=multiply times=100000000]; | ||
ds3_bid_multiply [type=multiply times=100000000]; | ||
ds3_ask_multiply [type=multiply times=100000000]; | ||
ds1_payload -> ds1_median -> ds1_median_multiply -> benchmark_price; | ||
ds2_payload -> ds2_median -> ds2_median_multiply -> benchmark_price; | ||
ds3_payload -> ds3_median -> ds3_median_multiply -> benchmark_price; | ||
benchmark_price [type=median allowedFaults=2 index=0]; | ||
ds1_payload -> ds1_bid -> ds1_bid_multiply -> bid_price; | ||
ds2_payload -> ds2_bid -> ds2_bid_multiply -> bid_price; | ||
ds3_payload -> ds3_bid -> ds3_bid_multiply -> bid_price; | ||
bid_price [type=median allowedFaults=2 index=1]; | ||
ds1_payload -> ds1_ask -> ds1_ask_multiply -> ask_price; | ||
ds2_payload -> ds2_ask -> ds2_ask_multiply -> ask_price; | ||
ds3_payload -> ds3_ask -> ds3_ask_multiply -> ask_price; | ||
ask_price [type=median allowedFaults=2 index=2]; | ||
""" | ||
|
||
[pluginConfig] | ||
serverURL = "$mercury_server_url" | ||
serverPubKey = "$mercury_server_public_key" | ||
|
||
[relayConfig] | ||
chainID = $evm_chain_id | ||
fromBlock = $from_block | ||
``` | ||
</details> | ||
|
||
## Nodes | ||
|
||
**🚨 Important config** | ||
|
||
`OCR2.Enabled` - must be `true` - Mercury uses OCR2. | ||
|
||
`P2P.V2.Enabled` - required in order for OCR2 to work. | ||
|
||
`Feature.LogPoller` - required in order for OCR2 to work. You will get fatal errors if not set. | ||
|
||
`JobPipeline.MaxSuccessfulRuns` - set to `0` to disable saving pipeline runs to reduce load on the db. Obviously this means you won’t see anything in the UI. | ||
|
||
`TelemetryIngress.SendInterval` - How frequently to send telemetry batches. Mercury generates a lot of telemetry data due to the throughput. `100ms` has been tested for a single feed with 5 nodes - this will need to be monitored (along with relevant config) as we add more feeds to a node. | ||
|
||
`Database` - **must** increase connection limits above the standard defaults | ||
|
||
<details><summary>Example node config TOML</summary> | ||
|
||
```toml | ||
RootDir = '$ROOT_DIR' | ||
|
||
[JobPipeline] | ||
MaxSuccessfulRuns = 0 # you may set to some small value like '10' or similar if you like looking at job runs in the UI | ||
|
||
[Feature] | ||
UICSAKeys = true # required | ||
LogPoller = true # required | ||
|
||
[Log] | ||
Level = 'info' # this should be 'debug' for chainlink internal deployments, nops may use 'info' to reduce log volume | ||
|
||
[Log.File] | ||
< standard values > | ||
|
||
[WebServer] | ||
< standard values > | ||
|
||
[WebServer.TLS] | ||
< standard values > | ||
|
||
[[EVM]] | ||
ChainID = '42161' # change as needed based on target chain | ||
|
||
[OCR] | ||
Enabled = false # turn off OCR 1 | ||
|
||
[P2P] | ||
TraceLogging = false # this should be 'true' for chainlink internal deployments, we may ask nops to set this to true for debugging | ||
PeerID = '$PEERID' | ||
|
||
[P2P.V2] | ||
Enabled = true # required | ||
DefaultBootstrappers = < mercury bootstrap nodes > # Note that this should ideally be set in the job spec, this is just a fallback | ||
# Make sure these IPs are properly configured in the firewall. May not be necessary for internal nodes | ||
AnnounceAddresses = ['$EXTERNAL_IP:$EXTERNAL_PORT'] # Use whichever port you like, pls randomize, MAKE SURE ITS CONFIGURED IN THE FIREWALL | ||
ListenAddresses = ['0.0.0.0:$INTERNAL_PORT'] # Use whichever port you like, pls randomize, MAKE SURE ITS CONFIGURED IN THE FIREWALL | ||
|
||
[OCR2] | ||
Enabled = true # required | ||
KeyBundleID = '$KEY_BUNDLE_ID' # Note that this should ideally be set in the job spec, this is just a fallback | ||
CaptureEATelemetry = true | ||
|
||
[TelemetryIngress] | ||
UniConn = false | ||
SendInterval = '250ms' | ||
BufferSize = 300 | ||
MaxBatchSize = 100 | ||
|
||
[[TelemetryIngress.Endpoints]] | ||
Network = 'EVM' | ||
ChainID = '42161' # change as needed based on target chain | ||
URL = '$TELEMETRY_ENDPOINT_URL' # Provided by Chainlink Labs RSTP team | ||
ServerPubKey = '$TELEMETRY_PUB_KEY' # Provided by Chainlink Labs RSTP team | ||
|
||
[Database] | ||
MaxIdleConns = 100 # should equal or greater than total number of mercury jobs | ||
MaxOpenConns = 400 # caution! ensure postgres is configured to support this | ||
|
||
[[EVM.Nodes]] | ||
< put RPC nodes here > | ||
``` | ||
</details> |