Download Zcash Sapling parameters and load them from cached files (#3057)
* Replace Zcash parameters crates with pre-downloaded local parameter files * Download Zcash parameters using the `zcashd` script in CI and Docker * Add a zcash_proofs dependency to zebra-consensus * Download Sapling parameters using zcash_proofs, rather than fetch-params.sh * Add a new `zebrad download` subcommand This command isn't required for nomrmal usage. But it's useful when testing, or launching multiple Zebra instances. * Use `zebrad download` in CI to pre-download parameters * Log a helpful hint if downloading fails * Allow some duplicate dependencies currently hidden by orchard * Spawn a separate task to download Groth16 parameters * Run the parameter download with code coverage This avoids re-compining Zebra with and without coverage. * Update Cargo.lock after rebase * Try to pass `download` as an argument to `zebrad` in coverage CI * Fix copy and paste comment typos * Add path and download examples, like zcash_proofs * Download params in CI just like zcash_proofs does * Delete a redundant build step * Implement graceful shutdown for zebrad start * Send coverage summary to /dev/null when getting the params path * Use the correct parameters path and download commands in CI * Explain pre-downloads * Avoid calling params_folder twice * Rename parameter types and methods for consistency ```sh fastmod SaplingParams SaplingParameters zebra* fastmod Groth16Params Groth16Parameters zebra* fastmod PARAMS GROTH16_PARAMETERS zebra* fastmod params_folder directory zebra* ``` And a manual variable name tweak. * rustfmt * Remove a redundant coverage step Co-authored-by: Janito Vaqueiro Ferreira Filho <janito.vff@gmail.com>
This commit is contained in:
parent
26b3a50e01
commit
f7202bfbc0
|
@ -48,22 +48,13 @@ jobs:
|
|||
echo "C:\Program Files\LLVM\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append
|
||||
echo "LIBCLANG_PATH=C:\Program Files\LLVM\bin" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append
|
||||
|
||||
- name: Skip network tests on Ubuntu
|
||||
# Ubuntu runners don't have network or DNS configured during test steps
|
||||
if: matrix.os == 'ubuntu-latest'
|
||||
- name: Skip network tests on Ubuntu and Windows
|
||||
# Ubuntu runners don't have network or DNS configured during test steps.
|
||||
# Windows runners have an unreliable network.
|
||||
shell: bash
|
||||
if: matrix.os != 'macOS-latest'
|
||||
run: echo "ZEBRA_SKIP_NETWORK_TESTS=1" >> $GITHUB_ENV
|
||||
|
||||
- name: Skip network tests on Windows
|
||||
# Windows runners have an unreliable network
|
||||
if: matrix.os == 'windows-latest'
|
||||
run: echo "ZEBRA_SKIP_NETWORK_TESTS=1" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append
|
||||
|
||||
- name: Show env vars
|
||||
run: |
|
||||
echo "ZEBRA_SKIP_NETWORK_TESTS=${{ env.ZEBRA_SKIP_NETWORK_TESTS }}"
|
||||
echo "CARGO_INCREMENTAL=${{ env.CARGO_INCREMENTAL }}"
|
||||
echo "RUST_BACKTRACE=${{ env.RUST_BACKTRACE }}"
|
||||
|
||||
- name: Change target output directory on Windows
|
||||
# Windows doesn't have enough space on the D: drive, so we redirect the build output to the
|
||||
# larger C: drive.
|
||||
|
@ -73,6 +64,31 @@ jobs:
|
|||
mkdir C:\zebra-target
|
||||
echo "CARGO_TARGET_DIR=C:\zebra-target" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append
|
||||
|
||||
# Modified from:
|
||||
# https://github.com/zcash/librustzcash/blob/c48bb4def2e122289843ddb3cb2984c325c03ca0/.github/workflows/ci.yml#L20-L33
|
||||
- name: Fetch path to Zcash parameters
|
||||
working-directory: ./zebra-consensus
|
||||
shell: bash
|
||||
run: echo "ZCASH_PARAMS=$(cargo run --example get-params-path)" >> $GITHUB_ENV
|
||||
- name: Cache Zcash parameters
|
||||
id: cache-params
|
||||
uses: actions/cache@v2
|
||||
with:
|
||||
path: ${{ env.ZCASH_PARAMS }}
|
||||
key: ${{ runner.os }}-params
|
||||
- name: Fetch Zcash parameters
|
||||
if: steps.cache-params.outputs.cache-hit != 'true'
|
||||
working-directory: ./zebra-consensus
|
||||
run: cargo run --example download-params
|
||||
|
||||
- name: Show env vars
|
||||
run: |
|
||||
echo "ZEBRA_SKIP_NETWORK_TESTS=${{ env.ZEBRA_SKIP_NETWORK_TESTS }}"
|
||||
echo "ZCASH_PARAMS=${{ env.ZCASH_PARAMS }}"
|
||||
echo "CARGO_INCREMENTAL=${{ env.CARGO_INCREMENTAL }}"
|
||||
echo "CARGO_TARGET_DIR=${{ env.CARGO_TARGET_DIR }}"
|
||||
echo "RUST_BACKTRACE=${{ env.RUST_BACKTRACE }}"
|
||||
|
||||
- name: Run tests
|
||||
uses: actions-rs/cargo@v1.0.3
|
||||
with:
|
||||
|
@ -132,7 +148,9 @@ jobs:
|
|||
- name: Show env vars
|
||||
run: |
|
||||
echo "ZEBRA_SKIP_NETWORK_TESTS=${{ env.ZEBRA_SKIP_NETWORK_TESTS }}"
|
||||
echo "ZCASH_PARAMS=${{ env.ZCASH_PARAMS }}"
|
||||
echo "CARGO_INCREMENTAL=${{ env.CARGO_INCREMENTAL }}"
|
||||
echo "CARGO_TARGET_DIR=${{ env.CARGO_TARGET_DIR }}"
|
||||
echo "RUST_BACKTRACE=${{ env.RUST_BACKTRACE }}"
|
||||
|
||||
- name: Run build without features enabled
|
||||
|
@ -169,7 +187,9 @@ jobs:
|
|||
- name: Show env vars
|
||||
run: |
|
||||
echo "ZEBRA_SKIP_NETWORK_TESTS=${{ env.ZEBRA_SKIP_NETWORK_TESTS }}"
|
||||
echo "ZCASH_PARAMS=${{ env.ZCASH_PARAMS }}"
|
||||
echo "CARGO_INCREMENTAL=${{ env.CARGO_INCREMENTAL }}"
|
||||
echo "CARGO_TARGET_DIR=${{ env.CARGO_TARGET_DIR }}"
|
||||
echo "RUST_BACKTRACE=${{ env.RUST_BACKTRACE }}"
|
||||
|
||||
- name: Build
|
||||
|
@ -202,7 +222,9 @@ jobs:
|
|||
- name: Show env vars
|
||||
run: |
|
||||
echo "ZEBRA_SKIP_NETWORK_TESTS=${{ env.ZEBRA_SKIP_NETWORK_TESTS }}"
|
||||
echo "ZCASH_PARAMS=${{ env.ZCASH_PARAMS }}"
|
||||
echo "CARGO_INCREMENTAL=${{ env.CARGO_INCREMENTAL }}"
|
||||
echo "CARGO_TARGET_DIR=${{ env.CARGO_TARGET_DIR }}"
|
||||
echo "RUST_BACKTRACE=${{ env.RUST_BACKTRACE }}"
|
||||
|
||||
- name: Run clippy
|
||||
|
@ -244,7 +266,9 @@ jobs:
|
|||
- name: Show env vars
|
||||
run: |
|
||||
echo "ZEBRA_SKIP_NETWORK_TESTS=${{ env.ZEBRA_SKIP_NETWORK_TESTS }}"
|
||||
echo "ZCASH_PARAMS=${{ env.ZCASH_PARAMS }}"
|
||||
echo "CARGO_INCREMENTAL=${{ env.CARGO_INCREMENTAL }}"
|
||||
echo "CARGO_TARGET_DIR=${{ env.CARGO_TARGET_DIR }}"
|
||||
echo "RUST_BACKTRACE=${{ env.RUST_BACKTRACE }}"
|
||||
|
||||
- name: Check rustfmt
|
||||
|
|
|
@ -16,6 +16,10 @@ jobs:
|
|||
# The large timeout is to accommodate nightly builds
|
||||
timeout-minutes: 60
|
||||
runs-on: ubuntu-latest
|
||||
env:
|
||||
CARGO_INCREMENTAL: 0
|
||||
RUST_BACKTRACE: full
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v2.4.0
|
||||
with:
|
||||
|
@ -31,11 +35,44 @@ jobs:
|
|||
- name: Install cargo-llvm-cov cargo command
|
||||
run: cargo install cargo-llvm-cov
|
||||
|
||||
- name: Generate code coverage
|
||||
env:
|
||||
ZEBRA_SKIP_NETWORK_TESTS: 1
|
||||
CARGO_INCREMENTAL: 0
|
||||
run: cargo llvm-cov --lcov > lcov.info
|
||||
- name: Skip network tests on Ubuntu and Windows
|
||||
# Ubuntu runners don't have network or DNS configured during test steps.
|
||||
# Windows runners have an unreliable network.
|
||||
shell: bash
|
||||
if: matrix.os != 'macOS-latest'
|
||||
run: echo "ZEBRA_SKIP_NETWORK_TESTS=1" >> $GITHUB_ENV
|
||||
|
||||
# Modified from:
|
||||
# https://github.com/zcash/librustzcash/blob/c48bb4def2e122289843ddb3cb2984c325c03ca0/.github/workflows/ci.yml#L20-L33
|
||||
- name: Fetch path to Zcash parameters
|
||||
working-directory: ./zebra-consensus
|
||||
shell: bash
|
||||
# cargo-llvm-cov doesn't have a silent mode, so we have to extract the path from stderr
|
||||
run: echo "ZCASH_PARAMS=$(cargo llvm-cov --lcov --no-report run --example get-params-path 2>&1 >/dev/null | tail -1)" >> $GITHUB_ENV
|
||||
- name: Cache Zcash parameters
|
||||
id: cache-params
|
||||
uses: actions/cache@v2
|
||||
with:
|
||||
path: ${{ env.ZCASH_PARAMS }}
|
||||
key: ${{ runner.os }}-params
|
||||
- name: Fetch Zcash parameters
|
||||
if: steps.cache-params.outputs.cache-hit != 'true'
|
||||
working-directory: ./zebra-consensus
|
||||
run: cargo llvm-cov --lcov --no-report run --example download-params
|
||||
|
||||
- name: Show env vars
|
||||
run: |
|
||||
echo "ZEBRA_SKIP_NETWORK_TESTS=${{ env.ZEBRA_SKIP_NETWORK_TESTS }}"
|
||||
echo "ZCASH_PARAMS=${{ env.ZCASH_PARAMS }}"
|
||||
echo "CARGO_INCREMENTAL=${{ env.CARGO_INCREMENTAL }}"
|
||||
echo "CARGO_TARGET_DIR=${{ env.CARGO_TARGET_DIR }}"
|
||||
echo "RUST_BACKTRACE=${{ env.RUST_BACKTRACE }}"
|
||||
|
||||
- name: Run Zebra tests
|
||||
run: cargo llvm-cov --lcov --no-report
|
||||
|
||||
- name: Generate coverage report
|
||||
run: cargo llvm-cov --lcov --no-run --output-path lcov.info
|
||||
|
||||
- name: Upload coverage report to Codecov
|
||||
uses: codecov/codecov-action@v2.1.0
|
||||
|
|
|
@ -20,7 +20,7 @@ jobs:
|
|||
- uses: actions/checkout@v2.4.0
|
||||
with:
|
||||
persist-credentials: false
|
||||
|
||||
|
||||
- name: Set project and image names
|
||||
run: |
|
||||
BRANCH_NAME=$(expr $GITHUB_REF : '.*/\(.*\)') && \
|
||||
|
|
|
@ -974,6 +974,15 @@ dependencies = [
|
|||
"generic-array",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "directories"
|
||||
version = "4.0.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "f51c5d4ddabd36886dd3e1438cb358cdcb0d7c499cb99cb4ac2e38e18b5cb210"
|
||||
dependencies = [
|
||||
"dirs-sys",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "dirs"
|
||||
version = "4.0.0"
|
||||
|
@ -1074,16 +1083,6 @@ dependencies = [
|
|||
"termcolor",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "equihash"
|
||||
version = "0.1.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "4127688f6177e3f57521881cb1cfd90d1228214f9dc43b8efe6f6c6948cd8280"
|
||||
dependencies = [
|
||||
"blake2b_simd",
|
||||
"byteorder",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "equihash"
|
||||
version = "0.1.0"
|
||||
|
@ -1614,7 +1613,7 @@ dependencies = [
|
|||
[[package]]
|
||||
name = "incrementalmerkletree"
|
||||
version = "0.1.0"
|
||||
source = "git+https://github.com/zcash/incrementalmerkletree?rev=b7bd6246122a6e9ace8edb51553fbf5228906cbb#b7bd6246122a6e9ace8edb51553fbf5228906cbb"
|
||||
source = "git+https://github.com/zcash/incrementalmerkletree.git?rev=b7bd6246122a6e9ace8edb51553fbf5228906cbb#b7bd6246122a6e9ace8edb51553fbf5228906cbb"
|
||||
dependencies = [
|
||||
"serde",
|
||||
]
|
||||
|
@ -1987,6 +1986,19 @@ dependencies = [
|
|||
"autocfg",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "minreq"
|
||||
version = "2.4.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "d5f7db7a675c4b46b8842105b9371d6151e95fbbecd9b0e54dc2ea814397d2cc"
|
||||
dependencies = [
|
||||
"lazy_static",
|
||||
"log",
|
||||
"rustls",
|
||||
"webpki",
|
||||
"webpki-roots 0.18.0",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "mio"
|
||||
version = "0.7.6"
|
||||
|
@ -2817,7 +2829,7 @@ dependencies = [
|
|||
"wasm-bindgen",
|
||||
"wasm-bindgen-futures",
|
||||
"web-sys",
|
||||
"webpki-roots",
|
||||
"webpki-roots 0.21.1",
|
||||
"winreg",
|
||||
]
|
||||
|
||||
|
@ -3961,56 +3973,6 @@ version = "0.9.2"
|
|||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b5a972e5669d67ba988ce3dc826706fb0a8b01471c088cb0b6110b805cc36aed"
|
||||
|
||||
[[package]]
|
||||
name = "wagyu-zcash-parameters"
|
||||
version = "0.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "61c904628658374e651288f000934c33ef738b2d8b3e65d4100b70b395dbe2bb"
|
||||
dependencies = [
|
||||
"wagyu-zcash-parameters-1",
|
||||
"wagyu-zcash-parameters-2",
|
||||
"wagyu-zcash-parameters-3",
|
||||
"wagyu-zcash-parameters-4",
|
||||
"wagyu-zcash-parameters-5",
|
||||
"wagyu-zcash-parameters-6",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "wagyu-zcash-parameters-1"
|
||||
version = "0.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "90bf2e21bb027d3f8428c60d6a720b54a08bf6ce4e6f834ef8e0d38bb5695da8"
|
||||
|
||||
[[package]]
|
||||
name = "wagyu-zcash-parameters-2"
|
||||
version = "0.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a616ab2e51e74cc48995d476e94de810fb16fc73815f390bf2941b046cc9ba2c"
|
||||
|
||||
[[package]]
|
||||
name = "wagyu-zcash-parameters-3"
|
||||
version = "0.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "14da1e2e958ff93c0830ee68e91884069253bf3462a67831b02b367be75d6147"
|
||||
|
||||
[[package]]
|
||||
name = "wagyu-zcash-parameters-4"
|
||||
version = "0.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "f058aeef03a2070e8666ffb5d1057d8bb10313b204a254a6e6103eb958e9a6d6"
|
||||
|
||||
[[package]]
|
||||
name = "wagyu-zcash-parameters-5"
|
||||
version = "0.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "3ffe916b30e608c032ae1b734f02574a3e12ec19ab5cc5562208d679efe4969d"
|
||||
|
||||
[[package]]
|
||||
name = "wagyu-zcash-parameters-6"
|
||||
version = "0.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a7b6d5a78adc3e8f198e9cd730f219a695431467f7ec29dcfc63ade885feebe1"
|
||||
|
||||
[[package]]
|
||||
name = "wait-timeout"
|
||||
version = "0.2.0"
|
||||
|
@ -4139,6 +4101,15 @@ dependencies = [
|
|||
"untrusted",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "webpki-roots"
|
||||
version = "0.18.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "91cd5736df7f12a964a5067a12c62fa38e1bd8080aff1f80bc29be7c80d19ab4"
|
||||
dependencies = [
|
||||
"webpki",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "webpki-roots"
|
||||
version = "0.21.1"
|
||||
|
@ -4265,7 +4236,7 @@ dependencies = [
|
|||
"bls12_381 0.6.0",
|
||||
"byteorder",
|
||||
"chacha20poly1305",
|
||||
"equihash 0.1.0 (git+https://github.com/zcash/librustzcash.git?rev=53d0a51d33a421cb76d3e3124d1e4c1c9036068e)",
|
||||
"equihash",
|
||||
"ff 0.11.0",
|
||||
"fpe",
|
||||
"group 0.11.0",
|
||||
|
@ -4286,6 +4257,25 @@ dependencies = [
|
|||
"zcash_note_encryption",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "zcash_proofs"
|
||||
version = "0.5.0"
|
||||
source = "git+https://github.com/zcash/librustzcash.git?rev=53d0a51d33a421cb76d3e3124d1e4c1c9036068e#53d0a51d33a421cb76d3e3124d1e4c1c9036068e"
|
||||
dependencies = [
|
||||
"bellman",
|
||||
"blake2b_simd",
|
||||
"bls12_381 0.6.0",
|
||||
"byteorder",
|
||||
"directories",
|
||||
"ff 0.11.0",
|
||||
"group 0.11.0",
|
||||
"jubjub 0.8.0",
|
||||
"lazy_static",
|
||||
"minreq",
|
||||
"rand_core 0.6.3",
|
||||
"zcash_primitives",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "zcash_script"
|
||||
version = "0.1.6-alpha.0"
|
||||
|
@ -4324,7 +4314,7 @@ dependencies = [
|
|||
"criterion",
|
||||
"displaydoc",
|
||||
"ed25519-zebra",
|
||||
"equihash 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||
"equihash",
|
||||
"fpe",
|
||||
"futures",
|
||||
"group 0.11.0",
|
||||
|
@ -4372,6 +4362,7 @@ dependencies = [
|
|||
"bls12_381 0.6.0",
|
||||
"chrono",
|
||||
"color-eyre",
|
||||
"dirs",
|
||||
"displaydoc",
|
||||
"futures",
|
||||
"futures-util",
|
||||
|
@ -4397,7 +4388,7 @@ dependencies = [
|
|||
"tracing-error",
|
||||
"tracing-futures",
|
||||
"tracing-subscriber 0.2.25",
|
||||
"wagyu-zcash-parameters",
|
||||
"zcash_proofs",
|
||||
"zebra-chain",
|
||||
"zebra-script",
|
||||
"zebra-state",
|
||||
|
@ -4538,6 +4529,7 @@ dependencies = [
|
|||
"gumdrop",
|
||||
"hyper",
|
||||
"inferno",
|
||||
"lazy_static",
|
||||
"metrics",
|
||||
"metrics-exporter-prometheus",
|
||||
"once_cell",
|
||||
|
|
20
Cargo.toml
20
Cargo.toml
|
@ -22,11 +22,25 @@ panic = "abort"
|
|||
|
||||
[patch.crates-io]
|
||||
|
||||
# TODO: remove these after a new librustzcash release.
|
||||
# These are librustzcash requirements specified in its workspace Cargo.toml that we must replicate here
|
||||
incrementalmerkletree = { git = "https://github.com/zcash/incrementalmerkletree", rev = "b7bd6246122a6e9ace8edb51553fbf5228906cbb" }
|
||||
# TODO: replace with upstream orchard when these changes are merged
|
||||
# https://github.com/ZcashFoundation/zebra/issues/3056
|
||||
orchard = { git = "https://github.com/ZcashFoundation/orchard.git", rev = "568e24cd5f129158375d7ac7d98c89ebff4f982f" }
|
||||
|
||||
# TODO: remove these after a new librustzcash release.
|
||||
|
||||
# These are librustzcash git requirements specified in its workspace Cargo.toml,
|
||||
# that we must replicate here
|
||||
incrementalmerkletree = { git = "https://github.com/zcash/incrementalmerkletree.git", rev = "b7bd6246122a6e9ace8edb51553fbf5228906cbb" }
|
||||
# Replaced by the ZcashFoundation fork above
|
||||
#orchard = { git = "https://github.com/zcash/orchard.git", rev = "2c8241f25b943aa05203eacf9905db117c69bd29" }
|
||||
|
||||
# These are librustzcash file requirements specified in its workspace Cargo.toml,
|
||||
# that we must replace with git requirements
|
||||
zcash_note_encryption = { git = "https://github.com/zcash/librustzcash.git", rev = "53d0a51d33a421cb76d3e3124d1e4c1c9036068e" }
|
||||
|
||||
# These patches are not strictly required,
|
||||
# but they help avoid duplicate dependencies
|
||||
equihash = { git = "https://github.com/zcash/librustzcash.git", rev = "53d0a51d33a421cb76d3e3124d1e4c1c9036068e" }
|
||||
zcash_history = { git = "https://github.com/zcash/librustzcash.git", rev = "53d0a51d33a421cb76d3e3124d1e4c1c9036068e" }
|
||||
zcash_primitives = { git = "https://github.com/zcash/librustzcash.git", rev = "53d0a51d33a421cb76d3e3124d1e4c1c9036068e" }
|
||||
zcash_proofs = { git = "https://github.com/zcash/librustzcash.git", rev = "53d0a51d33a421cb76d3e3124d1e4c1c9036068e" }
|
||||
|
|
21
deny.toml
21
deny.toml
|
@ -34,12 +34,6 @@ skip-tree = [
|
|||
# ticket #2953: tracing dependencies
|
||||
{ name = "tracing-subscriber", version = "=0.1.6" },
|
||||
|
||||
# ticket #2982: librustzcash and orchard git versions
|
||||
# Note that the equihash duplication is probably because `zcash_primitives`
|
||||
# (which imports it with a path import) is being imported as a git dependency.
|
||||
{ name = "equihash", version = "=0.1.0" },
|
||||
{ name = "orchard", version = "=0.0.0" },
|
||||
|
||||
# ticket #2983: criterion dependencies
|
||||
{ name = "criterion", version = "=0.3.4" },
|
||||
|
||||
|
@ -49,6 +43,9 @@ skip-tree = [
|
|||
# ticket #2981: bindgen dependencies
|
||||
{ name = "rocksdb", version = "=0.16.0" },
|
||||
|
||||
# ticket #3063: redjubjub dependencies
|
||||
{ name = "redjubjub", version = "=0.4.0" },
|
||||
|
||||
# ticket #2984: owo-colors dependencies
|
||||
{ name = "color-eyre", version = "=0.5.11" },
|
||||
|
||||
|
@ -61,14 +58,24 @@ skip-tree = [
|
|||
# ticket #2999: http dependencies
|
||||
{ name = "bytes", version = "=0.5.6" },
|
||||
|
||||
# ticket #3061: reqwest and minreq dependencies
|
||||
{ name = "webpki-roots", version = "=0.18.0" },
|
||||
|
||||
# ticket #2980: inferno and orchard/cryptographic dependencies
|
||||
{ name = "inferno", version = "=0.10.8" },
|
||||
{ name = "orchard", version = "=0.0.0" },
|
||||
|
||||
# upgrade orchard from deprecated `bigint` to `uint`: https://github.com/zcash/orchard/issues/219
|
||||
# alternative: downgrade Zebra to `bigint`
|
||||
{ name = "bigint", version = "=4.4.3" },
|
||||
|
||||
# recent major version bumps
|
||||
# we should re-check these dependencies in February 2022
|
||||
|
||||
# wait for lots of crates in the cryptographic ecosystem to upgrade
|
||||
{ name = "rand", version = "=0.7.3" },
|
||||
|
||||
# wait for lots of crates in the tokio ecosystem to upgrade
|
||||
# we should re-check these dependencies in February 2022
|
||||
{ name = "redox_syscall", version = "=0.1.57" },
|
||||
{ name = "socket2", version = "=0.3.16" },
|
||||
]
|
||||
|
|
|
@ -20,7 +20,6 @@ COPY . .
|
|||
|
||||
RUN cd zebrad/; cargo build --release --features enable-sentry
|
||||
|
||||
|
||||
# Runner image
|
||||
FROM debian:buster-slim AS zebrad-release
|
||||
|
||||
|
@ -45,6 +44,9 @@ RUN printf "[tracing]\n" >> /zebrad.toml
|
|||
RUN printf "endpoint_addr = '0.0.0.0:3000'\n" >> /zebrad.toml
|
||||
RUN cat /zebrad.toml
|
||||
|
||||
# Pre-download Zcash Sprout and Sapling parameters
|
||||
RUN /zebrad download
|
||||
|
||||
EXPOSE 3000 8233 18233
|
||||
|
||||
ENV RUST_LOG debug
|
||||
|
|
|
@ -22,6 +22,9 @@ EXPOSE 8233 18233
|
|||
|
||||
COPY . .
|
||||
|
||||
# Pre-download Zcash Sprout and Sapling parameters
|
||||
RUN cargo run --verbose --bin zebrad download
|
||||
|
||||
RUN cargo test --all --no-run
|
||||
|
||||
CMD cargo test --workspace --no-fail-fast -- -Zunstable-options --include-ignored
|
||||
|
|
|
@ -13,16 +13,19 @@ proptest-impl = ["proptest", "proptest-derive", "zebra-chain/proptest-impl"]
|
|||
blake2b_simd = "0.5.11"
|
||||
bellman = "0.11.1"
|
||||
bls12_381 = "0.6.0"
|
||||
chrono = "0.4.19"
|
||||
displaydoc = "0.2.2"
|
||||
halo2 = "=0.1.0-beta.1"
|
||||
jubjub = "0.8.0"
|
||||
lazy_static = "1.4.0"
|
||||
once_cell = "1.8"
|
||||
rand = "0.8"
|
||||
|
||||
halo2 = "=0.1.0-beta.1"
|
||||
# TODO: replace with upstream orchard when these changes are merged
|
||||
# https://github.com/ZcashFoundation/zebra/issues/3056
|
||||
orchard = "0.0.0"
|
||||
rand = "0.8"
|
||||
|
||||
chrono = "0.4.19"
|
||||
dirs = "4.0.0"
|
||||
displaydoc = "0.2.2"
|
||||
lazy_static = "1.4.0"
|
||||
once_cell = "1.8"
|
||||
serde = { version = "1", features = ["serde_derive"] }
|
||||
|
||||
futures = "0.3.17"
|
||||
|
@ -34,12 +37,14 @@ tower = { version = "0.4.9", features = ["timeout", "util", "buffer"] }
|
|||
tracing = "0.1.29"
|
||||
tracing-futures = "0.2.5"
|
||||
|
||||
zcash_proofs = { git = "https://github.com/zcash/librustzcash.git", rev = "53d0a51d33a421cb76d3e3124d1e4c1c9036068e", features = ["local-prover", "multicore", "download-params"] }
|
||||
|
||||
tower-fallback = { path = "../tower-fallback/" }
|
||||
tower-batch = { path = "../tower-batch/" }
|
||||
|
||||
zebra-chain = { path = "../zebra-chain" }
|
||||
zebra-state = { path = "../zebra-state" }
|
||||
zebra-script = { path = "../zebra-script" }
|
||||
wagyu-zcash-parameters = "0.2.0"
|
||||
|
||||
proptest = { version = "0.10", optional = true }
|
||||
proptest-derive = { version = "0.3.0", optional = true }
|
||||
|
|
|
@ -0,0 +1,11 @@
|
|||
//! Download the Sapling and Sprout Groth16 parameters if needed,
|
||||
//! check they were downloaded correctly, and load them into Zebra.
|
||||
|
||||
// Has the same functionality as:
|
||||
// https://github.com/zcash/librustzcash/blob/c48bb4def2e122289843ddb3cb2984c325c03ca0/zcash_proofs/examples/download-params.rs
|
||||
|
||||
fn main() {
|
||||
// The lazy static initializer does the download, if needed,
|
||||
// and the file hash checks.
|
||||
lazy_static::initialize(&zebra_consensus::groth16::GROTH16_PARAMETERS);
|
||||
}
|
|
@ -0,0 +1,11 @@
|
|||
//! Print the Zcash parameter directory path to standard output.
|
||||
|
||||
// Modified from:
|
||||
// https://github.com/zcash/librustzcash/blob/c48bb4def2e122289843ddb3cb2984c325c03ca0/zcash_proofs/examples/get-params-path.rs
|
||||
|
||||
fn main() {
|
||||
let path = zebra_consensus::groth16::Groth16Parameters::directory();
|
||||
if let Some(path) = path.to_str() {
|
||||
println!("{}", path);
|
||||
}
|
||||
}
|
|
@ -22,6 +22,7 @@ use std::{
|
|||
use displaydoc::Display;
|
||||
use futures::{FutureExt, TryFutureExt};
|
||||
use thiserror::Error;
|
||||
use tokio::task::{spawn_blocking, JoinHandle};
|
||||
use tower::{buffer::Buffer, util::BoxService, Service, ServiceExt};
|
||||
use tracing::instrument;
|
||||
|
||||
|
@ -148,7 +149,11 @@ where
|
|||
}
|
||||
}
|
||||
|
||||
/// Initialize block and transaction verification services.
|
||||
/// Initialize block and transaction verification services,
|
||||
/// and pre-download Groth16 parameters if needed.
|
||||
///
|
||||
/// Returns a block verifier, transaction verifier,
|
||||
/// and the Groth16 parameter download task [`JoinHandle`].
|
||||
///
|
||||
/// The consensus configuration is specified by `config`, and the Zcash network
|
||||
/// to verify blocks for is specified by `network`.
|
||||
|
@ -160,6 +165,12 @@ where
|
|||
/// The transaction verification service asynchronously performs semantic verification
|
||||
/// checks. Transactions that pass semantic verification return an `Ok` result to the caller.
|
||||
///
|
||||
/// Pre-downloads the Sapling and Sprout Groth16 parameters if needed,
|
||||
/// checks they were downloaded correctly, and loads them into Zebra.
|
||||
/// (The transaction verifier automatically downloads the parameters on first use.
|
||||
/// But the parameter downloads can take around 10 minutes.
|
||||
/// So we pre-download the parameters, to avoid verification timeouts.)
|
||||
///
|
||||
/// This function should only be called once for a particular state service.
|
||||
///
|
||||
/// Dropped requests are cancelled on a best-effort basis, but may continue to be processed.
|
||||
|
@ -180,11 +191,22 @@ pub async fn init<S>(
|
|||
BoxService<transaction::Request, transaction::Response, TransactionError>,
|
||||
transaction::Request,
|
||||
>,
|
||||
JoinHandle<()>,
|
||||
)
|
||||
where
|
||||
S: Service<zs::Request, Response = zs::Response, Error = BoxError> + Send + Clone + 'static,
|
||||
S::Future: Send + 'static,
|
||||
{
|
||||
// pre-download Groth16 parameters async
|
||||
|
||||
let groth16_download_handle = spawn_blocking(|| {
|
||||
tracing::info!("checking if Zcash Sapling and Sprout parameters have been downloaded");
|
||||
|
||||
// The lazy static initializer does the download, if needed,
|
||||
// and the file hash checks.
|
||||
lazy_static::initialize(&crate::groth16::GROTH16_PARAMETERS);
|
||||
});
|
||||
|
||||
// transaction verification
|
||||
|
||||
let script = script::Verifier::new(state_service.clone());
|
||||
|
@ -225,5 +247,5 @@ where
|
|||
|
||||
let chain = Buffer::new(BoxService::new(chain), VERIFIER_BUFFER_BOUND);
|
||||
|
||||
(chain, transaction)
|
||||
(chain, transaction, groth16_download_handle)
|
||||
}
|
||||
|
|
|
@ -41,8 +41,8 @@ pub fn block_no_transactions() -> Block {
|
|||
}
|
||||
}
|
||||
|
||||
/// Return a new `(chain_verifier, state_service)` using the hard-coded
|
||||
/// checkpoint list for `network`.
|
||||
/// Return a new chain verifier and state service,
|
||||
/// using the hard-coded checkpoint list for `network`.
|
||||
async fn verifiers_from_network(
|
||||
network: Network,
|
||||
) -> (
|
||||
|
@ -64,9 +64,13 @@ async fn verifiers_from_network(
|
|||
+ 'static,
|
||||
) {
|
||||
let state_service = zs::init_test(network);
|
||||
let (chain_verifier, _transaction_verifier) =
|
||||
let (chain_verifier, _transaction_verifier, _groth16_download_handle) =
|
||||
crate::chain::init(Config::default(), network, state_service.clone()).await;
|
||||
|
||||
// We can drop the download task handle here, because:
|
||||
// - if the download task fails, the tests will panic, and
|
||||
// - if the download task hangs, the tests will hang.
|
||||
|
||||
(chain_verifier, state_service)
|
||||
}
|
||||
|
||||
|
@ -153,7 +157,9 @@ async fn verify_checkpoint(config: Config) -> Result<(), Report> {
|
|||
|
||||
// Test that the chain::init function works. Most of the other tests use
|
||||
// init_from_verifiers.
|
||||
let (chain_verifier, _transaction_verifier) =
|
||||
//
|
||||
// Download task panics and timeouts are propagated to the tests that use Groth16 verifiers.
|
||||
let (chain_verifier, _transaction_verifier, _groth16_download_handle) =
|
||||
super::init(config.clone(), network, zs::init_test(network)).await;
|
||||
|
||||
// Add a timeout layer
|
||||
|
|
|
@ -59,6 +59,7 @@ pub use checkpoint::MAX_CHECKPOINT_BYTE_COUNT;
|
|||
pub use checkpoint::MAX_CHECKPOINT_HEIGHT_GAP;
|
||||
pub use config::Config;
|
||||
pub use error::BlockError;
|
||||
pub use primitives::groth16;
|
||||
|
||||
/// A boxed [`std::error::Error`].
|
||||
pub type BoxError = Box<dyn std::error::Error + Send + Sync + 'static>;
|
||||
|
|
|
@ -18,17 +18,17 @@ use once_cell::sync::Lazy;
|
|||
use rand::thread_rng;
|
||||
use tokio::sync::broadcast::{channel, error::RecvError, Sender};
|
||||
use tower::{util::ServiceFn, Service};
|
||||
|
||||
use tower_batch::{Batch, BatchControl};
|
||||
use tower_fallback::Fallback;
|
||||
|
||||
use zebra_chain::sapling::{Output, PerSpendAnchor, Spend};
|
||||
|
||||
mod hash_reader;
|
||||
mod params;
|
||||
#[cfg(test)]
|
||||
mod tests;
|
||||
|
||||
use self::hash_reader::HashReader;
|
||||
use params::PARAMS;
|
||||
pub use params::{Groth16Parameters, GROTH16_PARAMETERS};
|
||||
|
||||
/// Global batch verification context for Groth16 proofs of Spend statements.
|
||||
///
|
||||
|
@ -40,28 +40,31 @@ use params::PARAMS;
|
|||
/// handle.
|
||||
pub static SPEND_VERIFIER: Lazy<
|
||||
Fallback<Batch<Verifier, Item>, ServiceFn<fn(Item) -> Ready<Result<(), VerificationError>>>>,
|
||||
> = Lazy::new(|| {
|
||||
Fallback::new(
|
||||
Batch::new(
|
||||
Verifier::new(&PARAMS.sapling.spend.vk),
|
||||
super::MAX_BATCH_SIZE,
|
||||
super::MAX_BATCH_LATENCY,
|
||||
),
|
||||
// We want to fallback to individual verification if batch verification
|
||||
// fails, so we need a Service to use. The obvious way to do this would
|
||||
// be to write a closure that returns an async block. But because we
|
||||
// have to specify the type of a static, we need to be able to write the
|
||||
// type of the closure and its return value, and both closures and async
|
||||
// blocks have eldritch types whose names cannot be written. So instead,
|
||||
// we use a Ready to avoid an async block and cast the closure to a
|
||||
// function (which is possible because it doesn't capture any state).
|
||||
tower::service_fn(
|
||||
(|item: Item| {
|
||||
ready(item.verify_single(&prepare_verifying_key(&PARAMS.sapling.spend.vk)))
|
||||
}) as fn(_) -> _,
|
||||
),
|
||||
)
|
||||
});
|
||||
> =
|
||||
Lazy::new(|| {
|
||||
Fallback::new(
|
||||
Batch::new(
|
||||
Verifier::new(&GROTH16_PARAMETERS.sapling.spend.vk),
|
||||
super::MAX_BATCH_SIZE,
|
||||
super::MAX_BATCH_LATENCY,
|
||||
),
|
||||
// We want to fallback to individual verification if batch verification
|
||||
// fails, so we need a Service to use. The obvious way to do this would
|
||||
// be to write a closure that returns an async block. But because we
|
||||
// have to specify the type of a static, we need to be able to write the
|
||||
// type of the closure and its return value, and both closures and async
|
||||
// blocks have eldritch types whose names cannot be written. So instead,
|
||||
// we use a Ready to avoid an async block and cast the closure to a
|
||||
// function (which is possible because it doesn't capture any state).
|
||||
tower::service_fn(
|
||||
(|item: Item| {
|
||||
ready(item.verify_single(&prepare_verifying_key(
|
||||
&GROTH16_PARAMETERS.sapling.spend.vk,
|
||||
)))
|
||||
}) as fn(_) -> _,
|
||||
),
|
||||
)
|
||||
});
|
||||
|
||||
/// Global batch verification context for Groth16 proofs of Output statements.
|
||||
///
|
||||
|
@ -76,7 +79,7 @@ pub static OUTPUT_VERIFIER: Lazy<
|
|||
> = Lazy::new(|| {
|
||||
Fallback::new(
|
||||
Batch::new(
|
||||
Verifier::new(&PARAMS.sapling.output.vk),
|
||||
Verifier::new(&GROTH16_PARAMETERS.sapling.output.vk),
|
||||
super::MAX_BATCH_SIZE,
|
||||
super::MAX_BATCH_LATENCY,
|
||||
),
|
||||
|
@ -90,7 +93,9 @@ pub static OUTPUT_VERIFIER: Lazy<
|
|||
// function (which is possible because it doesn't capture any state).
|
||||
tower::service_fn(
|
||||
(|item: Item| {
|
||||
ready(item.verify_single(&prepare_verifying_key(&PARAMS.sapling.output.vk)))
|
||||
ready(item.verify_single(&prepare_verifying_key(
|
||||
&GROTH16_PARAMETERS.sapling.output.vk,
|
||||
)))
|
||||
}) as fn(_) -> _,
|
||||
),
|
||||
)
|
||||
|
@ -99,6 +104,7 @@ pub static OUTPUT_VERIFIER: Lazy<
|
|||
/// A Groth16 verification item, used as the request type of the service.
|
||||
pub type Item = batch::Item<Bls12>;
|
||||
|
||||
/// A wrapper to workaround the missing `ServiceExt::map_err` method.
|
||||
pub struct ItemWrapper(Item);
|
||||
|
||||
impl From<&Spend<PerSpendAnchor>> for ItemWrapper {
|
||||
|
|
|
@ -1,43 +0,0 @@
|
|||
use std::io::{self, Read};
|
||||
|
||||
use blake2b_simd::State;
|
||||
|
||||
/// Abstraction over a reader which hashes the data being read.
|
||||
pub struct HashReader<R: Read> {
|
||||
reader: R,
|
||||
hasher: State,
|
||||
}
|
||||
|
||||
impl<R: Read> HashReader<R> {
|
||||
/// Construct a new `HashReader` given an existing `reader` by value.
|
||||
pub fn new(reader: R) -> Self {
|
||||
HashReader {
|
||||
reader,
|
||||
hasher: State::new(),
|
||||
}
|
||||
}
|
||||
|
||||
/// Destroy this reader and return the hash of what was read.
|
||||
pub fn into_hash(self) -> String {
|
||||
let hash = self.hasher.finalize();
|
||||
|
||||
let mut s = String::new();
|
||||
for c in hash.as_bytes().iter() {
|
||||
s += &format!("{:02x}", c);
|
||||
}
|
||||
|
||||
s
|
||||
}
|
||||
}
|
||||
|
||||
impl<R: Read> Read for HashReader<R> {
|
||||
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
|
||||
let bytes = self.reader.read(buf)?;
|
||||
|
||||
if bytes > 0 {
|
||||
self.hasher.update(&buf[0..bytes]);
|
||||
}
|
||||
|
||||
Ok(bytes)
|
||||
}
|
||||
}
|
|
@ -1,32 +1,60 @@
|
|||
use std::io::{self, BufReader};
|
||||
//! Downloading, checking, and loading Groth16 Sapling and Sprout parameters.
|
||||
|
||||
use std::path::PathBuf;
|
||||
|
||||
use bellman::groth16;
|
||||
use bls12_381::Bls12;
|
||||
|
||||
use super::HashReader;
|
||||
|
||||
const SAPLING_SPEND_HASH: &str = "8270785a1a0d0bc77196f000ee6d221c9c9894f55307bd9357c3f0105d31ca63991ab91324160d8f53e2bbd3c2633a6eb8bdf5205d822e7f3f73edac51b2b70c";
|
||||
const SAPLING_OUTPUT_HASH: &str = "657e3d38dbb5cb5e7dd2970e8b03d69b4787dd907285b5a7f0790dcc8072f60bf593b32cc2d1c030e00ff5ae64bf84c5c3beb84ddc841d48264b4a171744d028";
|
||||
|
||||
lazy_static::lazy_static! {
|
||||
pub static ref PARAMS: Groth16Params = Groth16Params::new();
|
||||
/// Groth16 Zero-Knowledge Proof parameters for the Sapling and Sprout circuits.
|
||||
///
|
||||
/// When this static is accessed:
|
||||
/// - the parameters are downloded if needed, then cached to a shared directory,
|
||||
/// - the file hashes are checked, for both newly downloaded and previously cached files,
|
||||
/// - the parameters are loaded into Zebra.
|
||||
///
|
||||
/// # Panics
|
||||
///
|
||||
/// If the downloaded or pre-existing parameter files are invalid.
|
||||
pub static ref GROTH16_PARAMETERS: Groth16Parameters = Groth16Parameters::new();
|
||||
}
|
||||
|
||||
/// Groth16 Zero-Knowledge Proof parameters for the Sapling and Sprout circuits.
|
||||
#[non_exhaustive]
|
||||
pub struct Groth16Params {
|
||||
pub sapling: SaplingParams,
|
||||
pub struct Groth16Parameters {
|
||||
/// The Sapling circuit Groth16 parameters.
|
||||
pub sapling: SaplingParameters,
|
||||
}
|
||||
|
||||
impl Groth16Params {
|
||||
fn new() -> Self {
|
||||
Self {
|
||||
sapling: SaplingParams::new(),
|
||||
impl Groth16Parameters {
|
||||
/// Download if needed, cache, check, and load the Sprout and Sapling Groth16 parameters.
|
||||
///
|
||||
/// # Panics
|
||||
///
|
||||
/// If the downloaded or pre-existing parameter files are invalid.
|
||||
fn new() -> Groth16Parameters {
|
||||
Groth16Parameters {
|
||||
sapling: SaplingParameters::new(),
|
||||
}
|
||||
}
|
||||
|
||||
/// Returns the path to the Groth16 parameters directory.
|
||||
pub fn directory() -> PathBuf {
|
||||
zcash_proofs::default_params_folder().expect("unable to find user home directory")
|
||||
}
|
||||
|
||||
/// Returns a hint that helps users recover from parameter download failures.
|
||||
pub fn failure_hint() -> String {
|
||||
format!(
|
||||
"Hint: try deleting {:?}, then running 'zebrad download' to re-download the parameters",
|
||||
Groth16Parameters::directory(),
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
/// Groth16 Zero-Knowledge Proof spend and output parameters for the Sapling circuit.
|
||||
#[non_exhaustive]
|
||||
pub struct SaplingParams {
|
||||
pub struct SaplingParameters {
|
||||
pub spend: groth16::Parameters<Bls12>,
|
||||
pub spend_prepared_verifying_key: groth16::PreparedVerifyingKey<Bls12>,
|
||||
|
||||
|
@ -34,50 +62,41 @@ pub struct SaplingParams {
|
|||
pub output_prepared_verifying_key: groth16::PreparedVerifyingKey<Bls12>,
|
||||
}
|
||||
|
||||
impl SaplingParams {
|
||||
fn new() -> Self {
|
||||
let (spend, output) = wagyu_zcash_parameters::load_sapling_parameters();
|
||||
let spend_fs = BufReader::with_capacity(1024 * 1024, &spend[..]);
|
||||
let output_fs = BufReader::with_capacity(1024 * 1024, &output[..]);
|
||||
impl SaplingParameters {
|
||||
/// Download if needed, cache, check, and load the Sapling Groth16 parameters.
|
||||
///
|
||||
/// # Panics
|
||||
///
|
||||
/// If the downloaded or pre-existing parameter files are invalid.
|
||||
fn new() -> SaplingParameters {
|
||||
// TODO: Sprout
|
||||
|
||||
Self::read(spend_fs, output_fs)
|
||||
.expect("reading parameters from wagyu zcash parameter's vec will always succeed")
|
||||
}
|
||||
let params_directory = Groth16Parameters::directory();
|
||||
let spend_path = params_directory.join("sapling-spend.params");
|
||||
let output_path = params_directory.join("sapling-output.params");
|
||||
|
||||
fn read<R: io::Read>(spend_fs: R, output_fs: R) -> Result<Self, io::Error> {
|
||||
let mut spend_fs = HashReader::new(spend_fs);
|
||||
let mut output_fs = HashReader::new(output_fs);
|
||||
// Download parameters if needed.
|
||||
//
|
||||
// TODO: use try_exists when it stabilises, to exit early on permissions errors (#83186)
|
||||
if !spend_path.exists() || !output_path.exists() {
|
||||
tracing::info!("downloading Zcash Sapling parameters");
|
||||
zcash_proofs::download_parameters().unwrap_or_else(|_| {
|
||||
panic!(
|
||||
"error downloading parameter files. {}",
|
||||
Groth16Parameters::failure_hint()
|
||||
)
|
||||
});
|
||||
}
|
||||
|
||||
// Deserialize params
|
||||
let spend = groth16::Parameters::<Bls12>::read(&mut spend_fs, false)?;
|
||||
let output = groth16::Parameters::<Bls12>::read(&mut output_fs, false)?;
|
||||
// TODO: if loading fails, log a message including `failure_hint`
|
||||
tracing::info!("checking and loading Zcash Sapling parameters");
|
||||
let parameters = zcash_proofs::load_parameters(&spend_path, &output_path, None);
|
||||
|
||||
// There is extra stuff (the transcript) at the end of the parameter file which is
|
||||
// used to verify the parameter validity, but we're not interested in that. We do
|
||||
// want to read it, though, so that the BLAKE2b computed afterward is consistent
|
||||
// with `b2sum` on the files.
|
||||
let mut sink = io::sink();
|
||||
io::copy(&mut spend_fs, &mut sink)?;
|
||||
io::copy(&mut output_fs, &mut sink)?;
|
||||
|
||||
assert!(
|
||||
spend_fs.into_hash() == SAPLING_SPEND_HASH,
|
||||
"Sapling spend parameter is not correct."
|
||||
);
|
||||
assert!(
|
||||
output_fs.into_hash() == SAPLING_OUTPUT_HASH,
|
||||
"Sapling output parameter is not correct."
|
||||
);
|
||||
|
||||
// Prepare verifying keys
|
||||
let spend_prepared_verifying_key = groth16::prepare_verifying_key(&spend.vk);
|
||||
let output_prepared_verifying_key = groth16::prepare_verifying_key(&output.vk);
|
||||
|
||||
Ok(Self {
|
||||
spend,
|
||||
spend_prepared_verifying_key,
|
||||
output,
|
||||
output_prepared_verifying_key,
|
||||
})
|
||||
SaplingParameters {
|
||||
spend: parameters.spend_params,
|
||||
spend_prepared_verifying_key: parameters.spend_vk,
|
||||
output: parameters.output_params,
|
||||
output_prepared_verifying_key: parameters.output_vk,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -62,27 +62,32 @@ where
|
|||
#[tokio::test]
|
||||
async fn verify_sapling_groth16() {
|
||||
// Use separate verifiers so shared batch tasks aren't killed when the test ends (#2390)
|
||||
let mut spend_verifier = Fallback::new(
|
||||
Batch::new(
|
||||
Verifier::new(&PARAMS.sapling.spend.vk),
|
||||
crate::primitives::MAX_BATCH_SIZE,
|
||||
crate::primitives::MAX_BATCH_LATENCY,
|
||||
),
|
||||
tower::service_fn(
|
||||
(|item: Item| {
|
||||
ready(item.verify_single(&prepare_verifying_key(&PARAMS.sapling.spend.vk)))
|
||||
}) as fn(_) -> _,
|
||||
),
|
||||
);
|
||||
let mut spend_verifier =
|
||||
Fallback::new(
|
||||
Batch::new(
|
||||
Verifier::new(&GROTH16_PARAMETERS.sapling.spend.vk),
|
||||
crate::primitives::MAX_BATCH_SIZE,
|
||||
crate::primitives::MAX_BATCH_LATENCY,
|
||||
),
|
||||
tower::service_fn(
|
||||
(|item: Item| {
|
||||
ready(item.verify_single(&prepare_verifying_key(
|
||||
&GROTH16_PARAMETERS.sapling.spend.vk,
|
||||
)))
|
||||
}) as fn(_) -> _,
|
||||
),
|
||||
);
|
||||
let mut output_verifier = Fallback::new(
|
||||
Batch::new(
|
||||
Verifier::new(&PARAMS.sapling.output.vk),
|
||||
Verifier::new(&GROTH16_PARAMETERS.sapling.output.vk),
|
||||
crate::primitives::MAX_BATCH_SIZE,
|
||||
crate::primitives::MAX_BATCH_LATENCY,
|
||||
),
|
||||
tower::service_fn(
|
||||
(|item: Item| {
|
||||
ready(item.verify_single(&prepare_verifying_key(&PARAMS.sapling.output.vk)))
|
||||
ready(item.verify_single(&prepare_verifying_key(
|
||||
&GROTH16_PARAMETERS.sapling.output.vk,
|
||||
)))
|
||||
}) as fn(_) -> _,
|
||||
),
|
||||
);
|
||||
|
@ -152,13 +157,15 @@ async fn correctly_err_on_invalid_output_proof() {
|
|||
// Also, since we expect these to fail, we don't want to slow down the communal verifiers.
|
||||
let mut output_verifier = Fallback::new(
|
||||
Batch::new(
|
||||
Verifier::new(&PARAMS.sapling.output.vk),
|
||||
Verifier::new(&GROTH16_PARAMETERS.sapling.output.vk),
|
||||
crate::primitives::MAX_BATCH_SIZE,
|
||||
crate::primitives::MAX_BATCH_LATENCY,
|
||||
),
|
||||
tower::service_fn(
|
||||
(|item: Item| {
|
||||
ready(item.verify_single(&prepare_verifying_key(&PARAMS.sapling.output.vk)))
|
||||
ready(item.verify_single(&prepare_verifying_key(
|
||||
&GROTH16_PARAMETERS.sapling.output.vk,
|
||||
)))
|
||||
}) as fn(_) -> _,
|
||||
),
|
||||
);
|
||||
|
|
|
@ -342,6 +342,8 @@ where
|
|||
)?,
|
||||
};
|
||||
|
||||
// If the Groth16 parameter download hangs,
|
||||
// Zebra will timeout here, waiting for the async checks.
|
||||
async_checks.check().await?;
|
||||
|
||||
let mut spent_utxos = HashMap::new();
|
||||
|
|
|
@ -16,10 +16,11 @@ zebra-network = { path = "../zebra-network" }
|
|||
zebra-state = { path = "../zebra-state" }
|
||||
|
||||
abscissa_core = "0.5"
|
||||
chrono = "0.4"
|
||||
gumdrop = "0.7"
|
||||
lazy_static = "1.4.0"
|
||||
serde = { version = "1", features = ["serde_derive"] }
|
||||
toml = "0.5"
|
||||
chrono = "0.4"
|
||||
|
||||
hyper = { version = "0.14.15", features = ["full"] }
|
||||
futures = "0.3"
|
||||
|
|
|
@ -1,11 +1,12 @@
|
|||
//! Zebrad Subcommands
|
||||
|
||||
mod download;
|
||||
mod generate;
|
||||
mod start;
|
||||
mod version;
|
||||
|
||||
use self::ZebradCmd::*;
|
||||
use self::{generate::GenerateCmd, start::StartCmd, version::VersionCmd};
|
||||
use self::{download::DownloadCmd, generate::GenerateCmd, start::StartCmd, version::VersionCmd};
|
||||
|
||||
use crate::config::ZebradConfig;
|
||||
|
||||
|
@ -20,6 +21,10 @@ pub const CONFIG_FILE: &str = "zebrad.toml";
|
|||
/// Zebrad Subcommands
|
||||
#[derive(Command, Debug, Options)]
|
||||
pub enum ZebradCmd {
|
||||
/// The `download` subcommand
|
||||
#[options(help = "pre-download required parameter files")]
|
||||
Download(DownloadCmd),
|
||||
|
||||
/// The `generate` subcommand
|
||||
#[options(help = "generate a skeleton configuration")]
|
||||
Generate(GenerateCmd),
|
||||
|
@ -45,7 +50,7 @@ impl ZebradCmd {
|
|||
match self {
|
||||
// List all the commands, so new commands have to make a choice here
|
||||
Start(_) => true,
|
||||
Generate(_) | Help(_) | Version(_) => false,
|
||||
Download(_) | Generate(_) | Help(_) | Version(_) => false,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -53,6 +58,7 @@ impl ZebradCmd {
|
|||
impl Runnable for ZebradCmd {
|
||||
fn run(&self) {
|
||||
match self {
|
||||
Download(cmd) => cmd.run(),
|
||||
Generate(cmd) => cmd.run(),
|
||||
ZebradCmd::Help(cmd) => cmd.run(),
|
||||
Start(cmd) => cmd.run(),
|
||||
|
|
|
@ -0,0 +1,35 @@
|
|||
//! `download` subcommand - pre-download required parameter files
|
||||
//!
|
||||
//! `zebrad download` automatically downloads required paramter files the first time it is run.
|
||||
//!
|
||||
//! This command should be used if you're launching lots of `zebrad start` instances for testing,
|
||||
//! or you want to include the parameter files in a distribution package.
|
||||
|
||||
use abscissa_core::{Command, Options, Runnable};
|
||||
|
||||
/// `download` subcommand
|
||||
#[derive(Command, Debug, Default, Options)]
|
||||
pub struct DownloadCmd {}
|
||||
|
||||
impl DownloadCmd {
|
||||
/// Download the Sapling and Sprout Groth16 parameters if needed,
|
||||
/// check they were downloaded correctly, and load them into Zebra.
|
||||
///
|
||||
/// # Panics
|
||||
///
|
||||
/// If the downloaded or pre-existing parameter files are invalid.
|
||||
fn download_and_check(&self) {
|
||||
// The lazy static initializer does the download, if needed,
|
||||
// and the file hash checks.
|
||||
lazy_static::initialize(&zebra_consensus::groth16::GROTH16_PARAMETERS);
|
||||
}
|
||||
}
|
||||
|
||||
impl Runnable for DownloadCmd {
|
||||
/// Run the download command.
|
||||
fn run(&self) {
|
||||
info!("checking if Zcash Sapling and Sprout parameters have been downloaded");
|
||||
|
||||
self.download_and_check();
|
||||
}
|
||||
}
|
|
@ -50,8 +50,8 @@
|
|||
|
||||
use abscissa_core::{config, Command, FrameworkError, Options, Runnable};
|
||||
use color_eyre::eyre::{eyre, Report};
|
||||
use futures::{select, FutureExt};
|
||||
use tokio::sync::oneshot;
|
||||
use futures::FutureExt;
|
||||
use tokio::{pin, select, sync::oneshot};
|
||||
use tower::{builder::ServiceBuilder, util::BoxService};
|
||||
|
||||
use crate::{
|
||||
|
@ -79,19 +79,18 @@ impl StartCmd {
|
|||
info!(?config);
|
||||
|
||||
info!("initializing node state");
|
||||
// TODO: use ChainTipChange to get tip changes (#2374, #2710, #2711, #2712, #2713, #2714)
|
||||
let (state_service, latest_chain_tip, chain_tip_change) =
|
||||
zebra_state::init(config.state.clone(), config.network.network);
|
||||
let state = ServiceBuilder::new().buffer(20).service(state_service);
|
||||
|
||||
info!("initializing verifiers");
|
||||
// TODO: use the transaction verifier to verify mempool transactions (#2637, #2606)
|
||||
let (chain_verifier, tx_verifier) = zebra_consensus::chain::init(
|
||||
config.consensus.clone(),
|
||||
config.network.network,
|
||||
state.clone(),
|
||||
)
|
||||
.await;
|
||||
let (chain_verifier, tx_verifier, mut groth16_download_handle) =
|
||||
zebra_consensus::chain::init(
|
||||
config.consensus.clone(),
|
||||
config.network.network,
|
||||
state.clone(),
|
||||
)
|
||||
.await;
|
||||
|
||||
info!("initializing network");
|
||||
// The service that our node uses to respond to requests by peers. The
|
||||
|
@ -133,7 +132,7 @@ impl StartCmd {
|
|||
|
||||
let syncer_error_future = syncer.sync();
|
||||
|
||||
let sync_gossip_task_handle = tokio::spawn(sync::gossip_best_tip_block_hashes(
|
||||
let mut sync_gossip_task_handle = tokio::spawn(sync::gossip_best_tip_block_hashes(
|
||||
sync_status.clone(),
|
||||
chain_tip_change.clone(),
|
||||
peer_set.clone(),
|
||||
|
@ -154,25 +153,91 @@ impl StartCmd {
|
|||
peer_set,
|
||||
));
|
||||
|
||||
select! {
|
||||
sync_result = syncer_error_future.fuse() => sync_result,
|
||||
info!("started initial Zebra tasks");
|
||||
|
||||
sync_gossip_result = sync_gossip_task_handle.fuse() => sync_gossip_result
|
||||
.expect("unexpected panic in the chain tip block gossip task")
|
||||
.map_err(|e| eyre!(e)),
|
||||
// TODO: spawn the syncer task, after making the PeerSet sync and send
|
||||
// turn these tasks into a FuturesUnordered?
|
||||
|
||||
mempool_crawl_result = mempool_crawler_task_handle.fuse() => mempool_crawl_result
|
||||
.expect("unexpected panic in the mempool crawler")
|
||||
.map_err(|e| eyre!(e)),
|
||||
// ongoing futures & tasks
|
||||
pin!(syncer_error_future);
|
||||
pin!(mempool_crawler_task_handle);
|
||||
pin!(mempool_queue_checker_task_handle);
|
||||
pin!(tx_gossip_task_handle);
|
||||
|
||||
mempool_queue_result = mempool_queue_checker_task_handle.fuse() => mempool_queue_result
|
||||
.expect("unexpected panic in the mempool queue checker")
|
||||
.map_err(|e| eyre!(e)),
|
||||
// startup tasks
|
||||
let groth16_download_handle_fused = (&mut groth16_download_handle).fuse();
|
||||
pin!(groth16_download_handle_fused);
|
||||
|
||||
tx_gossip_result = tx_gossip_task_handle.fuse() => tx_gossip_result
|
||||
.expect("unexpected panic in the transaction gossip task")
|
||||
.map_err(|e| eyre!(e)),
|
||||
}
|
||||
// Wait for tasks to finish
|
||||
let exit_status = loop {
|
||||
let mut exit_when_task_finishes = true;
|
||||
|
||||
let result = select! {
|
||||
// We don't spawn the syncer future into a separate task yet.
|
||||
// So syncer panics automatically propagate to the main zebrad task.
|
||||
sync_result = &mut syncer_error_future => sync_result
|
||||
.map(|_| info!("syncer task exited")),
|
||||
|
||||
sync_gossip_result = &mut sync_gossip_task_handle => sync_gossip_result
|
||||
.expect("unexpected panic in the chain tip block gossip task")
|
||||
.map(|_| info!("chain tip block gossip task exited"))
|
||||
.map_err(|e| eyre!(e)),
|
||||
|
||||
mempool_crawl_result = &mut mempool_crawler_task_handle => mempool_crawl_result
|
||||
.expect("unexpected panic in the mempool crawler")
|
||||
.map(|_| info!("mempool crawler task exited"))
|
||||
.map_err(|e| eyre!(e)),
|
||||
|
||||
mempool_queue_result = &mut mempool_queue_checker_task_handle => mempool_queue_result
|
||||
.expect("unexpected panic in the mempool queue checker")
|
||||
.map(|_| info!("mempool queue checker task exited"))
|
||||
.map_err(|e| eyre!(e)),
|
||||
|
||||
tx_gossip_result = &mut tx_gossip_task_handle => tx_gossip_result
|
||||
.expect("unexpected panic in the transaction gossip task")
|
||||
.map(|_| info!("transaction gossip task exited"))
|
||||
.map_err(|e| eyre!(e)),
|
||||
|
||||
// Unlike other tasks, we expect the download task to finish while Zebra is running.
|
||||
groth16_download_result = &mut groth16_download_handle_fused => {
|
||||
groth16_download_result
|
||||
.unwrap_or_else(|_| panic!(
|
||||
"unexpected panic in the Groth16 pre-download and check task. {}",
|
||||
zebra_consensus::groth16::Groth16Parameters::failure_hint())
|
||||
);
|
||||
|
||||
info!("Groth16 pre-download and check task finished");
|
||||
exit_when_task_finishes = false;
|
||||
Ok(())
|
||||
}
|
||||
};
|
||||
|
||||
// Stop Zebra if a task finished and returned an error,
|
||||
// or if an ongoing task exited.
|
||||
if let Err(err) = result {
|
||||
break Err(err);
|
||||
}
|
||||
|
||||
if exit_when_task_finishes {
|
||||
break Ok(());
|
||||
}
|
||||
};
|
||||
|
||||
info!("exiting Zebra because an ongoing task exited: stopping other tasks");
|
||||
|
||||
// futures
|
||||
std::mem::drop(syncer_error_future);
|
||||
|
||||
// ongoing tasks
|
||||
sync_gossip_task_handle.abort();
|
||||
mempool_crawler_task_handle.abort();
|
||||
mempool_queue_checker_task_handle.abort();
|
||||
tx_gossip_task_handle.abort();
|
||||
|
||||
// startup tasks
|
||||
groth16_download_handle.abort();
|
||||
|
||||
exit_status
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -588,7 +588,8 @@ async fn setup(
|
|||
|
||||
let mut state_service = ServiceBuilder::new().buffer(1).service(state);
|
||||
|
||||
let (block_verifier, _transaction_verifier) =
|
||||
// Download task panics and timeouts are propagated to the tests that use Groth16 verifiers.
|
||||
let (block_verifier, _transaction_verifier, _groth16_download_handle) =
|
||||
zebra_consensus::chain::init(consensus_config.clone(), network, state_service.clone())
|
||||
.await;
|
||||
|
||||
|
|
Loading…
Reference in New Issue