diff --git a/.github/workflows/ci-coverage.yml b/.github/workflows/ci-coverage.yml index 1b87753b508..4a97563143b 100644 --- a/.github/workflows/ci-coverage.yml +++ b/.github/workflows/ci-coverage.yml @@ -103,4 +103,4 @@ jobs: run: cargo llvm-cov --lcov --no-run --output-path lcov.info - name: Upload coverage report to Codecov - uses: codecov/codecov-action@v5.0.7 + uses: codecov/codecov-action@v5.1.1 diff --git a/.github/workflows/ci-lint.yml b/.github/workflows/ci-lint.yml index 43acadbd8ec..26ec61089b3 100644 --- a/.github/workflows/ci-lint.yml +++ b/.github/workflows/ci-lint.yml @@ -44,7 +44,7 @@ jobs: - name: Rust files id: changed-files-rust - uses: tj-actions/changed-files@v45.0.4 + uses: tj-actions/changed-files@v45.0.5 with: files: | **/*.rs @@ -56,7 +56,7 @@ jobs: - name: Workflow files id: changed-files-workflows - uses: tj-actions/changed-files@v45.0.4 + uses: tj-actions/changed-files@v45.0.5 with: files: | .github/workflows/*.yml diff --git a/.github/workflows/sub-build-docker-image.yml b/.github/workflows/sub-build-docker-image.yml index 9050d223080..ac6d5bbbecc 100644 --- a/.github/workflows/sub-build-docker-image.yml +++ b/.github/workflows/sub-build-docker-image.yml @@ -152,7 +152,7 @@ jobs: # Setup Docker Buildx to use Docker Build Cloud - name: Set up Docker Buildx id: buildx - uses: docker/setup-buildx-action@v3.7.1 + uses: docker/setup-buildx-action@v3.8.0 with: version: "lab:latest" driver: cloud @@ -193,7 +193,7 @@ jobs: # - `dev` for a pull request event - name: Docker Scout id: docker-scout - uses: docker/scout-action@v1.15.1 + uses: docker/scout-action@v1.16.1 # We only run Docker Scout on the `runtime` target, as the other targets are not meant to be released # and are commonly used for testing, and thus are ephemeral. # TODO: Remove the `contains` check once we have a better way to determine if just new vulnerabilities are present. diff --git a/Cargo.lock b/Cargo.lock index 78a19264e9c..30273b8d3f2 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -78,7 +78,7 @@ version = "0.8.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b169f7a6d4742236a0a00c541b845991d0ac43e546831af1249753ab4c3aa3a0" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "cipher", "cpufeatures", ] @@ -89,7 +89,7 @@ version = "0.8.11" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e89da841a80418a9b391ebaea17f5c112ffaaa96f621d2c285b5174da76b9011" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "getrandom 0.2.15", "once_cell", "version_check", @@ -293,8 +293,8 @@ dependencies = [ "axum-core", "bytes", "futures-util", - "http 1.1.0", - "http-body 1.0.1", + "http", + "http-body", "http-body-util", "itoa", "matchit", @@ -319,8 +319,8 @@ dependencies = [ "async-trait", "bytes", "futures-util", - "http 1.1.0", - "http-body 1.0.1", + "http", + "http-body", "http-body-util", "mime", "pin-project-lite", @@ -338,7 +338,7 @@ checksum = "26b05800d2e817c8b3b4b54abd461726265fa9789ae34330622f2db9ee696f9d" dependencies = [ "addr2line", "cc", - "cfg-if 1.0.0", + "cfg-if", "libc", "miniz_oxide 0.7.4", "object", @@ -573,16 +573,6 @@ dependencies = [ "tinyvec", ] -[[package]] -name = "bstr" -version = "1.10.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "40723b8fb387abc38f4f4a37c09073622e41dd12327033091ef8950659e6dc0c" -dependencies = [ - "memchr", - "serde", -] - [[package]] name = "bumpalo" version = "3.16.0" @@ -697,12 +687,6 @@ dependencies = [ "nom", ] -[[package]] -name = "cfg-if" -version = "0.1.10" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4785bdd1c96b2a846b2bd7cc02e86b6b3dbf14e7e53446c4f54c92a361040822" - [[package]] name = "cfg-if" version = "1.0.0" @@ -721,7 +705,7 @@ version = "0.9.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c3613f74bd2eac03dad61bd53dbe620703d4371614fe0bc3b9f04dd36fe4e818" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "cipher", "cpufeatures", ] @@ -975,7 +959,7 @@ version = "1.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a97769d94ddab943e4510d138150169a2758b5ef3eb191a9ee688de3e23ef7b3" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", ] [[package]] @@ -1070,7 +1054,7 @@ version = "4.1.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "97fb8b7c4503de7d6ae7b42ab72a5a59857b4c937ec27a3d4539dba95b5ab2be" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "cpufeatures", "curve25519-dalek-derive", "digest", @@ -1538,7 +1522,7 @@ version = "0.1.16" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "libc", "wasi 0.9.0+wasi-snapshot-preview1", ] @@ -1549,7 +1533,7 @@ version = "0.2.15" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c4567c8db10ae91089c99af84c68c38da3ec2f087c3f82960bcdbf3656b6f4d7" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "js-sys", "libc", "wasi 0.11.0+wasi-snapshot-preview1", @@ -1581,19 +1565,6 @@ version = "0.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d2fabcfbdc87f4758337ca535fb41a6d701b65693ce38287d856d1674551ec9b" -[[package]] -name = "globset" -version = "0.4.15" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "15f1ce686646e7f1e19bf7d5533fe443a45dbfb990e00629110797578b42fb19" -dependencies = [ - "aho-corasick", - "bstr", - "log", - "regex-automata 0.4.8", - "regex-syntax 0.8.5", -] - [[package]] name = "group" version = "0.13.0" @@ -1617,11 +1588,11 @@ dependencies = [ "fnv", "futures-core", "futures-sink", - "http 1.1.0", + "http", "indexmap 2.7.0", "slab", "tokio", - "tokio-util 0.7.13", + "tokio-util", "tracing", ] @@ -1631,7 +1602,7 @@ version = "2.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6dd08c532ae367adf81c312a4580bc67f1d0fe8bc9c460520283f4c0ff277888" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "crunchy", ] @@ -1788,7 +1759,7 @@ version = "0.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f9c7c7c8ac16c798734b8a24560c1362120597c40d5e1459f09498f8f6c8f2ba" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "libc", "windows", ] @@ -1801,18 +1772,7 @@ checksum = "f34059280f617a59ee59a0455e93460d67e5c76dec42dd262d38f0f390f437b2" dependencies = [ "flume", "indicatif", - "parking_lot 0.12.3", -] - -[[package]] -name = "http" -version = "0.2.12" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "601cbb57e577e2f5ef5be8e7b83f0f63994f25aa94d673e54a92d5c516d101f1" -dependencies = [ - "bytes", - "fnv", - "itoa", + "parking_lot", ] [[package]] @@ -1826,17 +1786,6 @@ dependencies = [ "itoa", ] -[[package]] -name = "http-body" -version = "0.4.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7ceab25649e9960c0311ea418d17bee82c0dcec1bd053b5f9a66e265a693bed2" -dependencies = [ - "bytes", - "http 0.2.12", - "pin-project-lite", -] - [[package]] name = "http-body" version = "1.0.1" @@ -1844,7 +1793,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184" dependencies = [ "bytes", - "http 1.1.0", + "http", ] [[package]] @@ -1855,8 +1804,8 @@ checksum = "793429d76616a256bcb62c2a2ec2bed781c8307e797e2598c50010f2bee2544f" dependencies = [ "bytes", "futures-util", - "http 1.1.0", - "http-body 1.0.1", + "http", + "http-body", "pin-project-lite", ] @@ -1894,29 +1843,6 @@ dependencies = [ "serde", ] -[[package]] -name = "hyper" -version = "0.14.31" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8c08302e8fa335b151b788c775ff56e7a03ae64ff85c548ee820fecb70356e85" -dependencies = [ - "bytes", - "futures-channel", - "futures-core", - "futures-util", - "http 0.2.12", - "http-body 0.4.6", - "httparse", - "httpdate", - "itoa", - "pin-project-lite", - "socket2", - "tokio", - "tower-service", - "tracing", - "want", -] - [[package]] name = "hyper" version = "1.5.1" @@ -1927,8 +1853,8 @@ dependencies = [ "futures-channel", "futures-util", "h2", - "http 1.1.0", - "http-body 1.0.1", + "http", + "http-body", "httparse", "httpdate", "itoa", @@ -1945,8 +1871,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "08afdbb5c31130e3034af566421053ab03787c640246a446327f550d11bcb333" dependencies = [ "futures-util", - "http 1.1.0", - "hyper 1.5.1", + "http", + "hyper", "hyper-util", "rustls", "rustls-pki-types", @@ -1962,7 +1888,7 @@ version = "0.5.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3203a961e5c83b6f5498933e78b6b263e208c197b63e9c6c53cc82ffd3f63793" dependencies = [ - "hyper 1.5.1", + "hyper", "hyper-util", "pin-project-lite", "tokio", @@ -1978,9 +1904,9 @@ dependencies = [ "bytes", "futures-channel", "futures-util", - "http 1.1.0", - "http-body 1.0.1", - "hyper 1.5.1", + "http", + "http-body", + "hyper", "pin-project-lite", "socket2", "tokio", @@ -2138,15 +2064,6 @@ dependencies = [ "similar", ] -[[package]] -name = "instant" -version = "0.1.13" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e0242819d153cba4b4b05a5a8f2a7e9bbf97b6055b2a002b395c96b5ff3c0222" -dependencies = [ - "cfg-if 1.0.0", -] - [[package]] name = "ipnet" version = "2.10.1" @@ -2248,49 +2165,90 @@ dependencies = [ ] [[package]] -name = "jsonrpc-derive" -version = "18.0.0" +name = "jsonrpsee" +version = "0.24.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5b939a78fa820cdfcb7ee7484466746a7377760970f6f9c6fe19f9edcc8a38d2" +checksum = "c5c71d8c1a731cc4227c2f698d377e7848ca12c8a48866fc5e6951c43a4db843" dependencies = [ - "proc-macro-crate 0.1.5", - "proc-macro2", - "quote", - "syn 1.0.109", + "jsonrpsee-core", + "jsonrpsee-server", + "jsonrpsee-types", + "tokio", ] [[package]] -name = "jsonrpc-http-server" -version = "18.0.0" +name = "jsonrpsee-core" +version = "0.24.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e1dea6e07251d9ce6a552abfb5d7ad6bc290a4596c8dcc3d795fae2bbdc1f3ff" +checksum = "f2882f6f8acb9fdaec7cefc4fd607119a9bd709831df7d7672a1d3b644628280" dependencies = [ - "futures", - "hyper 0.14.31", - "jsonrpc-core", - "jsonrpc-server-utils", - "log", - "net2", - "parking_lot 0.11.2", - "unicase", + "async-trait", + "bytes", + "futures-util", + "http", + "http-body", + "http-body-util", + "jsonrpsee-types", + "parking_lot", + "rand 0.8.5", + "rustc-hash 2.0.0", + "serde", + "serde_json", + "thiserror 1.0.69", + "tokio", + "tracing", ] [[package]] -name = "jsonrpc-server-utils" -version = "18.0.0" +name = "jsonrpsee-proc-macros" +version = "0.24.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fa4fdea130485b572c39a460d50888beb00afb3e35de23ccd7fad8ff19f0e0d4" +checksum = "c06c01ae0007548e73412c08e2285ffe5d723195bf268bce67b1b77c3bb2a14d" dependencies = [ - "bytes", - "futures", - "globset", - "jsonrpc-core", - "lazy_static", - "log", + "heck 0.5.0", + "proc-macro-crate", + "proc-macro2", + "quote", + "syn 2.0.90", +] + +[[package]] +name = "jsonrpsee-server" +version = "0.24.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "82ad8ddc14be1d4290cd68046e7d1d37acd408efed6d3ca08aefcc3ad6da069c" +dependencies = [ + "futures-util", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-util", + "jsonrpsee-core", + "jsonrpsee-types", + "pin-project", + "route-recognizer", + "serde", + "serde_json", + "soketto", + "thiserror 1.0.69", "tokio", "tokio-stream", - "tokio-util 0.6.10", - "unicase", + "tokio-util", + "tower 0.4.13", + "tracing", +] + +[[package]] +name = "jsonrpsee-types" +version = "0.24.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a178c60086f24cc35bb82f57c651d0d25d99c4742b4d335de04e97fa1f08a8a1" +dependencies = [ + "http", + "serde", + "serde_json", + "thiserror 1.0.69", ] [[package]] @@ -2355,7 +2313,7 @@ version = "0.8.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4979f22fdb869068da03c9f7528f8297c6fd2606bc3a4affe42e6a823fdb8da4" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "windows-targets 0.52.6", ] @@ -2477,7 +2435,7 @@ version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8ea1f30cedd69f0a2954655f7188c6a834246d2bcf1e315e2ac40c4b24dc9519" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "rayon", ] @@ -2514,7 +2472,7 @@ checksum = "85b6f8152da6d7892ff1b7a1c0fa3f435e92b5918ad67035c3bb432111d9a29b" dependencies = [ "base64 0.22.1", "http-body-util", - "hyper 1.5.1", + "hyper", "hyper-util", "indexmap 2.7.0", "ipnet", @@ -2603,17 +2561,6 @@ dependencies = [ "getrandom 0.2.15", ] -[[package]] -name = "net2" -version = "0.2.39" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b13b648036a2339d06de780866fbdfda0dde886de7b3af2ddeba8b14f4ee34ac" -dependencies = [ - "cfg-if 0.1.10", - "libc", - "winapi", -] - [[package]] name = "nix" version = "0.29.0" @@ -2621,7 +2568,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "71e2746dc3a24dd78b3cfcb7be93368c6de9963d30f43a6a73998a9cf4b17b46" dependencies = [ "bitflags 2.6.0", - "cfg-if 1.0.0", + "cfg-if", "cfg_aliases", "libc", ] @@ -2854,23 +2801,12 @@ version = "3.6.12" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d830939c76d294956402033aee57a6da7b438f2294eb94864c37b0569053a42c" dependencies = [ - "proc-macro-crate 3.2.0", + "proc-macro-crate", "proc-macro2", "quote", "syn 1.0.109", ] -[[package]] -name = "parking_lot" -version = "0.11.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7d17b78036a60663b797adeaee46f5c9dfebb86948d1255007a1d6be0271ff99" -dependencies = [ - "instant", - "lock_api", - "parking_lot_core 0.8.6", -] - [[package]] name = "parking_lot" version = "0.12.3" @@ -2878,21 +2814,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f1bf18183cf54e8d6059647fc3063646a1801cf30896933ec2311622cc4b9a27" dependencies = [ "lock_api", - "parking_lot_core 0.9.10", -] - -[[package]] -name = "parking_lot_core" -version = "0.8.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "60a2cfe6f0ad2bfc16aefa463b497d5c7a5ecd44a23efa72aa342d90177356dc" -dependencies = [ - "cfg-if 1.0.0", - "instant", - "libc", - "redox_syscall 0.2.16", - "smallvec", - "winapi", + "parking_lot_core", ] [[package]] @@ -2901,9 +2823,9 @@ version = "0.9.10" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1e401f977ab385c9e4e3ab30627d6f26d00e2c73eef317493c4ec6d468726cf8" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "libc", - "redox_syscall 0.5.7", + "redox_syscall", "smallvec", "windows-targets 0.52.6", ] @@ -3113,15 +3035,6 @@ dependencies = [ "uint 0.9.5", ] -[[package]] -name = "proc-macro-crate" -version = "0.1.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1d6ea3c4595b96363c13943497db34af4460fb474a95c43f4446ad341b8c9785" -dependencies = [ - "toml 0.5.11", -] - [[package]] name = "proc-macro-crate" version = "3.2.0" @@ -3505,15 +3418,6 @@ dependencies = [ "zeroize", ] -[[package]] -name = "redox_syscall" -version = "0.2.16" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fb5a58c1855b4b6819d59012155603f0b22ad30cad752600aadfcb695265519a" -dependencies = [ - "bitflags 1.3.2", -] - [[package]] name = "redox_syscall" version = "0.5.7" @@ -3590,10 +3494,10 @@ dependencies = [ "futures-channel", "futures-core", "futures-util", - "http 1.1.0", - "http-body 1.0.1", + "http", + "http-body", "http-body-util", - "hyper 1.5.1", + "hyper", "hyper-rustls", "hyper-util", "ipnet", @@ -3613,7 +3517,7 @@ dependencies = [ "sync_wrapper 1.0.1", "tokio", "tokio-rustls", - "tokio-util 0.7.13", + "tokio-util", "tower-service", "url", "wasm-bindgen", @@ -3639,7 +3543,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c17fa4cb658e3583423e915b9f3acc01cceaee1860e33d59ebae66adc3a2dc0d" dependencies = [ "cc", - "cfg-if 1.0.0", + "cfg-if", "getrandom 0.2.15", "libc", "spin", @@ -3686,6 +3590,12 @@ dependencies = [ "serde", ] +[[package]] +name = "route-recognizer" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "afab94fb28594581f62d981211a9a4d53cc8130bbcbbb89a0440d9b8e81a7746" + [[package]] name = "rustc-demangle" version = "0.1.24" @@ -4076,13 +3986,24 @@ dependencies = [ "version_check", ] +[[package]] +name = "sha1" +version = "0.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3bf829a2d51ab4a5ddf1352d8470c140cadc8301b2ae1789db023f01cedd6ba" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + [[package]] name = "sha2" version = "0.10.8" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "cpufeatures", "digest", ] @@ -4169,6 +4090,22 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "soketto" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2e859df029d160cb88608f5d7df7fb4753fd20fdfb4de5644f3d8b8440841721" +dependencies = [ + "base64 0.22.1", + "bytes", + "futures", + "http", + "httparse", + "log", + "rand 0.8.5", + "sha1", +] + [[package]] name = "spandoc" version = "0.2.2" @@ -4325,7 +4262,7 @@ version = "3.14.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "28cce251fcbc87fac86a866eeb0d6c2d536fc16d06f184bb61aeae11aa4cee0c" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "fastrand", "once_cell", "rustix", @@ -4397,7 +4334,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cfe075d7053dae61ac5413a34ea7d4913b6e6207844fd726bdd858b37ff72bf5" dependencies = [ "bitflags 2.6.0", - "cfg-if 1.0.0", + "cfg-if", "libc", "log", "rustversion", @@ -4410,7 +4347,7 @@ version = "1.1.8" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8b9ef9bad013ada3808854ceac7b46812a6465ba368859a37e2100283d2d719c" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "once_cell", ] @@ -4482,7 +4419,7 @@ dependencies = [ "bytes", "libc", "mio", - "parking_lot 0.12.3", + "parking_lot", "pin-project-lite", "signal-hook-registry", "socket2", @@ -4522,7 +4459,7 @@ dependencies = [ "futures-core", "pin-project-lite", "tokio", - "tokio-util 0.7.13", + "tokio-util", ] [[package]] @@ -4538,20 +4475,6 @@ dependencies = [ "tokio-stream", ] -[[package]] -name = "tokio-util" -version = "0.6.10" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "36943ee01a6d67977dd3f84a5a1d2efeb4ada3a1ae771cadfaa535d9d9fc6507" -dependencies = [ - "bytes", - "futures-core", - "futures-sink", - "log", - "pin-project-lite", - "tokio", -] - [[package]] name = "tokio-util" version = "0.7.13" @@ -4560,6 +4483,7 @@ checksum = "d7fcaa8d55a2bdd6b83ace262b016eca0d79ee02818c5c1bcdf0305114081078" dependencies = [ "bytes", "futures-core", + "futures-io", "futures-sink", "pin-project-lite", "tokio", @@ -4620,10 +4544,10 @@ dependencies = [ "base64 0.22.1", "bytes", "h2", - "http 1.1.0", - "http-body 1.0.1", + "http", + "http-body", "http-body-util", - "hyper 1.5.1", + "hyper", "hyper-timeout", "hyper-util", "percent-encoding", @@ -4680,7 +4604,7 @@ dependencies = [ "rand 0.8.5", "slab", "tokio", - "tokio-util 0.7.13", + "tokio-util", "tower-layer", "tower-service", "tracing", @@ -4714,7 +4638,7 @@ dependencies = [ "tinyvec", "tokio", "tokio-test", - "tokio-util 0.7.13", + "tokio-util", "tower 0.4.13", "tower-fallback", "tower-test", @@ -4966,12 +4890,6 @@ version = "0.1.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "eaea85b334db583fe3274d12b4cd1880032beab409c0d774be044d4480ab9a94" -[[package]] -name = "unicase" -version = "2.8.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7e51b68083f157f853b6379db119d1c1be0e6e4dec98101079dec41f6f5cf6df" - [[package]] name = "unicode-bidi" version = "0.3.17" @@ -5101,7 +5019,7 @@ checksum = "2990d9ea5967266ea0ccf413a4aa5c42a93dbcfda9cb49a97de6931726b12566" dependencies = [ "anyhow", "cargo_metadata", - "cfg-if 1.0.0", + "cfg-if", "git2", "regex", "rustc_version", @@ -5228,7 +5146,7 @@ version = "0.2.95" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "128d1e363af62632b8eb57219c8fd7877144af57558fb2ef0368d0087bddeb2e" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "once_cell", "wasm-bindgen-macro", ] @@ -5254,7 +5172,7 @@ version = "0.4.45" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cc7ec4f8827a71586374db3e87abdb5a2bb3a15afed140221307c3ec06b1f63b" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "js-sys", "wasm-bindgen", "web-sys", @@ -5962,7 +5880,7 @@ dependencies = [ "thiserror 2.0.6", "tokio", "tokio-stream", - "tokio-util 0.7.13", + "tokio-util", "toml 0.8.19", "tower 0.4.13", "tracing", @@ -5994,11 +5912,13 @@ dependencies = [ "color-eyre", "futures", "hex", + "http-body-util", + "hyper", "indexmap 2.7.0", "insta", - "jsonrpc-core", - "jsonrpc-derive", - "jsonrpc-http-server", + "jsonrpsee", + "jsonrpsee-proc-macros", + "jsonrpsee-types", "nix", "proptest", "prost", @@ -6204,13 +6124,13 @@ dependencies = [ "howudoin", "http-body-util", "humantime-serde", - "hyper 1.5.1", + "hyper", "hyper-util", "indexmap 2.7.0", "indicatif", "inferno", "insta", - "jsonrpc-core", + "jsonrpsee-types", "lazy_static", "log", "metrics", diff --git a/deny.toml b/deny.toml index 7f804946767..3ae46206943 100644 --- a/deny.toml +++ b/deny.toml @@ -78,19 +78,8 @@ skip-tree = [ { name = "base64", version = "=0.21.7" }, { name = "sync_wrapper", version = "0.1.2" }, - # wait for jsonrpc-http-server to update hyper or for Zebra to replace jsonrpc (#8682) - { name = "h2", version = "=0.3.26" }, - { name = "http", version = "=0.2.12" }, - { name = "http-body", version = "=0.4.6" }, - { name = "hyper", version = "=0.14.31" }, - { name = "hyper-rustls", version = "=0.24.2" }, - - { name = "reqwest", version = "=0.11.27" }, - { name = "rustls", version = "=0.21.12" }, - { name = "rustls-pemfile", version = "=1.0.4" }, - { name = "rustls-webpki", version = "=0.101.7" }, - { name = "tokio-rustls", version = "=0.24.1" }, - { name = "webpki-roots", version = "=0.25.4" }, + # wait for abscissa_core to update toml + { name = "toml", version = "=0.5.11" }, # wait for structopt-derive to update heck { name = "heck", version = "=0.3.3" }, diff --git a/zebra-chain/src/block/arbitrary.rs b/zebra-chain/src/block/arbitrary.rs index 5a39afa2ee4..4961c873b3e 100644 --- a/zebra-chain/src/block/arbitrary.rs +++ b/zebra-chain/src/block/arbitrary.rs @@ -568,7 +568,7 @@ where + Copy + 'static, { - let mut spend_restriction = transaction.coinbase_spend_restriction(height); + let mut spend_restriction = transaction.coinbase_spend_restriction(&Network::Mainnet, height); let mut new_inputs = Vec::new(); let mut spent_outputs = HashMap::new(); @@ -650,7 +650,8 @@ where + 'static, { let has_shielded_outputs = transaction.has_shielded_outputs(); - let delete_transparent_outputs = CoinbaseSpendRestriction::OnlyShieldedOutputs { spend_height }; + let delete_transparent_outputs = + CoinbaseSpendRestriction::CheckCoinbaseMaturity { spend_height }; let mut attempts: usize = 0; // choose an arbitrary spendable UTXO, in hash set order diff --git a/zebra-chain/src/parameters/network/testnet.rs b/zebra-chain/src/parameters/network/testnet.rs index 78f7a69a302..1f77e95e750 100644 --- a/zebra-chain/src/parameters/network/testnet.rs +++ b/zebra-chain/src/parameters/network/testnet.rs @@ -232,6 +232,9 @@ pub struct ParametersBuilder { target_difficulty_limit: ExpandedDifficulty, /// A flag for disabling proof-of-work checks when Zebra is validating blocks disable_pow: bool, + /// Whether to allow transactions with transparent outputs to spend coinbase outputs, + /// similar to `fCoinbaseMustBeShielded` in zcashd. + should_allow_unshielded_coinbase_spends: bool, /// The pre-Blossom halving interval for this network pre_blossom_halving_interval: HeightDiff, /// The post-Blossom halving interval for this network @@ -271,6 +274,7 @@ impl Default for ParametersBuilder { should_lock_funding_stream_address_period: false, pre_blossom_halving_interval: PRE_BLOSSOM_HALVING_INTERVAL, post_blossom_halving_interval: POST_BLOSSOM_HALVING_INTERVAL, + should_allow_unshielded_coinbase_spends: false, } } } @@ -439,6 +443,15 @@ impl ParametersBuilder { self } + /// Sets the `disable_pow` flag to be used in the [`Parameters`] being built. + pub fn with_unshielded_coinbase_spends( + mut self, + should_allow_unshielded_coinbase_spends: bool, + ) -> Self { + self.should_allow_unshielded_coinbase_spends = should_allow_unshielded_coinbase_spends; + self + } + /// Sets the pre and post Blosssom halving intervals to be used in the [`Parameters`] being built. pub fn with_halving_interval(mut self, pre_blossom_halving_interval: HeightDiff) -> Self { if self.should_lock_funding_stream_address_period { @@ -464,6 +477,7 @@ impl ParametersBuilder { should_lock_funding_stream_address_period: _, target_difficulty_limit, disable_pow, + should_allow_unshielded_coinbase_spends, pre_blossom_halving_interval, post_blossom_halving_interval, } = self; @@ -478,6 +492,7 @@ impl ParametersBuilder { post_nu6_funding_streams, target_difficulty_limit, disable_pow, + should_allow_unshielded_coinbase_spends, pre_blossom_halving_interval, post_blossom_halving_interval, } @@ -516,6 +531,7 @@ impl ParametersBuilder { should_lock_funding_stream_address_period: _, target_difficulty_limit, disable_pow, + should_allow_unshielded_coinbase_spends, pre_blossom_halving_interval, post_blossom_halving_interval, } = Self::default(); @@ -528,6 +544,8 @@ impl ParametersBuilder { && self.post_nu6_funding_streams == post_nu6_funding_streams && self.target_difficulty_limit == target_difficulty_limit && self.disable_pow == disable_pow + && self.should_allow_unshielded_coinbase_spends + == should_allow_unshielded_coinbase_spends && self.pre_blossom_halving_interval == pre_blossom_halving_interval && self.post_blossom_halving_interval == post_blossom_halving_interval } @@ -560,6 +578,9 @@ pub struct Parameters { target_difficulty_limit: ExpandedDifficulty, /// A flag for disabling proof-of-work checks when Zebra is validating blocks disable_pow: bool, + /// Whether to allow transactions with transparent outputs to spend coinbase outputs, + /// similar to `fCoinbaseMustBeShielded` in zcashd. + should_allow_unshielded_coinbase_spends: bool, /// Pre-Blossom halving interval for this network pre_blossom_halving_interval: HeightDiff, /// Post-Blossom halving interval for this network @@ -597,6 +618,7 @@ impl Parameters { // This value is chosen to match zcashd, see: .with_target_difficulty_limit(U256::from_big_endian(&[0x0f; 32])) .with_disable_pow(true) + .with_unshielded_coinbase_spends(true) .with_slow_start_interval(Height::MIN) // Removes default Testnet activation heights if not configured, // most network upgrades are disabled by default for Regtest in zcashd @@ -645,6 +667,7 @@ impl Parameters { post_nu6_funding_streams, target_difficulty_limit, disable_pow, + should_allow_unshielded_coinbase_spends, pre_blossom_halving_interval, post_blossom_halving_interval, } = Self::new_regtest(None, None); @@ -657,6 +680,8 @@ impl Parameters { && self.post_nu6_funding_streams == post_nu6_funding_streams && self.target_difficulty_limit == target_difficulty_limit && self.disable_pow == disable_pow + && self.should_allow_unshielded_coinbase_spends + == should_allow_unshielded_coinbase_spends && self.pre_blossom_halving_interval == pre_blossom_halving_interval && self.post_blossom_halving_interval == post_blossom_halving_interval } @@ -711,6 +736,12 @@ impl Parameters { self.disable_pow } + /// Returns true if this network should allow transactions with transparent outputs + /// that spend coinbase outputs. + pub fn should_allow_unshielded_coinbase_spends(&self) -> bool { + self.should_allow_unshielded_coinbase_spends + } + /// Returns the pre-Blossom halving interval for this network pub fn pre_blossom_halving_interval(&self) -> HeightDiff { self.pre_blossom_halving_interval @@ -786,4 +817,14 @@ impl Network { self.post_nu6_funding_streams() } } + + /// Returns true if this network should allow transactions with transparent outputs + /// that spend coinbase outputs. + pub fn should_allow_unshielded_coinbase_spends(&self) -> bool { + if let Self::Testnet(params) = self { + params.should_allow_unshielded_coinbase_spends() + } else { + false + } + } } diff --git a/zebra-chain/src/transaction.rs b/zebra-chain/src/transaction.rs index c74f68134bb..c04f4155b4f 100644 --- a/zebra-chain/src/transaction.rs +++ b/zebra-chain/src/transaction.rs @@ -41,7 +41,7 @@ pub use unmined::{ use crate::{ amount::{Amount, Error as AmountError, NegativeAllowed, NonNegative}, block, orchard, - parameters::{ConsensusBranchId, NetworkUpgrade}, + parameters::{ConsensusBranchId, Network, NetworkUpgrade}, primitives::{ed25519, Bctv14Proof, Groth16Proof}, sapling, serialization::ZcashSerialize, @@ -308,14 +308,15 @@ impl Transaction { /// assuming it is mined at `spend_height`. pub fn coinbase_spend_restriction( &self, + network: &Network, spend_height: block::Height, ) -> CoinbaseSpendRestriction { - if self.outputs().is_empty() { - // we know this transaction must have shielded outputs, - // because of other consensus rules - OnlyShieldedOutputs { spend_height } + if self.outputs().is_empty() || network.should_allow_unshielded_coinbase_spends() { + // we know this transaction must have shielded outputs if it has no + // transparent outputs, because of other consensus rules. + CheckCoinbaseMaturity { spend_height } } else { - SomeTransparentOutputs + DisallowCoinbaseSpend } } diff --git a/zebra-chain/src/transaction/hash.rs b/zebra-chain/src/transaction/hash.rs index a7fa60066d2..98ba39692b7 100644 --- a/zebra-chain/src/transaction/hash.rs +++ b/zebra-chain/src/transaction/hash.rs @@ -34,7 +34,6 @@ use std::{fmt, sync::Arc}; use proptest_derive::Arbitrary; use hex::{FromHex, ToHex}; -use serde::{Deserialize, Serialize}; use crate::serialization::{ ReadZcashExt, SerializationError, WriteZcashExt, ZcashDeserialize, ZcashSerialize, @@ -56,7 +55,7 @@ use super::{txid::TxIdBuilder, AuthDigest, Transaction}; /// /// [ZIP-244]: https://zips.z.cash/zip-0244 /// [Spec: Transaction Identifiers]: https://zips.z.cash/protocol/protocol.pdf#txnidentifiers -#[derive(Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Serialize, Deserialize, Hash)] +#[derive(Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash)] #[cfg_attr(any(test, feature = "proptest-impl"), derive(Arbitrary))] pub struct Hash(pub [u8; 32]); diff --git a/zebra-chain/src/transparent/utxo.rs b/zebra-chain/src/transparent/utxo.rs index 0158165193a..1b5f49bcd89 100644 --- a/zebra-chain/src/transparent/utxo.rs +++ b/zebra-chain/src/transparent/utxo.rs @@ -126,10 +126,14 @@ impl OrderedUtxo { )] pub enum CoinbaseSpendRestriction { /// The UTXO is spent in a transaction with one or more transparent outputs - SomeTransparentOutputs, + /// on a network where coinbase outputs must not be spent by transactions + /// with transparent outputs. + DisallowCoinbaseSpend, - /// The UTXO is spent in a transaction which only has shielded outputs - OnlyShieldedOutputs { + /// The UTXO is spent in a transaction which only has shielded outputs, or + /// transactions spending coinbase outputs may have transparent outputs on + /// this network. + CheckCoinbaseMaturity { /// The height at which the UTXO is spent spend_height: block::Height, }, diff --git a/zebra-consensus/src/block.rs b/zebra-consensus/src/block.rs index 247079c401a..1c959dd284e 100644 --- a/zebra-consensus/src/block.rs +++ b/zebra-consensus/src/block.rs @@ -8,6 +8,7 @@ //! verification, where it may be accepted or rejected. use std::{ + collections::HashSet, future::Future, pin::Pin, sync::Arc, @@ -25,7 +26,7 @@ use zebra_chain::{ amount::Amount, block, parameters::{subsidy::FundingStreamReceiver, Network}, - transparent, + transaction, transparent, work::equihash, }; use zebra_state as zs; @@ -232,13 +233,21 @@ where &block, &transaction_hashes, )); - for transaction in &block.transactions { + + let known_outpoint_hashes: Arc> = + Arc::new(known_utxos.keys().map(|outpoint| outpoint.hash).collect()); + + for (&transaction_hash, transaction) in + transaction_hashes.iter().zip(block.transactions.iter()) + { let rsp = transaction_verifier .ready() .await .expect("transaction verifier is always ready") .call(tx::Request::Block { + transaction_hash, transaction: transaction.clone(), + known_outpoint_hashes: known_outpoint_hashes.clone(), known_utxos: known_utxos.clone(), height, time: block.header.time, diff --git a/zebra-consensus/src/transaction.rs b/zebra-consensus/src/transaction.rs index ef20881bbbf..044a9569f9a 100644 --- a/zebra-consensus/src/transaction.rs +++ b/zebra-consensus/src/transaction.rs @@ -1,7 +1,7 @@ //! Asynchronous verification of transactions. use std::{ - collections::HashMap, + collections::{HashMap, HashSet}, future::Future, pin::Pin, sync::Arc, @@ -146,8 +146,12 @@ where pub enum Request { /// Verify the supplied transaction as part of a block. Block { + /// The transaction hash. + transaction_hash: transaction::Hash, /// The transaction itself. transaction: Arc, + /// Set of transaction hashes that create new transparent outputs. + known_outpoint_hashes: Arc>, /// Additional UTXOs which are known at the time of verification. known_utxos: Arc>, /// The height of the block containing this transaction. @@ -259,6 +263,16 @@ impl Request { } } + /// The mined transaction ID for the transaction in this request. + pub fn tx_mined_id(&self) -> transaction::Hash { + match self { + Request::Block { + transaction_hash, .. + } => *transaction_hash, + Request::Mempool { transaction, .. } => transaction.id.mined_id(), + } + } + /// The set of additional known unspent transaction outputs that's in this request. pub fn known_utxos(&self) -> Arc> { match self { @@ -267,6 +281,17 @@ impl Request { } } + /// The set of additional known [`transparent::OutPoint`]s of unspent transaction outputs that's in this request. + pub fn known_outpoint_hashes(&self) -> Arc> { + match self { + Request::Block { + known_outpoint_hashes, + .. + } => known_outpoint_hashes.clone(), + Request::Mempool { .. } => HashSet::new().into(), + } + } + /// The height used to select the consensus rules for verifying this transaction. pub fn height(&self) -> block::Height { match self { @@ -377,6 +402,16 @@ where async move { tracing::trace!(?tx_id, ?req, "got tx verify request"); + if let Some(result) = Self::try_find_verified_unmined_tx(&req, mempool.clone()).await { + let verified_tx = result?; + + return Ok(Response::Block { + tx_id, + miner_fee: Some(verified_tx.miner_fee), + legacy_sigop_count: verified_tx.legacy_sigop_count + }); + } + // Do quick checks first check::has_inputs_and_outputs(&tx)?; check::has_enough_orchard_flags(&tx)?; @@ -451,7 +486,7 @@ where // WONTFIX: Return an error for Request::Block as well to replace this check in // the state once #2336 has been implemented? if req.is_mempool() { - Self::check_maturity_height(&req, &spent_utxos)?; + Self::check_maturity_height(&network, &req, &spent_utxos)?; } let cached_ffi_transaction = @@ -609,8 +644,52 @@ where } } - /// Waits for the UTXOs that are being spent by the given transaction to arrive in - /// the state for [`Block`](Request::Block) requests. + /// Attempts to find a transaction in the mempool by its transaction hash and checks + /// that all of its dependencies are available in the block. + /// + /// Returns [`Some(Ok(VerifiedUnminedTx))`](VerifiedUnminedTx) if successful, + /// None if the transaction id was not found in the mempool, + /// or `Some(Err(TransparentInputNotFound))` if the transaction was found, but some of its + /// dependencies are missing in the block. + async fn try_find_verified_unmined_tx( + req: &Request, + mempool: Option>, + ) -> Option> { + if req.is_mempool() || req.transaction().is_coinbase() { + return None; + } + + let mempool = mempool?; + let known_outpoint_hashes = req.known_outpoint_hashes(); + let tx_id = req.tx_mined_id(); + + let mempool::Response::TransactionWithDeps { + transaction, + dependencies, + } = mempool + .oneshot(mempool::Request::TransactionWithDepsByMinedId(tx_id)) + .await + .ok()? + else { + panic!("unexpected response to TransactionWithDepsByMinedId request"); + }; + + // Note: This does not verify that the spends are in order, the spend order + // should be verified during contextual validation in zebra-state. + let has_all_tx_deps = dependencies + .into_iter() + .all(|dependency_id| known_outpoint_hashes.contains(&dependency_id)); + + let result = if has_all_tx_deps { + Ok(transaction) + } else { + Err(TransactionError::TransparentInputNotFound) + }; + + Some(result) + } + + /// Wait for the UTXOs that are being spent by the given transaction. /// /// Looks up UTXOs that are being spent by the given transaction in the state or waits /// for them to be added to the mempool for [`Mempool`](Request::Mempool) requests. @@ -728,10 +807,12 @@ where /// mature and valid for the request height, or a [`TransactionError`] if the transaction /// spends transparent coinbase outputs that are immature and invalid for the request height. pub fn check_maturity_height( + network: &Network, request: &Request, spent_utxos: &HashMap, ) -> Result<(), TransactionError> { check::tx_transparent_coinbase_spends_maturity( + network, request.transaction(), request.height(), request.known_utxos(), diff --git a/zebra-consensus/src/transaction/check.rs b/zebra-consensus/src/transaction/check.rs index d3ddc460264..b7338bbdadd 100644 --- a/zebra-consensus/src/transaction/check.rs +++ b/zebra-consensus/src/transaction/check.rs @@ -476,6 +476,7 @@ fn validate_expiry_height_mined( /// Returns `Ok(())` if spent transparent coinbase outputs are /// valid for the block height, or a [`Err(TransactionError)`](TransactionError) pub fn tx_transparent_coinbase_spends_maturity( + network: &Network, tx: Arc, height: Height, block_new_outputs: Arc>, @@ -488,7 +489,7 @@ pub fn tx_transparent_coinbase_spends_maturity( .or_else(|| spent_utxos.get(&spend).cloned()) .expect("load_spent_utxos_fut.await should return an error if a utxo is missing"); - let spend_restriction = tx.coinbase_spend_restriction(height); + let spend_restriction = tx.coinbase_spend_restriction(network, height); zebra_state::check::transparent_coinbase_spend(spend, spend_restriction, &utxo)?; } diff --git a/zebra-consensus/src/transaction/tests.rs b/zebra-consensus/src/transaction/tests.rs index 9b3c7c61e40..f44c3a5734d 100644 --- a/zebra-consensus/src/transaction/tests.rs +++ b/zebra-consensus/src/transaction/tests.rs @@ -2,7 +2,10 @@ // // TODO: split fixed test vectors into a `vectors` module? -use std::{collections::HashMap, sync::Arc}; +use std::{ + collections::{HashMap, HashSet}, + sync::Arc, +}; use chrono::{DateTime, TimeZone, Utc}; use color_eyre::eyre::Report; @@ -26,7 +29,7 @@ use zebra_chain::{ }, zip317, Hash, HashType, JoinSplitData, LockTime, Transaction, }, - transparent::{self, CoinbaseData}, + transparent::{self, CoinbaseData, CoinbaseSpendRestriction}, }; use zebra_node_services::mempool; @@ -700,13 +703,180 @@ async fn mempool_request_with_unmined_output_spends_is_accepted() { ); tokio::time::sleep(POLL_MEMPOOL_DELAY * 2).await; + // polled before AwaitOutput request and after a mempool transaction with transparent outputs + // is successfully verified assert_eq!( mempool.poll_count(), 2, - "the mempool service should have been polled twice, \ - first before being called with an AwaitOutput request, \ - then again shortly after a mempool transaction with transparent outputs \ - is successfully verified" + "the mempool service should have been polled twice" + ); +} + +#[tokio::test] +async fn skips_verification_of_block_transactions_in_mempool() { + let mut state: MockService<_, _, _, _> = MockService::build().for_prop_tests(); + let mempool: MockService<_, _, _, _> = MockService::build().for_prop_tests(); + let (mempool_setup_tx, mempool_setup_rx) = tokio::sync::oneshot::channel(); + let verifier = Verifier::new(&Network::Mainnet, state.clone(), mempool_setup_rx); + let verifier = Buffer::new(verifier, 1); + + mempool_setup_tx + .send(mempool.clone()) + .ok() + .expect("send should succeed"); + + let height = NetworkUpgrade::Nu6 + .activation_height(&Network::Mainnet) + .expect("Canopy activation height is specified"); + let fund_height = (height - 1).expect("fake source fund block height is too small"); + let (input, output, known_utxos) = mock_transparent_transfer( + fund_height, + true, + 0, + Amount::try_from(10001).expect("invalid value"), + ); + + // Create a non-coinbase V4 tx with the last valid expiry height. + let tx = Transaction::V5 { + network_upgrade: NetworkUpgrade::Nu6, + inputs: vec![input], + outputs: vec![output], + lock_time: LockTime::min_lock_time_timestamp(), + expiry_height: height, + sapling_shielded_data: None, + orchard_shielded_data: None, + }; + + let tx_hash = tx.hash(); + let input_outpoint = match tx.inputs()[0] { + transparent::Input::PrevOut { outpoint, .. } => outpoint, + transparent::Input::Coinbase { .. } => panic!("requires a non-coinbase transaction"), + }; + + tokio::spawn(async move { + state + .expect_request(zebra_state::Request::BestChainNextMedianTimePast) + .await + .expect("verifier should call mock state service with correct request") + .respond(zebra_state::Response::BestChainNextMedianTimePast( + DateTime32::MAX, + )); + + state + .expect_request(zebra_state::Request::UnspentBestChainUtxo(input_outpoint)) + .await + .expect("verifier should call mock state service with correct request") + .respond(zebra_state::Response::UnspentBestChainUtxo(None)); + + state + .expect_request_that(|req| { + matches!( + req, + zebra_state::Request::CheckBestChainTipNullifiersAndAnchors(_) + ) + }) + .await + .expect("verifier should call mock state service with correct request") + .respond(zebra_state::Response::ValidBestChainTipNullifiersAndAnchors); + }); + + let mut mempool_clone = mempool.clone(); + tokio::spawn(async move { + mempool_clone + .expect_request(mempool::Request::AwaitOutput(input_outpoint)) + .await + .expect("verifier should call mock state service with correct request") + .respond(mempool::Response::UnspentOutput( + known_utxos + .get(&input_outpoint) + .expect("input outpoint should exist in known_utxos") + .utxo + .output + .clone(), + )); + }); + + let verifier_response = verifier + .clone() + .oneshot(Request::Mempool { + transaction: tx.clone().into(), + height, + }) + .await; + + assert!( + verifier_response.is_ok(), + "expected successful verification, got: {verifier_response:?}" + ); + + let crate::transaction::Response::Mempool { + transaction, + spent_mempool_outpoints, + } = verifier_response.expect("already checked that response is ok") + else { + panic!("unexpected response variant from transaction verifier for Mempool request") + }; + + assert_eq!( + spent_mempool_outpoints, + vec![input_outpoint], + "spent_mempool_outpoints in tx verifier response should match input_outpoint" + ); + + let mut mempool_clone = mempool.clone(); + tokio::spawn(async move { + for _ in 0..2 { + mempool_clone + .expect_request(mempool::Request::TransactionWithDepsByMinedId(tx_hash)) + .await + .expect("verifier should call mock state service with correct request") + .respond(mempool::Response::TransactionWithDeps { + transaction: transaction.clone(), + dependencies: [input_outpoint.hash].into(), + }); + } + }); + + let make_request = |known_outpoint_hashes| Request::Block { + transaction_hash: tx_hash, + transaction: Arc::new(tx), + known_outpoint_hashes, + known_utxos: Arc::new(HashMap::new()), + height, + time: Utc::now(), + }; + + let crate::transaction::Response::Block { .. } = verifier + .clone() + .oneshot(make_request.clone()(Arc::new([input_outpoint.hash].into()))) + .await + .expect("should return Ok without calling state service") + else { + panic!("unexpected response variant from transaction verifier for Block request") + }; + + let verifier_response_err = *verifier + .clone() + .oneshot(make_request(Arc::new(HashSet::new()))) + .await + .expect_err("should return Err without calling state service") + .downcast::() + .expect("tx verifier error type should be TransactionError"); + + assert_eq!( + verifier_response_err, + TransactionError::TransparentInputNotFound, + "should be a transparent input not found error" + ); + + tokio::time::sleep(POLL_MEMPOOL_DELAY * 2).await; + // polled before AwaitOutput request, after a mempool transaction with transparent outputs, + // is successfully verified, and twice more when checking if a transaction in a block is + // already the mempool. + assert_eq!( + mempool.poll_count(), + 4, + "the mempool service should have been polled 4 times" ); } @@ -745,7 +915,7 @@ async fn mempool_request_with_immature_spend_is_rejected() { transparent::Input::Coinbase { .. } => panic!("requires a non-coinbase transaction"), }; - let spend_restriction = tx.coinbase_spend_restriction(height); + let spend_restriction = tx.coinbase_spend_restriction(&Network::Mainnet, height); let coinbase_spend_height = Height(5); @@ -813,6 +983,100 @@ async fn mempool_request_with_immature_spend_is_rejected() { ); } +/// Tests that calls to the transaction verifier with a mempool request that spends +/// mature coinbase outputs to transparent outputs will return Ok() on Regtest. +#[tokio::test] +async fn mempool_request_with_transparent_coinbase_spend_is_accepted_on_regtest() { + let _init_guard = zebra_test::init(); + + let network = Network::new_regtest(None, Some(1_000)); + let mut state: MockService<_, _, _, _> = MockService::build().for_unit_tests(); + let verifier = Verifier::new_for_tests(&network, state.clone()); + + let height = NetworkUpgrade::Nu6 + .activation_height(&network) + .expect("NU6 activation height is specified"); + let fund_height = (height - 1).expect("fake source fund block height is too small"); + let (input, output, known_utxos) = mock_transparent_transfer( + fund_height, + true, + 0, + Amount::try_from(10001).expect("invalid value"), + ); + + // Create a non-coinbase V5 tx with the last valid expiry height. + let tx = Transaction::V5 { + network_upgrade: NetworkUpgrade::Nu6, + inputs: vec![input], + outputs: vec![output], + lock_time: LockTime::min_lock_time_timestamp(), + expiry_height: height, + sapling_shielded_data: None, + orchard_shielded_data: None, + }; + + let input_outpoint = match tx.inputs()[0] { + transparent::Input::PrevOut { outpoint, .. } => outpoint, + transparent::Input::Coinbase { .. } => panic!("requires a non-coinbase transaction"), + }; + + let spend_restriction = tx.coinbase_spend_restriction(&network, height); + + assert_eq!( + spend_restriction, + CoinbaseSpendRestriction::CheckCoinbaseMaturity { + spend_height: height + } + ); + + let coinbase_spend_height = Height(5); + + let utxo = known_utxos + .get(&input_outpoint) + .map(|utxo| { + let mut utxo = utxo.utxo.clone(); + utxo.height = coinbase_spend_height; + utxo.from_coinbase = true; + utxo + }) + .expect("known_utxos should contain the outpoint"); + + zebra_state::check::transparent_coinbase_spend(input_outpoint, spend_restriction, &utxo) + .expect("check should pass"); + + tokio::spawn(async move { + state + .expect_request(zebra_state::Request::BestChainNextMedianTimePast) + .await + .respond(zebra_state::Response::BestChainNextMedianTimePast( + DateTime32::MAX, + )); + + state + .expect_request(zebra_state::Request::UnspentBestChainUtxo(input_outpoint)) + .await + .respond(zebra_state::Response::UnspentBestChainUtxo(Some(utxo))); + + state + .expect_request_that(|req| { + matches!( + req, + zebra_state::Request::CheckBestChainTipNullifiersAndAnchors(_) + ) + }) + .await + .respond(zebra_state::Response::ValidBestChainTipNullifiersAndAnchors); + }); + + verifier + .oneshot(Request::Mempool { + transaction: tx.into(), + height, + }) + .await + .expect("verification of transaction with mature spend to transparent outputs should pass"); +} + /// Tests that errors from the read state service are correctly converted into /// transaction verifier errors. #[tokio::test] @@ -952,7 +1216,7 @@ fn v5_coinbase_transaction_with_enable_spends_flag_fails_validation() { #[tokio::test] async fn v5_transaction_is_rejected_before_nu5_activation() { - let canopy = NetworkUpgrade::Canopy; + let sapling = NetworkUpgrade::Sapling; for net in Network::iter() { let verifier = Verifier::new_for_tests( @@ -960,16 +1224,20 @@ async fn v5_transaction_is_rejected_before_nu5_activation() { service_fn(|_| async { unreachable!("Service should not be called") }), ); + let tx = v5_transactions(net.block_iter()).next().expect("V5 tx"); + assert_eq!( verifier .oneshot(Request::Block { - transaction: Arc::new(v5_transactions(net.block_iter()).next().expect("V5 tx")), + transaction_hash: tx.hash(), + transaction: Arc::new(tx), known_utxos: Arc::new(HashMap::new()), - height: canopy.activation_height(&net).expect("height"), + known_outpoint_hashes: Arc::new(HashSet::new()), + height: sapling.activation_height(&net).expect("height"), time: DateTime::::MAX_UTC, }) .await, - Err(TransactionError::UnsupportedByNetworkUpgrade(5, canopy)) + Err(TransactionError::UnsupportedByNetworkUpgrade(5, sapling)) ); } } @@ -988,8 +1256,10 @@ async fn v5_transaction_is_accepted_after_nu5_activation() { let verif_res = Verifier::new_for_tests(&net, state) .oneshot(Request::Block { + transaction_hash: tx.hash(), transaction: Arc::new(tx), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height: tx_height, time: DateTime::::MAX_UTC, }) @@ -1040,8 +1310,10 @@ async fn v4_transaction_with_transparent_transfer_is_accepted() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: transaction_block_height, time: DateTime::::MAX_UTC, }) @@ -1084,8 +1356,10 @@ async fn v4_transaction_with_last_valid_expiry_height() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction.clone()), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: block_height, time: DateTime::::MAX_UTC, }) @@ -1129,8 +1403,10 @@ async fn v4_coinbase_transaction_with_low_expiry_height() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction.clone()), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height: block_height, time: DateTime::::MAX_UTC, }) @@ -1176,8 +1452,10 @@ async fn v4_transaction_with_too_low_expiry_height() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction.clone()), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: block_height, time: DateTime::::MAX_UTC, }) @@ -1226,8 +1504,10 @@ async fn v4_transaction_with_exceeding_expiry_height() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction.clone()), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: block_height, time: DateTime::::MAX_UTC, }) @@ -1279,8 +1559,10 @@ async fn v4_coinbase_transaction_with_exceeding_expiry_height() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction.clone()), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height: block_height, time: DateTime::::MAX_UTC, }) @@ -1330,8 +1612,10 @@ async fn v4_coinbase_transaction_is_accepted() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height: transaction_block_height, time: DateTime::::MAX_UTC, }) @@ -1385,8 +1669,10 @@ async fn v4_transaction_with_transparent_transfer_is_rejected_by_the_script() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: transaction_block_height, time: DateTime::::MAX_UTC, }) @@ -1440,8 +1726,10 @@ async fn v4_transaction_with_conflicting_transparent_spend_is_rejected() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: transaction_block_height, time: DateTime::::MAX_UTC, }) @@ -1511,8 +1799,10 @@ fn v4_transaction_with_conflicting_sprout_nullifier_inside_joinsplit_is_rejected let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height: transaction_block_height, time: DateTime::::MAX_UTC, }) @@ -1587,8 +1877,10 @@ fn v4_transaction_with_conflicting_sprout_nullifier_across_joinsplits_is_rejecte let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height: transaction_block_height, time: DateTime::::MAX_UTC, }) @@ -1646,8 +1938,10 @@ async fn v5_transaction_with_transparent_transfer_is_accepted() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: transaction_block_height, time: DateTime::::MAX_UTC, }) @@ -1692,8 +1986,10 @@ async fn v5_transaction_with_last_valid_expiry_height() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction.clone()), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: block_height, time: DateTime::::MAX_UTC, }) @@ -1737,8 +2033,10 @@ async fn v5_coinbase_transaction_expiry_height() { let result = verifier .clone() .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction.clone()), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height: block_height, time: DateTime::::MAX_UTC, }) @@ -1758,8 +2056,10 @@ async fn v5_coinbase_transaction_expiry_height() { let result = verifier .clone() .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(new_transaction.clone()), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height: block_height, time: DateTime::::MAX_UTC, }) @@ -1787,8 +2087,10 @@ async fn v5_coinbase_transaction_expiry_height() { let result = verifier .clone() .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(new_transaction.clone()), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height: block_height, time: DateTime::::MAX_UTC, }) @@ -1824,8 +2126,10 @@ async fn v5_coinbase_transaction_expiry_height() { let verification_result = verifier .clone() .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(new_transaction.clone()), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height: new_expiry_height, time: DateTime::::MAX_UTC, }) @@ -1875,8 +2179,10 @@ async fn v5_transaction_with_too_low_expiry_height() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction.clone()), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: block_height, time: DateTime::::MAX_UTC, }) @@ -1924,8 +2230,10 @@ async fn v5_transaction_with_exceeding_expiry_height() { let verification_result = Verifier::new_for_tests(&Network::Mainnet, state) .oneshot(Request::Block { - transaction: Arc::new(transaction), + transaction_hash: transaction.hash(), + transaction: Arc::new(transaction.clone()), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: height_max, time: DateTime::::MAX_UTC, }) @@ -1978,8 +2286,10 @@ async fn v5_coinbase_transaction_is_accepted() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: transaction_block_height, time: DateTime::::MAX_UTC, }) @@ -2035,8 +2345,10 @@ async fn v5_transaction_with_transparent_transfer_is_rejected_by_the_script() { let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height: transaction_block_height, time: DateTime::::MAX_UTC, }) @@ -2083,8 +2395,10 @@ async fn v5_transaction_with_conflicting_transparent_spend_is_rejected() { let verification_result = Verifier::new_for_tests(&network, state) .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: Arc::new(transaction), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height, time: DateTime::::MAX_UTC, }) @@ -2126,8 +2440,10 @@ fn v4_with_signed_sprout_transfer_is_accepted() { // Test the transaction verifier let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction, known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height, time: DateTime::::MAX_UTC, }) @@ -2215,8 +2531,10 @@ async fn v4_with_joinsplit_is_rejected_for_modification( let result = verifier .clone() .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction: transaction.clone(), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height, time: DateTime::::MAX_UTC, }) @@ -2261,8 +2579,10 @@ fn v4_with_sapling_spends() { // Test the transaction verifier let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction, known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height, time: DateTime::::MAX_UTC, }) @@ -2303,8 +2623,10 @@ fn v4_with_duplicate_sapling_spends() { // Test the transaction verifier let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction, known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height, time: DateTime::::MAX_UTC, }) @@ -2347,8 +2669,10 @@ fn v4_with_sapling_outputs_and_no_spends() { // Test the transaction verifier let result = verifier .oneshot(Request::Block { + transaction_hash: transaction.hash(), transaction, known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height, time: DateTime::::MAX_UTC, }) @@ -2387,8 +2711,10 @@ async fn v5_with_sapling_spends() { assert_eq!( verifier .oneshot(Request::Block { + transaction_hash: tx.hash(), transaction: Arc::new(tx), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height, time: DateTime::::MAX_UTC, }) @@ -2424,8 +2750,10 @@ async fn v5_with_duplicate_sapling_spends() { assert_eq!( verifier .oneshot(Request::Block { + transaction_hash: tx.hash(), transaction: Arc::new(tx), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height, time: DateTime::::MAX_UTC, }) @@ -2475,8 +2803,10 @@ async fn v5_with_duplicate_orchard_action() { assert_eq!( verifier .oneshot(Request::Block { + transaction_hash: tx.hash(), transaction: Arc::new(tx), known_utxos: Arc::new(HashMap::new()), + known_outpoint_hashes: Arc::new(HashSet::new()), height, time: DateTime::::MAX_UTC, }) @@ -2530,8 +2860,10 @@ async fn v5_consensus_branch_ids() { let block_req = verifier .clone() .oneshot(Request::Block { + transaction_hash: tx.hash(), transaction: Arc::new(tx.clone()), known_utxos: known_utxos.clone(), + known_outpoint_hashes: Arc::new(HashSet::new()), // The consensus branch ID of the tx is outdated for this height. height, time: DateTime::::MAX_UTC, @@ -2558,8 +2890,10 @@ async fn v5_consensus_branch_ids() { let block_req = verifier .clone() .oneshot(Request::Block { + transaction_hash: tx.hash(), transaction: Arc::new(tx.clone()), known_utxos: known_utxos.clone(), + known_outpoint_hashes: Arc::new(HashSet::new()), // The consensus branch ID of the tx is supported by this height. height, time: DateTime::::MAX_UTC, @@ -2615,8 +2949,10 @@ async fn v5_consensus_branch_ids() { let block_req = verifier .clone() .oneshot(Request::Block { + transaction_hash: tx.hash(), transaction: Arc::new(tx.clone()), known_utxos: known_utxos.clone(), + known_outpoint_hashes: Arc::new(HashSet::new()), // The consensus branch ID of the tx is not supported by this height. height, time: DateTime::::MAX_UTC, diff --git a/zebra-consensus/src/transaction/tests/prop.rs b/zebra-consensus/src/transaction/tests/prop.rs index 856742e5d74..8fea9cf3433 100644 --- a/zebra-consensus/src/transaction/tests/prop.rs +++ b/zebra-consensus/src/transaction/tests/prop.rs @@ -1,6 +1,9 @@ //! Randomised property tests for transaction verification. -use std::{collections::HashMap, sync::Arc}; +use std::{ + collections::{HashMap, HashSet}, + sync::Arc, +}; use chrono::{DateTime, Duration, Utc}; use proptest::{collection::vec, prelude::*}; @@ -452,13 +455,16 @@ fn validate( tower::service_fn(|_| async { unreachable!("State service should not be called") }); let verifier = transaction::Verifier::new_for_tests(&network, state_service); let verifier = Buffer::new(verifier, 10); + let transaction_hash = transaction.hash(); // Test the transaction verifier verifier .clone() .oneshot(transaction::Request::Block { + transaction_hash, transaction: Arc::new(transaction), known_utxos: Arc::new(known_utxos), + known_outpoint_hashes: Arc::new(HashSet::new()), height, time: block_time, }) diff --git a/zebra-node-services/src/mempool.rs b/zebra-node-services/src/mempool.rs index ad3e28c7eec..6c035e6dc44 100644 --- a/zebra-node-services/src/mempool.rs +++ b/zebra-node-services/src/mempool.rs @@ -6,13 +6,10 @@ use std::collections::HashSet; use tokio::sync::oneshot; use zebra_chain::{ - transaction::{self, UnminedTx, UnminedTxId}, + transaction::{self, UnminedTx, UnminedTxId, VerifiedUnminedTx}, transparent, }; -#[cfg(feature = "getblocktemplate-rpcs")] -use zebra_chain::transaction::VerifiedUnminedTx; - use crate::BoxError; mod gossip; @@ -58,6 +55,9 @@ pub enum Request { /// Outdated requests are pruned on a regular basis. AwaitOutput(transparent::OutPoint), + /// Request a [`VerifiedUnminedTx`] and its dependencies by its mined id. + TransactionWithDepsByMinedId(transaction::Hash), + /// Get all the [`VerifiedUnminedTx`] in the mempool. /// /// Equivalent to `TransactionsById(TransactionIds)`, @@ -121,6 +121,14 @@ pub enum Response { /// Response to [`Request::AwaitOutput`] with the transparent output UnspentOutput(transparent::Output), + /// Response to [`Request::TransactionWithDepsByMinedId`]. + TransactionWithDeps { + /// The queried transaction + transaction: VerifiedUnminedTx, + /// A list of dependencies of the queried transaction. + dependencies: HashSet, + }, + /// Returns all [`VerifiedUnminedTx`] in the mempool. // // TODO: make the Transactions response return VerifiedUnminedTx, diff --git a/zebra-rpc/Cargo.toml b/zebra-rpc/Cargo.toml index 56b7f3c60f0..d180f049dc5 100644 --- a/zebra-rpc/Cargo.toml +++ b/zebra-rpc/Cargo.toml @@ -59,9 +59,11 @@ chrono = { version = "0.4.39", default-features = false, features = [ ] } futures = "0.3.31" -jsonrpc-core = "18.0.0" -jsonrpc-derive = "18.0.0" -jsonrpc-http-server = "18.0.0" +jsonrpsee = { version = "0.24.7", features = ["server"] } +jsonrpsee-types = "0.24.7" +jsonrpsee-proc-macros = "0.24.7" +hyper = "1.5.0" +http-body-util = "0.1.2" # zebra-rpc needs the preserve_order feature in serde_json, which is a dependency of jsonrpc-core serde_json = { version = "1.0.133", features = ["preserve_order"] } diff --git a/zebra-rpc/src/config.rs b/zebra-rpc/src/config.rs index 57187163e55..2a91d14334b 100644 --- a/zebra-rpc/src/config.rs +++ b/zebra-rpc/src/config.rs @@ -50,24 +50,12 @@ pub struct Config { /// The number of threads used to process RPC requests and responses. /// - /// Zebra's RPC server has a separate thread pool and a `tokio` executor for each thread. - /// State queries are run concurrently using the shared thread pool controlled by - /// the [`SyncSection.parallel_cpu_threads`](https://docs.rs/zebrad/latest/zebrad/components/sync/struct.Config.html#structfield.parallel_cpu_threads) config. - /// - /// If the number of threads is not configured or zero, Zebra uses the number of logical cores. - /// If the number of logical cores can't be detected, Zebra uses one thread. - /// - /// Set to `1` to run all RPC queries on a single thread, and detect RPC port conflicts from - /// multiple Zebra or `zcashd` instances. - /// - /// For details, see [the `jsonrpc_http_server` documentation](https://docs.rs/jsonrpc-http-server/latest/jsonrpc_http_server/struct.ServerBuilder.html#method.threads). - /// - /// ## Warning - /// - /// The default config uses multiple threads, which disables RPC port conflict detection. - /// This can allow multiple Zebra instances to share the same RPC port. - /// - /// If some of those instances are outdated or failed, RPC queries can be slow or inconsistent. + /// This field is deprecated and could be removed in a future release. + /// We keep it just for backward compatibility but it actually do nothing. + /// It was something configurable when the RPC server was based in the jsonrpc-core crate, + /// not anymore since we migrated to jsonrpsee. + // TODO: Prefix this field name with an underscore so it's clear that it's now unused, and + // use serde(rename) to continue successfully deserializing old configs. pub parallel_cpu_threads: usize, /// Test-only option that makes Zebra say it is at the chain tip, diff --git a/zebra-rpc/src/constants.rs b/zebra-rpc/src/constants.rs deleted file mode 100644 index 14f89df6618..00000000000 --- a/zebra-rpc/src/constants.rs +++ /dev/null @@ -1,44 +0,0 @@ -//! Constants for RPC methods and server responses. - -use jsonrpc_core::{Error, ErrorCode}; - -/// The RPC error code used by `zcashd` for incorrect RPC parameters. -/// -/// [`jsonrpc_core`] uses these codes: -/// -/// -/// `node-stratum-pool` mining pool library expects error code `-1` to detect available RPC methods: -/// -pub const INVALID_PARAMETERS_ERROR_CODE: ErrorCode = ErrorCode::ServerError(-1); - -/// The RPC error code used by `zcashd` for missing blocks, when looked up -/// by hash. -pub const INVALID_ADDRESS_OR_KEY_ERROR_CODE: ErrorCode = ErrorCode::ServerError(-5); - -/// The RPC error code used by `zcashd` for missing blocks. -/// -/// `lightwalletd` expects error code `-8` when a block is not found: -/// -pub const MISSING_BLOCK_ERROR_CODE: ErrorCode = ErrorCode::ServerError(-8); - -/// The RPC error code used by `zcashd` when there are no blocks in the state. -/// -/// `lightwalletd` expects error code `0` when there are no blocks in the state. -// -// TODO: find the source code that expects or generates this error -pub const NO_BLOCKS_IN_STATE_ERROR_CODE: ErrorCode = ErrorCode::ServerError(0); - -/// The RPC error used by `zcashd` when there are no blocks in the state. -// -// TODO: find the source code that expects or generates this error text, if there is any -// replace literal Error { ... } with this error -pub fn no_blocks_in_state_error() -> Error { - Error { - code: NO_BLOCKS_IN_STATE_ERROR_CODE, - message: "No blocks in state".to_string(), - data: None, - } -} - -/// When logging parameter data, only log this much data. -pub const MAX_PARAMS_LOG_LENGTH: usize = 100; diff --git a/zebra-rpc/src/lib.rs b/zebra-rpc/src/lib.rs index 778788c9edf..a5c2f3e5a17 100644 --- a/zebra-rpc/src/lib.rs +++ b/zebra-rpc/src/lib.rs @@ -5,7 +5,6 @@ #![doc(html_root_url = "https://docs.rs/zebra_rpc")] pub mod config; -pub mod constants; pub mod methods; pub mod queue; pub mod server; diff --git a/zebra-rpc/src/methods.rs b/zebra-rpc/src/methods.rs index 52f28d606b3..8634ec43ef5 100644 --- a/zebra-rpc/src/methods.rs +++ b/zebra-rpc/src/methods.rs @@ -9,12 +9,13 @@ use std::{collections::HashSet, fmt::Debug, sync::Arc}; use chrono::Utc; -use futures::{stream::FuturesOrdered, FutureExt, StreamExt, TryFutureExt}; +use futures::{stream::FuturesOrdered, StreamExt, TryFutureExt}; use hex::{FromHex, ToHex}; use hex_data::HexData; use indexmap::IndexMap; -use jsonrpc_core::{self, BoxFuture, Error, ErrorCode, Result}; -use jsonrpc_derive::rpc; +use jsonrpsee::core::{async_trait, RpcResult as Result}; +use jsonrpsee_proc_macros::rpc; +use jsonrpsee_types::{ErrorCode, ErrorObject}; use tokio::{sync::broadcast, task::JoinHandle}; use tower::{Service, ServiceExt}; use tracing::Instrument; @@ -34,21 +35,19 @@ use zebra_chain::{ }, }; use zebra_node_services::mempool; -use zebra_state::{HashOrHeight, MinedTx, OutputIndex, OutputLocation, TransactionLocation}; +use zebra_state::{HashOrHeight, OutputIndex, OutputLocation, TransactionLocation}; use crate::{ - constants::{ - INVALID_ADDRESS_OR_KEY_ERROR_CODE, INVALID_PARAMETERS_ERROR_CODE, MISSING_BLOCK_ERROR_CODE, - }, methods::trees::{GetSubtrees, GetTreestate, SubtreeRpcData}, queue::Queue, + server::{ + self, + error::{MapError, OkOrError}, + }, }; -mod errors; pub mod hex_data; -use errors::{MapServerError, OkOrServerError}; - // We don't use a types/ module here, because it is redundant. pub mod trees; @@ -58,7 +57,7 @@ pub mod types; pub mod get_block_template_rpcs; #[cfg(feature = "getblocktemplate-rpcs")] -pub use get_block_template_rpcs::{GetBlockTemplateRpc, GetBlockTemplateRpcImpl}; +pub use get_block_template_rpcs::{GetBlockTemplateRpcImpl, GetBlockTemplateRpcServer}; #[cfg(test)] mod tests; @@ -66,7 +65,6 @@ mod tests; #[rpc(server)] /// RPC method signatures. pub trait Rpc { - #[rpc(name = "getinfo")] /// Returns software information from the RPC server, as a [`GetInfo`] JSON struct. /// /// zcashd reference: [`getinfo`](https://zcash.github.io/rpc/getinfo.html) @@ -81,6 +79,7 @@ pub trait Rpc { /// /// Some fields from the zcashd reference are missing from Zebra's [`GetInfo`]. It only contains the fields /// [required for lightwalletd support.](https://github.com/zcash/lightwalletd/blob/v0.4.9/common/common.go#L91-L95) + #[method(name = "getinfo")] fn get_info(&self) -> Result; /// Returns blockchain state information, as a [`GetBlockChainInfo`] JSON struct. @@ -93,8 +92,8 @@ pub trait Rpc { /// /// Some fields from the zcashd reference are missing from Zebra's [`GetBlockChainInfo`]. It only contains the fields /// [required for lightwalletd support.](https://github.com/zcash/lightwalletd/blob/v0.4.9/common/common.go#L72-L89) - #[rpc(name = "getblockchaininfo")] - fn get_blockchain_info(&self) -> BoxFuture>; + #[method(name = "getblockchaininfo")] + async fn get_blockchain_info(&self) -> Result; /// Returns the total balance of a provided `addresses` in an [`AddressBalance`] instance. /// @@ -118,11 +117,8 @@ pub trait Rpc { /// The RPC documentation says that the returned object has a string `balance` field, but /// zcashd actually [returns an /// integer](https://github.com/zcash/lightwalletd/blob/bdaac63f3ee0dbef62bde04f6817a9f90d483b00/common/common.go#L128-L130). - #[rpc(name = "getaddressbalance")] - fn get_address_balance( - &self, - address_strings: AddressStrings, - ) -> BoxFuture>; + #[method(name = "getaddressbalance")] + async fn get_address_balance(&self, address_strings: AddressStrings) -> Result; /// Sends the raw bytes of a signed transaction to the local node's mempool, if the transaction is valid. /// Returns the [`SentTransactionHash`] for the transaction, as a JSON string. @@ -139,11 +135,11 @@ pub trait Rpc { /// /// zcashd accepts an optional `allowhighfees` parameter. Zebra doesn't support this parameter, /// because lightwalletd doesn't use it. - #[rpc(name = "sendrawtransaction")] - fn send_raw_transaction( + #[method(name = "sendrawtransaction")] + async fn send_raw_transaction( &self, raw_transaction_hex: String, - ) -> BoxFuture>; + ) -> Result; /// Returns the requested block by hash or height, as a [`GetBlock`] JSON string. /// If the block is not in Zebra's state, returns @@ -169,12 +165,8 @@ pub trait Rpc { /// use verbosity=3. /// /// The undocumented `chainwork` field is not returned. - #[rpc(name = "getblock")] - fn get_block( - &self, - hash_or_height: String, - verbosity: Option, - ) -> BoxFuture>; + #[method(name = "getblock")] + async fn get_block(&self, hash_or_height: String, verbosity: Option) -> Result; /// Returns the requested block header by hash or height, as a [`GetBlockHeader`] JSON string. /// If the block is not in Zebra's state, @@ -193,19 +185,19 @@ pub trait Rpc { /// # Notes /// /// The undocumented `chainwork` field is not returned. - #[rpc(name = "getblockheader")] - fn get_block_header( + #[method(name = "getblockheader")] + async fn get_block_header( &self, hash_or_height: String, verbose: Option, - ) -> BoxFuture>; + ) -> Result; /// Returns the hash of the current best blockchain tip block, as a [`GetBlockHash`] JSON string. /// /// zcashd reference: [`getbestblockhash`](https://zcash.github.io/rpc/getbestblockhash.html) /// method: post /// tags: blockchain - #[rpc(name = "getbestblockhash")] + #[method(name = "getbestblockhash")] fn get_best_block_hash(&self) -> Result; /// Returns the height and hash of the current best blockchain tip block, as a [`GetBlockHeightAndHash`] JSON struct. @@ -213,7 +205,7 @@ pub trait Rpc { /// zcashd reference: none /// method: post /// tags: blockchain - #[rpc(name = "getbestblockheightandhash")] + #[method(name = "getbestblockheightandhash")] fn get_best_block_height_and_hash(&self) -> Result; /// Returns all transaction ids in the memory pool, as a JSON array. @@ -221,8 +213,8 @@ pub trait Rpc { /// zcashd reference: [`getrawmempool`](https://zcash.github.io/rpc/getrawmempool.html) /// method: post /// tags: blockchain - #[rpc(name = "getrawmempool")] - fn get_raw_mempool(&self) -> BoxFuture>>; + #[method(name = "getrawmempool")] + async fn get_raw_mempool(&self) -> Result>; /// Returns information about the given block's Sapling & Orchard tree state. /// @@ -240,8 +232,8 @@ pub trait Rpc { /// negative where -1 is the last known valid block". On the other hand, /// `lightwalletd` only uses positive heights, so Zebra does not support /// negative heights. - #[rpc(name = "z_gettreestate")] - fn z_get_treestate(&self, hash_or_height: String) -> BoxFuture>; + #[method(name = "z_gettreestate")] + async fn z_get_treestate(&self, hash_or_height: String) -> Result; /// Returns information about a range of Sapling or Orchard subtrees. /// @@ -261,13 +253,13 @@ pub trait Rpc { /// starting at the chain tip. This RPC will return an empty list if the `start_index` subtree /// exists, but has not been rebuilt yet. This matches `zcashd`'s behaviour when subtrees aren't /// available yet. (But `zcashd` does its rebuild before syncing any blocks.) - #[rpc(name = "z_getsubtreesbyindex")] - fn z_get_subtrees_by_index( + #[method(name = "z_getsubtreesbyindex")] + async fn z_get_subtrees_by_index( &self, pool: String, start_index: NoteCommitmentSubtreeIndex, limit: Option, - ) -> BoxFuture>; + ) -> Result; /// Returns the raw transaction data, as a [`GetRawTransaction`] JSON string or structure. /// @@ -288,12 +280,12 @@ pub trait Rpc { /// In verbose mode, we only expose the `hex` and `height` fields since /// lightwalletd uses only those: /// - #[rpc(name = "getrawtransaction")] - fn get_raw_transaction( + #[method(name = "getrawtransaction")] + async fn get_raw_transaction( &self, - txid_hex: String, + txid: String, verbose: Option, - ) -> BoxFuture>; + ) -> Result; /// Returns the transaction ids made by the provided transparent addresses. /// @@ -312,9 +304,8 @@ pub trait Rpc { /// /// Only the multi-argument format is used by lightwalletd and this is what we currently support: /// - #[rpc(name = "getaddresstxids")] - fn get_address_tx_ids(&self, request: GetAddressTxIdsRequest) - -> BoxFuture>>; + #[method(name = "getaddresstxids")] + async fn get_address_tx_ids(&self, request: GetAddressTxIdsRequest) -> Result>; /// Returns all unspent outputs for a list of addresses. /// @@ -330,11 +321,11 @@ pub trait Rpc { /// /// lightwalletd always uses the multi-address request, without chaininfo: /// - #[rpc(name = "getaddressutxos")] - fn get_address_utxos( + #[method(name = "getaddressutxos")] + async fn get_address_utxos( &self, address_strings: AddressStrings, - ) -> BoxFuture>>; + ) -> Result>; /// Stop the running zebrad process. /// @@ -346,7 +337,7 @@ pub trait Rpc { /// zcashd reference: [`stop`](https://zcash.github.io/rpc/stop.html) /// method: post /// tags: control - #[rpc(name = "stop")] + #[method(name = "stop")] fn stop(&self) -> Result; } @@ -518,7 +509,8 @@ where } } -impl Rpc for RpcImpl +#[async_trait] +impl RpcServer for RpcImpl where Mempool: Service< mempool::Request, @@ -550,206 +542,199 @@ where } #[allow(clippy::unwrap_in_result)] - fn get_blockchain_info(&self) -> BoxFuture> { + async fn get_blockchain_info(&self) -> Result { let network = self.network.clone(); let debug_force_finished_sync = self.debug_force_finished_sync; let mut state = self.state.clone(); - async move { - // `chain` field - let chain = network.bip70_network_name(); - - let request = zebra_state::ReadRequest::TipPoolValues; - let response: zebra_state::ReadResponse = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; - - let zebra_state::ReadResponse::TipPoolValues { - tip_height, - tip_hash, - value_balance, - } = response - else { - unreachable!("unmatched response to a TipPoolValues request") - }; - - let request = zebra_state::ReadRequest::BlockHeader(tip_hash.into()); - let response: zebra_state::ReadResponse = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; - - let zebra_state::ReadResponse::BlockHeader { header, .. } = response else { - unreachable!("unmatched response to a BlockHeader request") - }; + // `chain` field + let chain = network.bip70_network_name(); + + let request = zebra_state::ReadRequest::TipPoolValues; + let response: zebra_state::ReadResponse = state + .ready() + .and_then(|service| service.call(request)) + .await + .map_misc_error()?; + + let zebra_state::ReadResponse::TipPoolValues { + tip_height, + tip_hash, + value_balance, + } = response + else { + unreachable!("unmatched response to a TipPoolValues request") + }; - let tip_block_time = header.time; + let request = zebra_state::ReadRequest::BlockHeader(tip_hash.into()); + let response: zebra_state::ReadResponse = state + .ready() + .and_then(|service| service.call(request)) + .await + .map_misc_error()?; - let now = Utc::now(); - let zebra_estimated_height = - NetworkChainTipHeightEstimator::new(tip_block_time, tip_height, &network) - .estimate_height_at(now); + let zebra_state::ReadResponse::BlockHeader { header, .. } = response else { + unreachable!("unmatched response to a BlockHeader request") + }; - // If we're testing the mempool, force the estimated height to be the actual tip height, otherwise, - // check if the estimated height is below Zebra's latest tip height, or if the latest tip's block time is - // later than the current time on the local clock. - let estimated_height = if tip_block_time > now - || zebra_estimated_height < tip_height - || debug_force_finished_sync - { - tip_height - } else { - zebra_estimated_height - }; + let tip_block_time = header.time; + + let now = Utc::now(); + let zebra_estimated_height = + NetworkChainTipHeightEstimator::new(tip_block_time, tip_height, &network) + .estimate_height_at(now); + + // If we're testing the mempool, force the estimated height to be the actual tip height, otherwise, + // check if the estimated height is below Zebra's latest tip height, or if the latest tip's block time is + // later than the current time on the local clock. + let estimated_height = if tip_block_time > now + || zebra_estimated_height < tip_height + || debug_force_finished_sync + { + tip_height + } else { + zebra_estimated_height + }; - // `upgrades` object + // `upgrades` object + // + // Get the network upgrades in height order, like `zcashd`. + let mut upgrades = IndexMap::new(); + for (activation_height, network_upgrade) in network.full_activation_list() { + // Zebra defines network upgrades based on incompatible consensus rule changes, + // but zcashd defines them based on ZIPs. // - // Get the network upgrades in height order, like `zcashd`. - let mut upgrades = IndexMap::new(); - for (activation_height, network_upgrade) in network.full_activation_list() { - // Zebra defines network upgrades based on incompatible consensus rule changes, - // but zcashd defines them based on ZIPs. - // - // All the network upgrades with a consensus branch ID are the same in Zebra and zcashd. - if let Some(branch_id) = network_upgrade.branch_id() { - // zcashd's RPC seems to ignore Disabled network upgrades, so Zebra does too. - let status = if tip_height >= activation_height { - NetworkUpgradeStatus::Active - } else { - NetworkUpgradeStatus::Pending - }; - - let upgrade = NetworkUpgradeInfo { - name: network_upgrade, - activation_height, - status, - }; - upgrades.insert(ConsensusBranchIdHex(branch_id), upgrade); - } + // All the network upgrades with a consensus branch ID are the same in Zebra and zcashd. + if let Some(branch_id) = network_upgrade.branch_id() { + // zcashd's RPC seems to ignore Disabled network upgrades, so Zebra does too. + let status = if tip_height >= activation_height { + NetworkUpgradeStatus::Active + } else { + NetworkUpgradeStatus::Pending + }; + + let upgrade = NetworkUpgradeInfo { + name: network_upgrade, + activation_height, + status, + }; + upgrades.insert(ConsensusBranchIdHex(branch_id), upgrade); } + } - // `consensus` object - let next_block_height = - (tip_height + 1).expect("valid chain tips are a lot less than Height::MAX"); - let consensus = TipConsensusBranch { - chain_tip: ConsensusBranchIdHex( - NetworkUpgrade::current(&network, tip_height) - .branch_id() - .unwrap_or(ConsensusBranchId::RPC_MISSING_ID), - ), - next_block: ConsensusBranchIdHex( - NetworkUpgrade::current(&network, next_block_height) - .branch_id() - .unwrap_or(ConsensusBranchId::RPC_MISSING_ID), - ), - }; + // `consensus` object + let next_block_height = + (tip_height + 1).expect("valid chain tips are a lot less than Height::MAX"); + let consensus = TipConsensusBranch { + chain_tip: ConsensusBranchIdHex( + NetworkUpgrade::current(&network, tip_height) + .branch_id() + .unwrap_or(ConsensusBranchId::RPC_MISSING_ID), + ), + next_block: ConsensusBranchIdHex( + NetworkUpgrade::current(&network, next_block_height) + .branch_id() + .unwrap_or(ConsensusBranchId::RPC_MISSING_ID), + ), + }; - let response = GetBlockChainInfo { - chain, - blocks: tip_height, - best_block_hash: tip_hash, - estimated_height, - value_pools: types::ValuePoolBalance::from_value_balance(value_balance), - upgrades, - consensus, - }; + let response = GetBlockChainInfo { + chain, + blocks: tip_height, + best_block_hash: tip_hash, + estimated_height, + value_pools: types::ValuePoolBalance::from_value_balance(value_balance), + upgrades, + consensus, + }; - Ok(response) - } - .boxed() + Ok(response) } - fn get_address_balance( - &self, - address_strings: AddressStrings, - ) -> BoxFuture> { + async fn get_address_balance(&self, address_strings: AddressStrings) -> Result { let state = self.state.clone(); - async move { - let valid_addresses = address_strings.valid_addresses()?; + let valid_addresses = address_strings.valid_addresses()?; - let request = zebra_state::ReadRequest::AddressBalance(valid_addresses); - let response = state.oneshot(request).await.map_server_error()?; + let request = zebra_state::ReadRequest::AddressBalance(valid_addresses); + let response = state.oneshot(request).await.map_misc_error()?; - match response { - zebra_state::ReadResponse::AddressBalance(balance) => Ok(AddressBalance { - balance: u64::from(balance), - }), - _ => unreachable!("Unexpected response from state service: {response:?}"), - } + match response { + zebra_state::ReadResponse::AddressBalance(balance) => Ok(AddressBalance { + balance: u64::from(balance), + }), + _ => unreachable!("Unexpected response from state service: {response:?}"), } - .boxed() } // TODO: use HexData or GetRawTransaction::Bytes to handle the transaction data argument - fn send_raw_transaction( + async fn send_raw_transaction( &self, raw_transaction_hex: String, - ) -> BoxFuture> { + ) -> Result { let mempool = self.mempool.clone(); let queue_sender = self.queue_sender.clone(); - async move { - let raw_transaction_bytes = Vec::from_hex(raw_transaction_hex).map_err(|_| { - Error::invalid_params("raw transaction is not specified as a hex string") - })?; - let raw_transaction = Transaction::zcash_deserialize(&*raw_transaction_bytes) - .map_err(|_| Error::invalid_params("raw transaction is structurally invalid"))?; - - let transaction_hash = raw_transaction.hash(); + // Reference for the legacy error code: + // + let raw_transaction_bytes = Vec::from_hex(raw_transaction_hex) + .map_error(server::error::LegacyCode::Deserialization)?; + let raw_transaction = Transaction::zcash_deserialize(&*raw_transaction_bytes) + .map_error(server::error::LegacyCode::Deserialization)?; - // send transaction to the rpc queue, ignore any error. - let unmined_transaction = UnminedTx::from(raw_transaction.clone()); - let _ = queue_sender.send(unmined_transaction); + let transaction_hash = raw_transaction.hash(); - let transaction_parameter = mempool::Gossip::Tx(raw_transaction.into()); - let request = mempool::Request::Queue(vec![transaction_parameter]); + // send transaction to the rpc queue, ignore any error. + let unmined_transaction = UnminedTx::from(raw_transaction.clone()); + let _ = queue_sender.send(unmined_transaction); - let response = mempool.oneshot(request).await.map_server_error()?; + let transaction_parameter = mempool::Gossip::Tx(raw_transaction.into()); + let request = mempool::Request::Queue(vec![transaction_parameter]); - let mut queue_results = match response { - mempool::Response::Queued(results) => results, - _ => unreachable!("incorrect response variant from mempool service"), - }; - - assert_eq!( - queue_results.len(), - 1, - "mempool service returned more results than expected" - ); + let response = mempool.oneshot(request).await.map_misc_error()?; - let queue_result = queue_results - .pop() - .expect("there should be exactly one item in Vec") - .inspect_err(|err| tracing::debug!("sent transaction to mempool: {:?}", &err)) - .map_server_error()? - .await; + let mut queue_results = match response { + mempool::Response::Queued(results) => results, + _ => unreachable!("incorrect response variant from mempool service"), + }; - tracing::debug!("sent transaction to mempool: {:?}", &queue_result); + assert_eq!( + queue_results.len(), + 1, + "mempool service returned more results than expected" + ); - queue_result - .map_server_error()? - .map(|_| SentTransactionHash(transaction_hash)) - .map_server_error() - } - .boxed() + let queue_result = queue_results + .pop() + .expect("there should be exactly one item in Vec") + .inspect_err(|err| tracing::debug!("sent transaction to mempool: {:?}", &err)) + .map_misc_error()? + .await + .map_misc_error()?; + + tracing::debug!("sent transaction to mempool: {:?}", &queue_result); + + queue_result + .map(|_| SentTransactionHash(transaction_hash)) + // Reference for the legacy error code: + // + // Note that this error code might not exactly match the one returned by zcashd + // since zcashd's error code selection logic is more granular. We'd need to + // propagate the error coming from the verifier to be able to return more specific + // error codes. + .map_error(server::error::LegacyCode::Verify) } + // # Performance + // + // `lightwalletd` calls this RPC with verosity 1 for its initial sync of 2 million blocks, the + // performance of this RPC with verbosity 1 significantly affects `lightwalletd`s sync time. + // // TODO: // - use `height_from_signed_int()` to handle negative heights // (this might be better in the state request, because it needs the state height) - fn get_block( - &self, - hash_or_height: String, - verbosity: Option, - ) -> BoxFuture> { - // From - const DEFAULT_GETBLOCK_VERBOSITY: u8 = 1; - + async fn get_block(&self, hash_or_height: String, verbosity: Option) -> Result { let mut state = self.state.clone(); - let verbosity = verbosity.unwrap_or(DEFAULT_GETBLOCK_VERBOSITY); + let verbosity = verbosity.unwrap_or(1); let network = self.network.clone(); let original_hash_or_height = hash_or_height.clone(); @@ -760,278 +745,286 @@ where None }; - async move { - let hash_or_height: HashOrHeight = hash_or_height.parse().map_server_error()?; + let hash_or_height: HashOrHeight = hash_or_height + .parse() + // Reference for the legacy error code: + // + .map_error(server::error::LegacyCode::InvalidParameter)?; - if verbosity == 0 { - // # Performance - // - // This RPC is used in `lightwalletd`'s initial sync of 2 million blocks, - // so it needs to load block data very efficiently. - let request = zebra_state::ReadRequest::Block(hash_or_height); - let response = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; - - match response { - zebra_state::ReadResponse::Block(Some(block)) => { - Ok(GetBlock::Raw(block.into())) - } - zebra_state::ReadResponse::Block(None) => Err(Error { - code: MISSING_BLOCK_ERROR_CODE, - message: "Block not found".to_string(), - data: None, - }), - _ => unreachable!("unmatched response to a block request"), + if verbosity == 0 { + let request = zebra_state::ReadRequest::Block(hash_or_height); + let response = state + .ready() + .and_then(|service| service.call(request)) + .await + .map_misc_error()?; + + match response { + zebra_state::ReadResponse::Block(Some(block)) => Ok(GetBlock::Raw(block.into())), + zebra_state::ReadResponse::Block(None) => { + Err("Block not found").map_error(server::error::LegacyCode::InvalidParameter) } - } else if let Some(get_block_header_future) = get_block_header_future { - let get_block_header_result: Result = get_block_header_future.await; + _ => unreachable!("unmatched response to a block request"), + } + } else if let Some(get_block_header_future) = get_block_header_future { + let get_block_header_result: Result = get_block_header_future.await; - let GetBlockHeader::Object(block_header) = get_block_header_result? else { - panic!("must return Object") - }; + let GetBlockHeader::Object(block_header) = get_block_header_result? else { + panic!("must return Object") + }; - let GetBlockHeaderObject { - hash, - confirmations, - height, - version, - merkle_root, - final_sapling_root, - sapling_tree_size, - time, - nonce, - solution, - bits, - difficulty, - previous_block_hash, - next_block_hash, - } = *block_header; + let GetBlockHeaderObject { + hash, + confirmations, + height, + version, + merkle_root, + final_sapling_root, + sapling_tree_size, + time, + nonce, + solution, + bits, + difficulty, + previous_block_hash, + next_block_hash, + } = *block_header; + + let transactions_request = match verbosity { + 1 => zebra_state::ReadRequest::TransactionIdsForBlock(hash_or_height), + 2 => zebra_state::ReadRequest::Block(hash_or_height), + _other => panic!("get_block_header_fut should be none"), + }; + // # Concurrency + // + // We look up by block hash so the hash, transaction IDs, and confirmations + // are consistent. + let hash_or_height = hash.0.into(); + let requests = vec![ + // Get transaction IDs from the transaction index by block hash + // // # Concurrency // - // We look up by block hash so the hash, transaction IDs, and confirmations - // are consistent. - let hash_or_height = hash.0.into(); - let requests = vec![ - // Get transaction IDs from the transaction index by block hash - // - // # Concurrency - // - // A block's transaction IDs are never modified, so all possible responses are - // valid. Clients that query block heights must be able to handle chain forks, - // including getting transaction IDs from any chain fork. - zebra_state::ReadRequest::TransactionIdsForBlock(hash_or_height), - // Orchard trees - zebra_state::ReadRequest::OrchardTree(hash_or_height), - ]; - - let mut futs = FuturesOrdered::new(); - - for request in requests { - futs.push_back(state.clone().oneshot(request)); - } + // A block's transaction IDs are never modified, so all possible responses are + // valid. Clients that query block heights must be able to handle chain forks, + // including getting transaction IDs from any chain fork. + transactions_request, + // Orchard trees + zebra_state::ReadRequest::OrchardTree(hash_or_height), + ]; + + let mut futs = FuturesOrdered::new(); + + for request in requests { + futs.push_back(state.clone().oneshot(request)); + } - let tx_ids_response = futs.next().await.expect("`futs` should not be empty"); - let tx = match tx_ids_response.map_server_error()? { - zebra_state::ReadResponse::TransactionIdsForBlock(tx_ids) => tx_ids - .ok_or_server_error("Block not found")? - .iter() - .map(|tx_id| tx_id.encode_hex()) - .collect(), - _ => unreachable!("unmatched response to a transaction_ids_for_block request"), - }; + let tx_ids_response = futs.next().await.expect("`futs` should not be empty"); + let tx: Vec<_> = match tx_ids_response.map_misc_error()? { + zebra_state::ReadResponse::TransactionIdsForBlock(tx_ids) => tx_ids + .ok_or_misc_error("block not found")? + .iter() + .map(|tx_id| GetBlockTransaction::Hash(*tx_id)) + .collect(), + zebra_state::ReadResponse::Block(block) => block + .ok_or_misc_error("Block not found")? + .transactions + .iter() + .map(|tx| { + GetBlockTransaction::Object(TransactionObject::from_transaction( + tx.clone(), + Some(height), + Some( + confirmations + .try_into() + .expect("should be less than max block height, i32::MAX"), + ), + )) + }) + .collect(), + _ => unreachable!("unmatched response to a transaction_ids_for_block request"), + }; - let orchard_tree_response = futs.next().await.expect("`futs` should not be empty"); - let zebra_state::ReadResponse::OrchardTree(orchard_tree) = - orchard_tree_response.map_server_error()? - else { - unreachable!("unmatched response to a OrchardTree request"); - }; + let orchard_tree_response = futs.next().await.expect("`futs` should not be empty"); + let zebra_state::ReadResponse::OrchardTree(orchard_tree) = + orchard_tree_response.map_misc_error()? + else { + unreachable!("unmatched response to a OrchardTree request"); + }; - let nu5_activation = NetworkUpgrade::Nu5.activation_height(&network); + let nu5_activation = NetworkUpgrade::Nu5.activation_height(&network); - // This could be `None` if there's a chain reorg between state queries. - let orchard_tree = - orchard_tree.ok_or_server_error("missing orchard tree for block")?; + // This could be `None` if there's a chain reorg between state queries. + let orchard_tree = orchard_tree.ok_or_misc_error("missing Orchard tree")?; - let final_orchard_root = match nu5_activation { - Some(activation_height) if height >= activation_height => { - Some(orchard_tree.root().into()) - } - _other => None, - }; + let final_orchard_root = match nu5_activation { + Some(activation_height) if height >= activation_height => { + Some(orchard_tree.root().into()) + } + _other => None, + }; - let sapling = SaplingTrees { - size: sapling_tree_size, - }; + let sapling = SaplingTrees { + size: sapling_tree_size, + }; - let orchard_tree_size = orchard_tree.count(); - let orchard = OrchardTrees { - size: orchard_tree_size, - }; + let orchard_tree_size = orchard_tree.count(); + let orchard = OrchardTrees { + size: orchard_tree_size, + }; - let trees = GetBlockTrees { sapling, orchard }; - - Ok(GetBlock::Object { - hash, - confirmations, - height: Some(height), - version: Some(version), - merkle_root: Some(merkle_root), - time: Some(time), - nonce: Some(nonce), - solution: Some(solution), - bits: Some(bits), - difficulty: Some(difficulty), - tx, - trees, - size: None, - final_sapling_root: Some(final_sapling_root), - final_orchard_root, - previous_block_hash: Some(previous_block_hash), - next_block_hash, - }) - } else { - Err(Error { - code: ErrorCode::InvalidParams, - message: "Invalid verbosity value".to_string(), - data: None, - }) - } + let trees = GetBlockTrees { sapling, orchard }; + + Ok(GetBlock::Object { + hash, + confirmations, + height: Some(height), + version: Some(version), + merkle_root: Some(merkle_root), + time: Some(time), + nonce: Some(nonce), + solution: Some(solution), + bits: Some(bits), + difficulty: Some(difficulty), + tx, + trees, + size: None, + final_sapling_root: Some(final_sapling_root), + final_orchard_root, + previous_block_hash: Some(previous_block_hash), + next_block_hash, + }) + } else { + Err("invalid verbosity value").map_error(server::error::LegacyCode::InvalidParameter) } - .boxed() } - fn get_block_header( + async fn get_block_header( &self, hash_or_height: String, verbose: Option, - ) -> BoxFuture> { + ) -> Result { let state = self.state.clone(); let verbose = verbose.unwrap_or(true); let network = self.network.clone(); - async move { - let hash_or_height: HashOrHeight = hash_or_height.parse().map_server_error()?; - let zebra_state::ReadResponse::BlockHeader { - header, - hash, - height, - next_block_hash, - } = state + let hash_or_height: HashOrHeight = hash_or_height + .parse() + .map_error(server::error::LegacyCode::InvalidAddressOrKey)?; + let zebra_state::ReadResponse::BlockHeader { + header, + hash, + height, + next_block_hash, + } = state + .clone() + .oneshot(zebra_state::ReadRequest::BlockHeader(hash_or_height)) + .await + .map_err(|_| "block height not in best chain") + .map_error( + // ## Compatibility with `zcashd`. + // + // Since this function is reused by getblock(), we return the errors + // expected by it (they differ whether a hash or a height was passed). + if hash_or_height.hash().is_some() { + server::error::LegacyCode::InvalidAddressOrKey + } else { + server::error::LegacyCode::InvalidParameter + }, + )? + else { + panic!("unexpected response to BlockHeader request") + }; + + let response = if !verbose { + GetBlockHeader::Raw(HexData(header.zcash_serialize_to_vec().map_misc_error()?)) + } else { + let zebra_state::ReadResponse::SaplingTree(sapling_tree) = state .clone() - .oneshot(zebra_state::ReadRequest::BlockHeader(hash_or_height)) + .oneshot(zebra_state::ReadRequest::SaplingTree(hash_or_height)) .await - .map_err(|_| Error { - // Compatibility with zcashd. Note that since this function - // is reused by getblock(), we return the errors expected - // by it (they differ whether a hash or a height was passed) - code: if hash_or_height.hash().is_some() { - INVALID_ADDRESS_OR_KEY_ERROR_CODE - } else { - MISSING_BLOCK_ERROR_CODE - }, - message: "block height not in best chain".to_string(), - data: None, - })? + .map_misc_error()? else { - panic!("unexpected response to BlockHeader request") + panic!("unexpected response to SaplingTree request") }; - let response = if !verbose { - GetBlockHeader::Raw(HexData(header.zcash_serialize_to_vec().map_server_error()?)) - } else { - let zebra_state::ReadResponse::SaplingTree(sapling_tree) = state - .clone() - .oneshot(zebra_state::ReadRequest::SaplingTree(hash_or_height)) - .await - .map_server_error()? - else { - panic!("unexpected response to SaplingTree request") - }; + // This could be `None` if there's a chain reorg between state queries. + let sapling_tree = sapling_tree.ok_or_misc_error("missing Sapling tree")?; - // This could be `None` if there's a chain reorg between state queries. - let sapling_tree = - sapling_tree.ok_or_server_error("missing sapling tree for block")?; - - let zebra_state::ReadResponse::Depth(depth) = state - .clone() - .oneshot(zebra_state::ReadRequest::Depth(hash)) - .await - .map_server_error()? - else { - panic!("unexpected response to SaplingTree request") - }; + let zebra_state::ReadResponse::Depth(depth) = state + .clone() + .oneshot(zebra_state::ReadRequest::Depth(hash)) + .await + .map_misc_error()? + else { + panic!("unexpected response to SaplingTree request") + }; - // From - // TODO: Deduplicate const definition, consider refactoring this to avoid duplicate logic - const NOT_IN_BEST_CHAIN_CONFIRMATIONS: i64 = -1; - - // Confirmations are one more than the depth. - // Depth is limited by height, so it will never overflow an i64. - let confirmations = depth - .map(|depth| i64::from(depth) + 1) - .unwrap_or(NOT_IN_BEST_CHAIN_CONFIRMATIONS); - - let mut nonce = *header.nonce; - nonce.reverse(); - - let sapling_activation = NetworkUpgrade::Sapling.activation_height(&network); - let sapling_tree_size = sapling_tree.count(); - let final_sapling_root: [u8; 32] = - if sapling_activation.is_some() && height >= sapling_activation.unwrap() { - let mut root: [u8; 32] = sapling_tree.root().into(); - root.reverse(); - root - } else { - [0; 32] - }; - - let difficulty = header.difficulty_threshold.relative_to_network(&network); - - let block_header = GetBlockHeaderObject { - hash: GetBlockHash(hash), - confirmations, - height, - version: header.version, - merkle_root: header.merkle_root, - final_sapling_root, - sapling_tree_size, - time: header.time.timestamp(), - nonce, - solution: header.solution, - bits: header.difficulty_threshold, - difficulty, - previous_block_hash: GetBlockHash(header.previous_block_hash), - next_block_hash: next_block_hash.map(GetBlockHash), + // From + // TODO: Deduplicate const definition, consider refactoring this to avoid duplicate logic + const NOT_IN_BEST_CHAIN_CONFIRMATIONS: i64 = -1; + + // Confirmations are one more than the depth. + // Depth is limited by height, so it will never overflow an i64. + let confirmations = depth + .map(|depth| i64::from(depth) + 1) + .unwrap_or(NOT_IN_BEST_CHAIN_CONFIRMATIONS); + + let mut nonce = *header.nonce; + nonce.reverse(); + + let sapling_activation = NetworkUpgrade::Sapling.activation_height(&network); + let sapling_tree_size = sapling_tree.count(); + let final_sapling_root: [u8; 32] = + if sapling_activation.is_some() && height >= sapling_activation.unwrap() { + let mut root: [u8; 32] = sapling_tree.root().into(); + root.reverse(); + root + } else { + [0; 32] }; - GetBlockHeader::Object(Box::new(block_header)) + let difficulty = header.difficulty_threshold.relative_to_network(&network); + + let block_header = GetBlockHeaderObject { + hash: GetBlockHash(hash), + confirmations, + height, + version: header.version, + merkle_root: header.merkle_root, + final_sapling_root, + sapling_tree_size, + time: header.time.timestamp(), + nonce, + solution: header.solution, + bits: header.difficulty_threshold, + difficulty, + previous_block_hash: GetBlockHash(header.previous_block_hash), + next_block_hash: next_block_hash.map(GetBlockHash), }; - Ok(response) - } - .boxed() + GetBlockHeader::Object(Box::new(block_header)) + }; + + Ok(response) } fn get_best_block_hash(&self) -> Result { self.latest_chain_tip .best_tip_hash() .map(GetBlockHash) - .ok_or_server_error("No blocks in state") + .ok_or_misc_error("No blocks in state") } fn get_best_block_height_and_hash(&self) -> Result { self.latest_chain_tip .best_tip_height_and_hash() .map(|(height, hash)| GetBlockHeightAndHash { height, hash }) - .ok_or_server_error("No blocks in state") + .ok_or_misc_error("No blocks in state") } - fn get_raw_mempool(&self) -> BoxFuture>> { + async fn get_raw_mempool(&self) -> Result> { #[cfg(feature = "getblocktemplate-rpcs")] use zebra_chain::block::MAX_BLOCK_BYTES; @@ -1041,420 +1034,400 @@ where let mut mempool = self.mempool.clone(); - async move { - #[cfg(feature = "getblocktemplate-rpcs")] - let request = if should_use_zcashd_order { - mempool::Request::FullTransactions - } else { - mempool::Request::TransactionIds - }; - - #[cfg(not(feature = "getblocktemplate-rpcs"))] - let request = mempool::Request::TransactionIds; - - // `zcashd` doesn't check if it is synced to the tip here, so we don't either. - let response = mempool - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; + #[cfg(feature = "getblocktemplate-rpcs")] + let request = if should_use_zcashd_order { + mempool::Request::FullTransactions + } else { + mempool::Request::TransactionIds + }; - match response { - #[cfg(feature = "getblocktemplate-rpcs")] - mempool::Response::FullTransactions { - mut transactions, - transaction_dependencies: _, - last_seen_tip_hash: _, - } => { - // Sort transactions in descending order by fee/size, using hash in serialized byte order as a tie-breaker - transactions.sort_by_cached_key(|tx| { - // zcashd uses modified fee here but Zebra doesn't currently - // support prioritizing transactions - std::cmp::Reverse(( - i64::from(tx.miner_fee) as u128 * MAX_BLOCK_BYTES as u128 - / tx.transaction.size as u128, - // transaction hashes are compared in their serialized byte-order. - tx.transaction.id.mined_id(), - )) - }); + #[cfg(not(feature = "getblocktemplate-rpcs"))] + let request = mempool::Request::TransactionIds; - let tx_ids: Vec = transactions - .iter() - .map(|unmined_tx| unmined_tx.transaction.id.mined_id().encode_hex()) - .collect(); + // `zcashd` doesn't check if it is synced to the tip here, so we don't either. + let response = mempool + .ready() + .and_then(|service| service.call(request)) + .await + .map_misc_error()?; - Ok(tx_ids) - } + match response { + #[cfg(feature = "getblocktemplate-rpcs")] + mempool::Response::FullTransactions { + mut transactions, + transaction_dependencies: _, + last_seen_tip_hash: _, + } => { + // Sort transactions in descending order by fee/size, using hash in serialized byte order as a tie-breaker + transactions.sort_by_cached_key(|tx| { + // zcashd uses modified fee here but Zebra doesn't currently + // support prioritizing transactions + std::cmp::Reverse(( + i64::from(tx.miner_fee) as u128 * MAX_BLOCK_BYTES as u128 + / tx.transaction.size as u128, + // transaction hashes are compared in their serialized byte-order. + tx.transaction.id.mined_id(), + )) + }); + + let tx_ids: Vec = transactions + .iter() + .map(|unmined_tx| unmined_tx.transaction.id.mined_id().encode_hex()) + .collect(); - mempool::Response::TransactionIds(unmined_transaction_ids) => { - let mut tx_ids: Vec = unmined_transaction_ids - .iter() - .map(|id| id.mined_id().encode_hex()) - .collect(); + Ok(tx_ids) + } - // Sort returned transaction IDs in numeric/string order. - tx_ids.sort(); + mempool::Response::TransactionIds(unmined_transaction_ids) => { + let mut tx_ids: Vec = unmined_transaction_ids + .iter() + .map(|id| id.mined_id().encode_hex()) + .collect(); - Ok(tx_ids) - } + // Sort returned transaction IDs in numeric/string order. + tx_ids.sort(); - _ => unreachable!("unmatched response to a transactionids request"), + Ok(tx_ids) } + + _ => unreachable!("unmatched response to a transactionids request"), } - .boxed() } - // TODO: use HexData or SentTransactionHash to handle the transaction ID - fn get_raw_transaction( + async fn get_raw_transaction( &self, - txid_hex: String, + txid: String, verbose: Option, - ) -> BoxFuture> { + ) -> Result { let mut state = self.state.clone(); let mut mempool = self.mempool.clone(); - let verbose = verbose.unwrap_or(0); - let verbose = verbose != 0; - - async move { - let txid = transaction::Hash::from_hex(txid_hex).map_err(|_| { - Error::invalid_params("transaction ID is not specified as a hex string") - })?; - - // Check the mempool first. - // - // # Correctness - // - // Transactions are removed from the mempool after they are mined into blocks, - // so the transaction could be just in the mempool, just in the state, or in both. - // (And the mempool and state transactions could have different authorising data.) - // But it doesn't matter which transaction we choose, because the effects are the same. - let mut txid_set = HashSet::new(); - txid_set.insert(txid); - let request = mempool::Request::TransactionsByMinedId(txid_set); - - let response = mempool - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; - - match response { - mempool::Response::Transactions(unmined_transactions) => { - if !unmined_transactions.is_empty() { - let tx = unmined_transactions[0].transaction.clone(); - return Ok(GetRawTransaction::from_transaction(tx, None, 0, verbose)); - } + let verbose = verbose.unwrap_or(0) != 0; + + // Reference for the legacy error code: + // + let txid = transaction::Hash::from_hex(txid) + .map_error(server::error::LegacyCode::InvalidAddressOrKey)?; + + // Check the mempool first. + match mempool + .ready() + .and_then(|service| { + service.call(mempool::Request::TransactionsByMinedId([txid].into())) + }) + .await + .map_misc_error()? + { + mempool::Response::Transactions(txns) => { + if let Some(tx) = txns.first() { + return Ok(if verbose { + GetRawTransaction::Object(TransactionObject::from_transaction( + tx.transaction.clone(), + None, + None, + )) + } else { + let hex = tx.transaction.clone().into(); + GetRawTransaction::Raw(hex) + }); } - _ => unreachable!("unmatched response to a transactionids request"), - }; + } - // Now check the state - let request = zebra_state::ReadRequest::Transaction(txid); - let response = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; + _ => unreachable!("unmatched response to a `TransactionsByMinedId` request"), + }; - match response { - zebra_state::ReadResponse::Transaction(Some(MinedTx { - tx, - height, - confirmations, - })) => Ok(GetRawTransaction::from_transaction( - tx, - Some(height), - confirmations, - verbose, - )), - zebra_state::ReadResponse::Transaction(None) => { - Err("Transaction not found").map_server_error() - } - _ => unreachable!("unmatched response to a transaction request"), + // If the tx wasn't in the mempool, check the state. + match state + .ready() + .and_then(|service| service.call(zebra_state::ReadRequest::Transaction(txid))) + .await + .map_misc_error()? + { + zebra_state::ReadResponse::Transaction(Some(tx)) => Ok(if verbose { + GetRawTransaction::Object(TransactionObject::from_transaction( + tx.tx.clone(), + Some(tx.height), + Some(tx.confirmations), + )) + } else { + let hex = tx.tx.into(); + GetRawTransaction::Raw(hex) + }), + + zebra_state::ReadResponse::Transaction(None) => { + Err("No such mempool or main chain transaction") + .map_error(server::error::LegacyCode::InvalidAddressOrKey) } + + _ => unreachable!("unmatched response to a `Transaction` read request"), } - .boxed() } // TODO: // - use `height_from_signed_int()` to handle negative heights // (this might be better in the state request, because it needs the state height) - fn z_get_treestate(&self, hash_or_height: String) -> BoxFuture> { + async fn z_get_treestate(&self, hash_or_height: String) -> Result { let mut state = self.state.clone(); let network = self.network.clone(); - async move { - // Convert the [`hash_or_height`] string into an actual hash or height. - let hash_or_height = hash_or_height.parse().map_server_error()?; + // Reference for the legacy error code: + // + let hash_or_height = hash_or_height + .parse() + .map_error(server::error::LegacyCode::InvalidParameter)?; - // Fetch the block referenced by [`hash_or_height`] from the state. - // - // # Concurrency - // - // For consistency, this lookup must be performed first, then all the other lookups must - // be based on the hash. - // - // TODO: If this RPC is called a lot, just get the block header, rather than the whole block. - let block = match state - .ready() - .and_then(|service| service.call(zebra_state::ReadRequest::Block(hash_or_height))) - .await - .map_server_error()? - { - zebra_state::ReadResponse::Block(Some(block)) => block, - zebra_state::ReadResponse::Block(None) => { - return Err(Error { - code: MISSING_BLOCK_ERROR_CODE, - message: "the requested block was not found".to_string(), - data: None, - }) - } - _ => unreachable!("unmatched response to a block request"), - }; + // Fetch the block referenced by [`hash_or_height`] from the state. + // + // # Concurrency + // + // For consistency, this lookup must be performed first, then all the other lookups must + // be based on the hash. + // + // TODO: If this RPC is called a lot, just get the block header, rather than the whole block. + let block = match state + .ready() + .and_then(|service| service.call(zebra_state::ReadRequest::Block(hash_or_height))) + .await + .map_misc_error()? + { + zebra_state::ReadResponse::Block(Some(block)) => block, + zebra_state::ReadResponse::Block(None) => { + // Reference for the legacy error code: + // + return Err("the requested block is not in the main chain") + .map_error(server::error::LegacyCode::InvalidParameter); + } + _ => unreachable!("unmatched response to a block request"), + }; - let hash = hash_or_height - .hash_or_else(|_| Some(block.hash())) - .expect("block hash"); + let hash = hash_or_height + .hash_or_else(|_| Some(block.hash())) + .expect("block hash"); - let height = hash_or_height - .height_or_else(|_| block.coinbase_height()) - .expect("verified blocks have a coinbase height"); + let height = hash_or_height + .height_or_else(|_| block.coinbase_height()) + .expect("verified blocks have a coinbase height"); - let time = u32::try_from(block.header.time.timestamp()) - .expect("Timestamps of valid blocks always fit into u32."); + let time = u32::try_from(block.header.time.timestamp()) + .expect("Timestamps of valid blocks always fit into u32."); - let sapling_nu = zcash_primitives::consensus::NetworkUpgrade::Sapling; - let sapling = if network.is_nu_active(sapling_nu, height.into()) { - match state - .ready() - .and_then(|service| { - service.call(zebra_state::ReadRequest::SaplingTree(hash.into())) - }) - .await - .map_server_error()? - { - zebra_state::ReadResponse::SaplingTree(tree) => tree.map(|t| t.to_rpc_bytes()), - _ => unreachable!("unmatched response to a Sapling tree request"), - } - } else { - None - }; + let sapling_nu = zcash_primitives::consensus::NetworkUpgrade::Sapling; + let sapling = if network.is_nu_active(sapling_nu, height.into()) { + match state + .ready() + .and_then(|service| { + service.call(zebra_state::ReadRequest::SaplingTree(hash.into())) + }) + .await + .map_misc_error()? + { + zebra_state::ReadResponse::SaplingTree(tree) => tree.map(|t| t.to_rpc_bytes()), + _ => unreachable!("unmatched response to a Sapling tree request"), + } + } else { + None + }; - let orchard_nu = zcash_primitives::consensus::NetworkUpgrade::Nu5; - let orchard = if network.is_nu_active(orchard_nu, height.into()) { - match state - .ready() - .and_then(|service| { - service.call(zebra_state::ReadRequest::OrchardTree(hash.into())) - }) - .await - .map_server_error()? - { - zebra_state::ReadResponse::OrchardTree(tree) => tree.map(|t| t.to_rpc_bytes()), - _ => unreachable!("unmatched response to an Orchard tree request"), - } - } else { - None - }; + let orchard_nu = zcash_primitives::consensus::NetworkUpgrade::Nu5; + let orchard = if network.is_nu_active(orchard_nu, height.into()) { + match state + .ready() + .and_then(|service| { + service.call(zebra_state::ReadRequest::OrchardTree(hash.into())) + }) + .await + .map_misc_error()? + { + zebra_state::ReadResponse::OrchardTree(tree) => tree.map(|t| t.to_rpc_bytes()), + _ => unreachable!("unmatched response to an Orchard tree request"), + } + } else { + None + }; - Ok(GetTreestate::from_parts( - hash, height, time, sapling, orchard, - )) - } - .boxed() + Ok(GetTreestate::from_parts( + hash, height, time, sapling, orchard, + )) } - fn z_get_subtrees_by_index( + async fn z_get_subtrees_by_index( &self, pool: String, start_index: NoteCommitmentSubtreeIndex, limit: Option, - ) -> BoxFuture> { + ) -> Result { let mut state = self.state.clone(); - async move { - const POOL_LIST: &[&str] = &["sapling", "orchard"]; + const POOL_LIST: &[&str] = &["sapling", "orchard"]; - if pool == "sapling" { - let request = zebra_state::ReadRequest::SaplingSubtrees { start_index, limit }; - let response = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; - - let subtrees = match response { - zebra_state::ReadResponse::SaplingSubtrees(subtrees) => subtrees, - _ => unreachable!("unmatched response to a subtrees request"), - }; + if pool == "sapling" { + let request = zebra_state::ReadRequest::SaplingSubtrees { start_index, limit }; + let response = state + .ready() + .and_then(|service| service.call(request)) + .await + .map_misc_error()?; - let subtrees = subtrees - .values() - .map(|subtree| SubtreeRpcData { - root: subtree.root.encode_hex(), - end_height: subtree.end_height, - }) - .collect(); + let subtrees = match response { + zebra_state::ReadResponse::SaplingSubtrees(subtrees) => subtrees, + _ => unreachable!("unmatched response to a subtrees request"), + }; - Ok(GetSubtrees { - pool, - start_index, - subtrees, + let subtrees = subtrees + .values() + .map(|subtree| SubtreeRpcData { + root: subtree.root.encode_hex(), + end_height: subtree.end_height, }) - } else if pool == "orchard" { - let request = zebra_state::ReadRequest::OrchardSubtrees { start_index, limit }; - let response = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; - - let subtrees = match response { - zebra_state::ReadResponse::OrchardSubtrees(subtrees) => subtrees, - _ => unreachable!("unmatched response to a subtrees request"), - }; + .collect(); - let subtrees = subtrees - .values() - .map(|subtree| SubtreeRpcData { - root: subtree.root.encode_hex(), - end_height: subtree.end_height, - }) - .collect(); + Ok(GetSubtrees { + pool, + start_index, + subtrees, + }) + } else if pool == "orchard" { + let request = zebra_state::ReadRequest::OrchardSubtrees { start_index, limit }; + let response = state + .ready() + .and_then(|service| service.call(request)) + .await + .map_misc_error()?; - Ok(GetSubtrees { - pool, - start_index, - subtrees, - }) - } else { - Err(Error { - code: INVALID_PARAMETERS_ERROR_CODE, - message: format!("invalid pool name, must be one of: {:?}", POOL_LIST), - data: None, + let subtrees = match response { + zebra_state::ReadResponse::OrchardSubtrees(subtrees) => subtrees, + _ => unreachable!("unmatched response to a subtrees request"), + }; + + let subtrees = subtrees + .values() + .map(|subtree| SubtreeRpcData { + root: subtree.root.encode_hex(), + end_height: subtree.end_height, }) - } + .collect(); + + Ok(GetSubtrees { + pool, + start_index, + subtrees, + }) + } else { + Err(ErrorObject::owned( + server::error::LegacyCode::Misc.into(), + format!("invalid pool name, must be one of: {:?}", POOL_LIST).as_str(), + None::<()>, + )) } - .boxed() } - fn get_address_tx_ids( - &self, - request: GetAddressTxIdsRequest, - ) -> BoxFuture>> { + async fn get_address_tx_ids(&self, request: GetAddressTxIdsRequest) -> Result> { let mut state = self.state.clone(); let latest_chain_tip = self.latest_chain_tip.clone(); let start = Height(request.start); let end = Height(request.end); - async move { - let chain_height = best_chain_tip_height(&latest_chain_tip)?; + let chain_height = best_chain_tip_height(&latest_chain_tip)?; - // height range checks - check_height_range(start, end, chain_height)?; + // height range checks + check_height_range(start, end, chain_height)?; - let valid_addresses = AddressStrings { - addresses: request.addresses, - } - .valid_addresses()?; + let valid_addresses = AddressStrings { + addresses: request.addresses, + } + .valid_addresses()?; - let request = zebra_state::ReadRequest::TransactionIdsByAddresses { - addresses: valid_addresses, - height_range: start..=end, - }; - let response = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; - - let hashes = match response { - zebra_state::ReadResponse::AddressesTransactionIds(hashes) => { - let mut last_tx_location = TransactionLocation::from_usize(Height(0), 0); - - hashes - .iter() - .map(|(tx_loc, tx_id)| { - // Check that the returned transactions are in chain order. - assert!( - *tx_loc > last_tx_location, - "Transactions were not in chain order:\n\ + let request = zebra_state::ReadRequest::TransactionIdsByAddresses { + addresses: valid_addresses, + height_range: start..=end, + }; + let response = state + .ready() + .and_then(|service| service.call(request)) + .await + .map_misc_error()?; + + let hashes = match response { + zebra_state::ReadResponse::AddressesTransactionIds(hashes) => { + let mut last_tx_location = TransactionLocation::from_usize(Height(0), 0); + + hashes + .iter() + .map(|(tx_loc, tx_id)| { + // Check that the returned transactions are in chain order. + assert!( + *tx_loc > last_tx_location, + "Transactions were not in chain order:\n\ {tx_loc:?} {tx_id:?} was after:\n\ {last_tx_location:?}", - ); + ); - last_tx_location = *tx_loc; + last_tx_location = *tx_loc; - tx_id.to_string() - }) - .collect() - } - _ => unreachable!("unmatched response to a TransactionsByAddresses request"), - }; + tx_id.to_string() + }) + .collect() + } + _ => unreachable!("unmatched response to a TransactionsByAddresses request"), + }; - Ok(hashes) - } - .boxed() + Ok(hashes) } - fn get_address_utxos( + async fn get_address_utxos( &self, address_strings: AddressStrings, - ) -> BoxFuture>> { + ) -> Result> { let mut state = self.state.clone(); let mut response_utxos = vec![]; - async move { - let valid_addresses = address_strings.valid_addresses()?; - - // get utxos data for addresses - let request = zebra_state::ReadRequest::UtxosByAddresses(valid_addresses); - let response = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; - let utxos = match response { - zebra_state::ReadResponse::AddressUtxos(utxos) => utxos, - _ => unreachable!("unmatched response to a UtxosByAddresses request"), - }; + let valid_addresses = address_strings.valid_addresses()?; + + // get utxos data for addresses + let request = zebra_state::ReadRequest::UtxosByAddresses(valid_addresses); + let response = state + .ready() + .and_then(|service| service.call(request)) + .await + .map_misc_error()?; + let utxos = match response { + zebra_state::ReadResponse::AddressUtxos(utxos) => utxos, + _ => unreachable!("unmatched response to a UtxosByAddresses request"), + }; - let mut last_output_location = OutputLocation::from_usize(Height(0), 0, 0); - - for utxo_data in utxos.utxos() { - let address = utxo_data.0; - let txid = *utxo_data.1; - let height = utxo_data.2.height(); - let output_index = utxo_data.2.output_index(); - let script = utxo_data.3.lock_script.clone(); - let satoshis = u64::from(utxo_data.3.value); - - let output_location = *utxo_data.2; - // Check that the returned UTXOs are in chain order. - assert!( - output_location > last_output_location, - "UTXOs were not in chain order:\n\ + let mut last_output_location = OutputLocation::from_usize(Height(0), 0, 0); + + for utxo_data in utxos.utxos() { + let address = utxo_data.0; + let txid = *utxo_data.1; + let height = utxo_data.2.height(); + let output_index = utxo_data.2.output_index(); + let script = utxo_data.3.lock_script.clone(); + let satoshis = u64::from(utxo_data.3.value); + + let output_location = *utxo_data.2; + // Check that the returned UTXOs are in chain order. + assert!( + output_location > last_output_location, + "UTXOs were not in chain order:\n\ {output_location:?} {address:?} {txid:?} was after:\n\ {last_output_location:?}", - ); - - let entry = GetAddressUtxos { - address, - txid, - output_index, - script, - satoshis, - height, - }; - response_utxos.push(entry); + ); - last_output_location = output_location; - } + let entry = GetAddressUtxos { + address, + txid, + output_index, + script, + satoshis, + height, + }; + response_utxos.push(entry); - Ok(response_utxos) + last_output_location = output_location; } - .boxed() + + Ok(response_utxos) } fn stop(&self) -> Result { @@ -1462,25 +1435,25 @@ where if self.network.is_regtest() { match nix::sys::signal::raise(nix::sys::signal::SIGINT) { Ok(_) => Ok("Zebra server stopping".to_string()), - Err(error) => Err(Error { - code: ErrorCode::InternalError, - message: format!("Failed to shut down: {}", error), - data: None, - }), + Err(error) => Err(ErrorObject::owned( + ErrorCode::InternalError.code(), + format!("Failed to shut down: {}", error).as_str(), + None::<()>, + )), } } else { - Err(Error { - code: ErrorCode::MethodNotFound, - message: "stop is only available on regtest networks".to_string(), - data: None, - }) + Err(ErrorObject::borrowed( + ErrorCode::MethodNotFound.code(), + "stop is only available on regtest networks", + None, + )) } #[cfg(target_os = "windows")] - Err(Error { - code: ErrorCode::MethodNotFound, - message: "stop is not available in windows targets".to_string(), - data: None, - }) + Err(ErrorObject::borrowed( + ErrorCode::MethodNotFound.code(), + "stop is not available in windows targets", + None, + )) } } @@ -1492,7 +1465,7 @@ where { latest_chain_tip .best_tip_height() - .ok_or_server_error("No blocks in state") + .ok_or_misc_error("No blocks in state") } /// Response to a `getinfo` RPC request. @@ -1567,8 +1540,8 @@ impl Default for GetBlockChainInfo { /// A wrapper type with a list of transparent address strings. /// -/// This is used for the input parameter of [`Rpc::get_address_balance`], -/// [`Rpc::get_address_tx_ids`] and [`Rpc::get_address_utxos`]. +/// This is used for the input parameter of [`RpcServer::get_address_balance`], +/// [`RpcServer::get_address_tx_ids`] and [`RpcServer::get_address_utxos`]. #[derive(Clone, Debug, Eq, PartialEq, Hash, serde::Deserialize)] pub struct AddressStrings { /// A list of transparent address strings. @@ -1586,13 +1559,15 @@ impl AddressStrings { /// - check if provided list have all valid transparent addresses. /// - return valid addresses as a set of `Address`. pub fn valid_addresses(self) -> Result> { + // Reference for the legacy error code: + // let valid_addresses: HashSet
= self .addresses .into_iter() .map(|address| { - address.parse().map_err(|error| { - Error::invalid_params(format!("invalid address {address:?}: {error}")) - }) + address + .parse() + .map_error(server::error::LegacyCode::InvalidAddressOrKey) }) .collect::>()?; @@ -1723,7 +1698,7 @@ impl Default for SentTransactionHash { /// Response to a `getblock` RPC request. /// -/// See the notes for the [`Rpc::get_block`] method. +/// See the notes for the [`RpcServer::get_block`] method. #[derive(Clone, Debug, PartialEq, serde::Serialize)] #[serde(untagged)] #[allow(clippy::large_enum_variant)] //TODO: create a struct for the Object and Box it @@ -1771,11 +1746,9 @@ pub enum GetBlock { // `chainhistoryroot` would be here. Undocumented. TODO: decide if we want to support it // - /// List of transaction IDs in block order, hex-encoded. - // - // TODO: use a typed Vec here - // TODO: support Objects - tx: Vec, + /// List of transactions in block order, hex-encoded if verbosity=1 or + /// as objects if verbosity=2. + tx: Vec, /// The height of the requested block. #[serde(skip_serializing_if = "Option::is_none")] @@ -1803,7 +1776,7 @@ pub enum GetBlock { difficulty: Option, // `chainwork` would be here, but we don't plan on supporting it - // `anchor` would be here. Undocumented. TODO: decide if we want to support it + // `anchor` would be here. Not planned to be supported. // `chainSupply` would be here, TODO: implement // `valuePools` would be here, TODO: implement // @@ -1844,9 +1817,20 @@ impl Default for GetBlock { } } +#[derive(Clone, Debug, PartialEq, serde::Serialize)] +#[serde(untagged)] +/// The transaction list in a `getblock` call. Can be a list of transaction +/// IDs or the full transaction details depending on verbosity. +pub enum GetBlockTransaction { + /// The transaction hash, hex-encoded. + Hash(#[serde(with = "hex")] transaction::Hash), + /// The block object. + Object(TransactionObject), +} + /// Response to a `getblockheader` RPC request. /// -/// See the notes for the [`Rpc::get_block_header`] method. +/// See the notes for the [`RpcServer::get_block_header`] method. #[derive(Clone, Debug, PartialEq, serde::Serialize)] #[serde(untagged)] pub enum GetBlockHeader { @@ -1860,7 +1844,7 @@ pub enum GetBlockHeader { #[derive(Clone, Debug, PartialEq, serde::Serialize)] /// Verbose response to a `getblockheader` RPC request. /// -/// See the notes for the [`Rpc::get_block_header`] method. +/// See the notes for the [`RpcServer::get_block_header`] method. pub struct GetBlockHeaderObject { /// The hash of the requested block. pub hash: GetBlockHash, @@ -1949,7 +1933,7 @@ impl Default for GetBlockHeaderObject { /// /// Contains the hex-encoded hash of the requested block. /// -/// Also see the notes for the [`Rpc::get_best_block_hash`] and `get_block_hash` methods. +/// Also see the notes for the [`RpcServer::get_best_block_hash`] and `get_block_hash` methods. #[derive(Copy, Clone, Debug, Eq, PartialEq, serde::Deserialize, serde::Serialize)] #[serde(transparent)] pub struct GetBlockHash(#[serde(with = "hex")] pub block::Hash); @@ -1987,27 +1971,57 @@ pub enum GetRawTransaction { /// The raw transaction, encoded as hex bytes. Raw(#[serde(with = "hex")] SerializedTransaction), /// The transaction object. - Object { - /// The raw transaction, encoded as hex bytes. - #[serde(with = "hex")] - hex: SerializedTransaction, - /// The height of the block in the best chain that contains the transaction, or -1 if - /// the transaction is in the mempool. - height: i32, - /// The confirmations of the block in the best chain that contains the transaction, - /// or 0 if the transaction is in the mempool. - confirmations: u32, - }, + Object(TransactionObject), } impl Default for GetRawTransaction { fn default() -> Self { - Self::Object { + Self::Object(TransactionObject::default()) + } +} + +/// A Transaction object as returned by `getrawtransaction` and `getblock` RPC +/// requests. +#[derive(Clone, Debug, Eq, PartialEq, serde::Serialize)] +pub struct TransactionObject { + /// The raw transaction, encoded as hex bytes. + #[serde(with = "hex")] + pub hex: SerializedTransaction, + /// The height of the block in the best chain that contains the tx or `None` if the tx is in + /// the mempool. + #[serde(skip_serializing_if = "Option::is_none")] + pub height: Option, + /// The height diff between the block containing the tx and the best chain tip + 1 or `None` + /// if the tx is in the mempool. + #[serde(skip_serializing_if = "Option::is_none")] + pub confirmations: Option, + // TODO: many fields not yet supported +} + +impl Default for TransactionObject { + fn default() -> Self { + Self { hex: SerializedTransaction::from( [0u8; zebra_chain::transaction::MIN_TRANSPARENT_TX_SIZE as usize].to_vec(), ), - height: i32::default(), - confirmations: u32::default(), + height: Option::default(), + confirmations: Option::default(), + } + } +} + +impl TransactionObject { + /// Converts `tx` and `height` into a new `GetRawTransaction` in the `verbose` format. + #[allow(clippy::unwrap_in_result)] + fn from_transaction( + tx: Arc, + height: Option, + confirmations: Option, + ) -> Self { + Self { + hex: tx.into(), + height: height.map(|height| height.0), + confirmations, } } } @@ -2070,33 +2084,6 @@ pub struct GetAddressTxIdsRequest { end: u32, } -impl GetRawTransaction { - /// Converts `tx` and `height` into a new `GetRawTransaction` in the `verbose` format. - #[allow(clippy::unwrap_in_result)] - fn from_transaction( - tx: Arc, - height: Option, - confirmations: u32, - verbose: bool, - ) -> Self { - if verbose { - GetRawTransaction::Object { - hex: tx.into(), - height: match height { - Some(height) => height - .0 - .try_into() - .expect("valid block heights are limited to i32::MAX"), - None => -1, - }, - confirmations, - } - } else { - GetRawTransaction::Raw(tx.into()) - } - } -} - /// Information about the sapling and orchard note commitment trees if any. #[derive(Copy, Clone, Debug, Eq, PartialEq, serde::Deserialize, serde::Serialize)] pub struct GetBlockTrees { @@ -2162,19 +2149,25 @@ impl OrchardTrees { /// Check if provided height range is valid for address indexes. fn check_height_range(start: Height, end: Height, chain_height: Height) -> Result<()> { if start == Height(0) || end == Height(0) { - return Err(Error::invalid_params(format!( - "start {start:?} and end {end:?} must both be greater than zero" - ))); + return Err(ErrorObject::owned( + ErrorCode::InvalidParams.code(), + format!("start {start:?} and end {end:?} must both be greater than zero"), + None::<()>, + )); } if start > end { - return Err(Error::invalid_params(format!( - "start {start:?} must be less than or equal to end {end:?}" - ))); + return Err(ErrorObject::owned( + ErrorCode::InvalidParams.code(), + format!("start {start:?} must be less than or equal to end {end:?}"), + None::<()>, + )); } if start > chain_height || end > chain_height { - return Err(Error::invalid_params(format!( - "start {start:?} and end {end:?} must both be less than or equal to the chain tip {chain_height:?}" - ))); + return Err(ErrorObject::owned( + ErrorCode::InvalidParams.code(), + format!("start {start:?} and end {end:?} must both be less than or equal to the chain tip {chain_height:?}"), + None::<()>, + )); } Ok(()) @@ -2192,8 +2185,10 @@ pub fn height_from_signed_int(index: i32, tip_height: Height) -> Result if index >= 0 { let height = index.try_into().expect("Positive i32 always fits in u32"); if height > tip_height.0 { - return Err(Error::invalid_params( + return Err(ErrorObject::borrowed( + ErrorCode::InvalidParams.code(), "Provided index is greater than the current tip", + None, )); } Ok(Height(height)) @@ -2204,17 +2199,27 @@ pub fn height_from_signed_int(index: i32, tip_height: Height) -> Result .checked_add(index + 1); let sanitized_height = match height { - None => return Err(Error::invalid_params("Provided index is not valid")), + None => { + return Err(ErrorObject::borrowed( + ErrorCode::InvalidParams.code(), + "Provided index is not valid", + None, + )) + } Some(h) => { if h < 0 { - return Err(Error::invalid_params( + return Err(ErrorObject::borrowed( + ErrorCode::InvalidParams.code(), "Provided negative index ends up with a negative height", + None, )); } let h: u32 = h.try_into().expect("Positive i32 always fits in u32"); if h > tip_height.0 { - return Err(Error::invalid_params( + return Err(ErrorObject::borrowed( + ErrorCode::InvalidParams.code(), "Provided index is greater than the current tip", + None, )); } diff --git a/zebra-rpc/src/methods/errors.rs b/zebra-rpc/src/methods/errors.rs deleted file mode 100644 index be9231d058d..00000000000 --- a/zebra-rpc/src/methods/errors.rs +++ /dev/null @@ -1,37 +0,0 @@ -//! Error conversions for Zebra's RPC methods. - -use jsonrpc_core::ErrorCode; - -pub(crate) trait MapServerError { - fn map_server_error(self) -> std::result::Result; -} - -pub(crate) trait OkOrServerError { - fn ok_or_server_error( - self, - message: S, - ) -> std::result::Result; -} - -impl MapServerError for Result -where - E: ToString, -{ - fn map_server_error(self) -> Result { - self.map_err(|error| jsonrpc_core::Error { - code: ErrorCode::ServerError(0), - message: error.to_string(), - data: None, - }) - } -} - -impl OkOrServerError for Option { - fn ok_or_server_error(self, message: S) -> Result { - self.ok_or(jsonrpc_core::Error { - code: ErrorCode::ServerError(0), - message: message.to_string(), - data: None, - }) - } -} diff --git a/zebra-rpc/src/methods/get_block_template_rpcs.rs b/zebra-rpc/src/methods/get_block_template_rpcs.rs index aed926b3635..2bb9a0ca393 100644 --- a/zebra-rpc/src/methods/get_block_template_rpcs.rs +++ b/zebra-rpc/src/methods/get_block_template_rpcs.rs @@ -2,9 +2,10 @@ use std::{fmt::Debug, sync::Arc, time::Duration}; -use futures::{future::OptionFuture, FutureExt, TryFutureExt}; -use jsonrpc_core::{self, BoxFuture, Error, ErrorCode, Result}; -use jsonrpc_derive::rpc; +use futures::{future::OptionFuture, TryFutureExt}; +use jsonrpsee::core::{async_trait, RpcResult as Result}; +use jsonrpsee_proc_macros::rpc; +use jsonrpsee_types::ErrorObject; use tower::{Service, ServiceExt}; use zcash_address::{unified::Encoding, TryFromAddress}; @@ -32,35 +33,37 @@ use zebra_network::AddressBookPeers; use zebra_node_services::mempool; use zebra_state::{ReadRequest, ReadResponse}; -use crate::methods::{ - best_chain_tip_height, - errors::MapServerError, - get_block_template_rpcs::{ - constants::{ - DEFAULT_SOLUTION_RATE_WINDOW_SIZE, GET_BLOCK_TEMPLATE_MEMPOOL_LONG_POLL_INTERVAL, - ZCASHD_FUNDING_STREAM_ORDER, - }, - get_block_template::{ - check_miner_address, check_synced_to_tip, fetch_mempool_transactions, - fetch_state_tip_and_local_time, validate_block_proposal, - }, - // TODO: move the types/* modules directly under get_block_template_rpcs, - // and combine any modules with the same names. - types::{ +use crate::{ + methods::{ + best_chain_tip_height, + get_block_template_rpcs::{ + constants::{ + DEFAULT_SOLUTION_RATE_WINDOW_SIZE, GET_BLOCK_TEMPLATE_MEMPOOL_LONG_POLL_INTERVAL, + ZCASHD_FUNDING_STREAM_ORDER, + }, get_block_template::{ - proposal::TimeSource, proposal_block_from_template, GetBlockTemplate, + check_miner_address, check_synced_to_tip, fetch_mempool_transactions, + fetch_state_tip_and_local_time, validate_block_proposal, + }, + // TODO: move the types/* modules directly under get_block_template_rpcs, + // and combine any modules with the same names. + types::{ + get_block_template::{ + proposal::TimeSource, proposal_block_from_template, GetBlockTemplate, + }, + get_mining_info, + long_poll::LongPollInput, + peer_info::PeerInfo, + submit_block, + subsidy::{BlockSubsidy, FundingStream}, + unified_address, validate_address, z_validate_address, }, - get_mining_info, - long_poll::LongPollInput, - peer_info::PeerInfo, - submit_block, - subsidy::{BlockSubsidy, FundingStream}, - unified_address, validate_address, z_validate_address, }, + height_from_signed_int, + hex_data::HexData, + GetBlockHash, }, - height_from_signed_int, - hex_data::HexData, - GetBlockHash, MISSING_BLOCK_ERROR_CODE, + server::{self, error::MapError}, }; pub mod constants; @@ -81,7 +84,7 @@ pub trait GetBlockTemplateRpc { /// # Notes /// /// This rpc method is available only if zebra is built with `--features getblocktemplate-rpcs`. - #[rpc(name = "getblockcount")] + #[method(name = "getblockcount")] fn get_block_count(&self) -> Result; /// Returns the hash of the block of a given height iff the index argument correspond @@ -100,8 +103,8 @@ pub trait GetBlockTemplateRpc { /// - If `index` is positive then index = block height. /// - If `index` is negative then -1 is the last known valid block. /// - This rpc method is available only if zebra is built with `--features getblocktemplate-rpcs`. - #[rpc(name = "getblockhash")] - fn get_block_hash(&self, index: i32) -> BoxFuture>; + #[method(name = "getblockhash")] + async fn get_block_hash(&self, index: i32) -> Result; /// Returns a block template for mining new Zcash blocks. /// @@ -126,11 +129,11 @@ pub trait GetBlockTemplateRpc { /// so moving between chains and forking chains is very cheap. /// /// This rpc method is available only if zebra is built with `--features getblocktemplate-rpcs`. - #[rpc(name = "getblocktemplate")] - fn get_block_template( + #[method(name = "getblocktemplate")] + async fn get_block_template( &self, parameters: Option, - ) -> BoxFuture>; + ) -> Result; /// Submits block to the node to be validated and committed. /// Returns the [`submit_block::Response`] for the operation, as a JSON string. @@ -147,20 +150,20 @@ pub trait GetBlockTemplateRpc { /// # Notes /// /// - `jsonparametersobject` holds a single field, workid, that must be included in submissions if provided by the server. - #[rpc(name = "submitblock")] - fn submit_block( + #[method(name = "submitblock")] + async fn submit_block( &self, hex_data: HexData, _parameters: Option, - ) -> BoxFuture>; + ) -> Result; /// Returns mining-related information. /// /// zcashd reference: [`getmininginfo`](https://zcash.github.io/rpc/getmininginfo.html) /// method: post /// tags: mining - #[rpc(name = "getmininginfo")] - fn get_mining_info(&self) -> BoxFuture>; + #[method(name = "getmininginfo")] + async fn get_mining_info(&self) -> Result; /// Returns the estimated network solutions per second based on the last `num_blocks` before /// `height`. @@ -172,12 +175,9 @@ pub trait GetBlockTemplateRpc { /// zcashd reference: [`getnetworksolps`](https://zcash.github.io/rpc/getnetworksolps.html) /// method: post /// tags: mining - #[rpc(name = "getnetworksolps")] - fn get_network_sol_ps( - &self, - num_blocks: Option, - height: Option, - ) -> BoxFuture>; + #[method(name = "getnetworksolps")] + async fn get_network_sol_ps(&self, num_blocks: Option, height: Option) + -> Result; /// Returns the estimated network solutions per second based on the last `num_blocks` before /// `height`. @@ -188,13 +188,13 @@ pub trait GetBlockTemplateRpc { /// zcashd reference: [`getnetworkhashps`](https://zcash.github.io/rpc/getnetworkhashps.html) /// method: post /// tags: mining - #[rpc(name = "getnetworkhashps")] - fn get_network_hash_ps( + #[method(name = "getnetworkhashps")] + async fn get_network_hash_ps( &self, num_blocks: Option, height: Option, - ) -> BoxFuture> { - self.get_network_sol_ps(num_blocks, height) + ) -> Result { + self.get_network_sol_ps(num_blocks, height).await } /// Returns data about each connected network node. @@ -202,8 +202,8 @@ pub trait GetBlockTemplateRpc { /// zcashd reference: [`getpeerinfo`](https://zcash.github.io/rpc/getpeerinfo.html) /// method: post /// tags: network - #[rpc(name = "getpeerinfo")] - fn get_peer_info(&self) -> BoxFuture>>; + #[method(name = "getpeerinfo")] + async fn get_peer_info(&self) -> Result>; /// Checks if a zcash address is valid. /// Returns information about the given address if valid. @@ -219,8 +219,8 @@ pub trait GetBlockTemplateRpc { /// # Notes /// /// - No notes - #[rpc(name = "validateaddress")] - fn validate_address(&self, address: String) -> BoxFuture>; + #[method(name = "validateaddress")] + async fn validate_address(&self, address: String) -> Result; /// Checks if a zcash address is valid. /// Returns information about the given address if valid. @@ -236,11 +236,11 @@ pub trait GetBlockTemplateRpc { /// # Notes /// /// - No notes - #[rpc(name = "z_validateaddress")] - fn z_validate_address( + #[method(name = "z_validateaddress")] + async fn z_validate_address( &self, address: String, - ) -> BoxFuture>; + ) -> Result; /// Returns the block subsidy reward of the block at `height`, taking into account the mining slow start. /// Returns an error if `height` is less than the height of the first halving for the current network. @@ -256,16 +256,16 @@ pub trait GetBlockTemplateRpc { /// # Notes /// /// If `height` is not supplied, uses the tip height. - #[rpc(name = "getblocksubsidy")] - fn get_block_subsidy(&self, height: Option) -> BoxFuture>; + #[method(name = "getblocksubsidy")] + async fn get_block_subsidy(&self, height: Option) -> Result; /// Returns the proof-of-work difficulty as a multiple of the minimum difficulty. /// /// zcashd reference: [`getdifficulty`](https://zcash.github.io/rpc/getdifficulty.html) /// method: post /// tags: blockchain - #[rpc(name = "getdifficulty")] - fn get_difficulty(&self) -> BoxFuture>; + #[method(name = "getdifficulty")] + async fn get_difficulty(&self) -> Result; /// Returns the list of individual payment addresses given a unified address. /// @@ -280,13 +280,10 @@ pub trait GetBlockTemplateRpc { /// # Notes /// /// - No notes - #[rpc(name = "z_listunifiedreceivers")] - fn z_list_unified_receivers( - &self, - address: String, - ) -> BoxFuture>; + #[method(name = "z_listunifiedreceivers")] + async fn z_list_unified_receivers(&self, address: String) -> Result; - #[rpc(name = "generate")] + #[method(name = "generate")] /// Mine blocks immediately. Returns the block hashes of the generated blocks. /// /// # Parameters @@ -300,7 +297,7 @@ pub trait GetBlockTemplateRpc { /// zcashd reference: [`generate`](https://zcash.github.io/rpc/generate.html) /// method: post /// tags: generating - fn generate(&self, num_blocks: u32) -> BoxFuture>>; + async fn generate(&self, num_blocks: u32) -> Result>; } /// RPC method implementations. @@ -534,7 +531,8 @@ where } } -impl GetBlockTemplateRpc +#[async_trait] +impl GetBlockTemplateRpcServer for GetBlockTemplateRpcImpl where Mempool: Service< @@ -569,40 +567,37 @@ where best_chain_tip_height(&self.latest_chain_tip).map(|height| height.0) } - fn get_block_hash(&self, index: i32) -> BoxFuture> { + async fn get_block_hash(&self, index: i32) -> Result { let mut state = self.state.clone(); let latest_chain_tip = self.latest_chain_tip.clone(); - async move { - // TODO: look up this height as part of the state request? - let tip_height = best_chain_tip_height(&latest_chain_tip)?; - - let height = height_from_signed_int(index, tip_height)?; - - let request = zebra_state::ReadRequest::BestChainBlockHash(height); - let response = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; - - match response { - zebra_state::ReadResponse::BlockHash(Some(hash)) => Ok(GetBlockHash(hash)), - zebra_state::ReadResponse::BlockHash(None) => Err(Error { - code: MISSING_BLOCK_ERROR_CODE, - message: "Block not found".to_string(), - data: None, - }), - _ => unreachable!("unmatched response to a block request"), - } + // TODO: look up this height as part of the state request? + let tip_height = best_chain_tip_height(&latest_chain_tip)?; + + let height = height_from_signed_int(index, tip_height)?; + + let request = zebra_state::ReadRequest::BestChainBlockHash(height); + let response = state + .ready() + .and_then(|service| service.call(request)) + .await + .map_error(server::error::LegacyCode::default())?; + + match response { + zebra_state::ReadResponse::BlockHash(Some(hash)) => Ok(GetBlockHash(hash)), + zebra_state::ReadResponse::BlockHash(None) => Err(ErrorObject::borrowed( + server::error::LegacyCode::InvalidParameter.into(), + "Block not found", + None, + )), + _ => unreachable!("unmatched response to a block request"), } - .boxed() } - fn get_block_template( + async fn get_block_template( &self, parameters: Option, - ) -> BoxFuture> { + ) -> Result { // Clone Configs let network = self.network.clone(); let miner_address = self.miner_address.clone(); @@ -626,399 +621,392 @@ where latest_chain_tip, sync_status, ) - .boxed(); + .await; } // To implement long polling correctly, we split this RPC into multiple phases. - async move { - get_block_template::check_parameters(¶meters)?; - - let client_long_poll_id = parameters.as_ref().and_then(|params| params.long_poll_id); - - // - One-off checks - - // Check config and parameters. - // These checks always have the same result during long polling. - let miner_address = check_miner_address(miner_address)?; + get_block_template::check_parameters(¶meters)?; + + let client_long_poll_id = parameters.as_ref().and_then(|params| params.long_poll_id); + + // - One-off checks + + // Check config and parameters. + // These checks always have the same result during long polling. + let miner_address = check_miner_address(miner_address)?; + + // - Checks and fetches that can change during long polling + // + // Set up the loop. + let mut max_time_reached = false; + + // The loop returns the server long poll ID, + // which should be different to the client long poll ID. + let ( + server_long_poll_id, + chain_tip_and_local_time, + mempool_txs, + mempool_tx_deps, + submit_old, + ) = loop { + // Check if we are synced to the tip. + // The result of this check can change during long polling. + // + // Optional TODO: + // - add `async changed()` method to ChainSyncStatus (like `ChainTip`) + check_synced_to_tip(&network, latest_chain_tip.clone(), sync_status.clone())?; + // TODO: return an error if we have no peers, like `zcashd` does, + // and add a developer config that mines regardless of how many peers we have. + // https://github.com/zcash/zcash/blob/6fdd9f1b81d3b228326c9826fa10696fc516444b/src/miner.cpp#L865-L880 + + // We're just about to fetch state data, then maybe wait for any changes. + // Mark all the changes before the fetch as seen. + // Changes are also ignored in any clones made after the mark. + latest_chain_tip.mark_best_tip_seen(); + + // Fetch the state data and local time for the block template: + // - if the tip block hash changes, we must return from long polling, + // - if the local clock changes on testnet, we might return from long polling + // + // We always return after 90 minutes on mainnet, even if we have the same response, + // because the max time has been reached. + let chain_tip_and_local_time @ zebra_state::GetBlockTemplateChainInfo { + tip_hash, + tip_height, + max_time, + cur_time, + .. + } = fetch_state_tip_and_local_time(state.clone()).await?; - // - Checks and fetches that can change during long polling + // Fetch the mempool data for the block template: + // - if the mempool transactions change, we might return from long polling. // - // Set up the loop. - let mut max_time_reached = false; - - // The loop returns the server long poll ID, - // which should be different to the client long poll ID. - let ( - server_long_poll_id, - chain_tip_and_local_time, - mempool_txs, - mempool_tx_deps, - submit_old, - ) = loop { - // Check if we are synced to the tip. - // The result of this check can change during long polling. - // - // Optional TODO: - // - add `async changed()` method to ChainSyncStatus (like `ChainTip`) - check_synced_to_tip(&network, latest_chain_tip.clone(), sync_status.clone())?; - // TODO: return an error if we have no peers, like `zcashd` does, - // and add a developer config that mines regardless of how many peers we have. - // https://github.com/zcash/zcash/blob/6fdd9f1b81d3b228326c9826fa10696fc516444b/src/miner.cpp#L865-L880 - - // We're just about to fetch state data, then maybe wait for any changes. - // Mark all the changes before the fetch as seen. - // Changes are also ignored in any clones made after the mark. - latest_chain_tip.mark_best_tip_seen(); - - // Fetch the state data and local time for the block template: - // - if the tip block hash changes, we must return from long polling, - // - if the local clock changes on testnet, we might return from long polling - // - // We always return after 90 minutes on mainnet, even if we have the same response, - // because the max time has been reached. - let chain_tip_and_local_time @ zebra_state::GetBlockTemplateChainInfo { - tip_hash, - tip_height, - max_time, - cur_time, - .. - } = fetch_state_tip_and_local_time(state.clone()).await?; - - // Fetch the mempool data for the block template: - // - if the mempool transactions change, we might return from long polling. - // - // If the chain fork has just changed, miners want to get the new block as fast - // as possible, rather than wait for transactions to re-verify. This increases - // miner profits (and any delays can cause chain forks). So we don't wait between - // the chain tip changing and getting mempool transactions. - // - // Optional TODO: - // - add a `MempoolChange` type with an `async changed()` method (like `ChainTip`) - let Some((mempool_txs, mempool_tx_deps)) = - fetch_mempool_transactions(mempool.clone(), tip_hash) - .await? - // If the mempool and state responses are out of sync: - // - if we are not long polling, omit mempool transactions from the template, - // - if we are long polling, continue to the next iteration of the loop to make fresh state and mempool requests. - .or_else(|| client_long_poll_id.is_none().then(Default::default)) - else { - continue; - }; - - // - Long poll ID calculation - let server_long_poll_id = LongPollInput::new( - tip_height, - tip_hash, - max_time, - mempool_txs.iter().map(|tx| tx.transaction.id), - ) - .generate_id(); - - // The loop finishes if: - // - the client didn't pass a long poll ID, - // - the server long poll ID is different to the client long poll ID, or - // - the previous loop iteration waited until the max time. - if Some(&server_long_poll_id) != client_long_poll_id.as_ref() || max_time_reached { - let mut submit_old = client_long_poll_id - .as_ref() - .map(|old_long_poll_id| server_long_poll_id.submit_old(old_long_poll_id)); - - // On testnet, the max time changes the block difficulty, so old shares are - // invalid. On mainnet, this means there has been 90 minutes without a new - // block or mempool transaction, which is very unlikely. So the miner should - // probably reset anyway. - if max_time_reached { - submit_old = Some(false); - } + // If the chain fork has just changed, miners want to get the new block as fast + // as possible, rather than wait for transactions to re-verify. This increases + // miner profits (and any delays can cause chain forks). So we don't wait between + // the chain tip changing and getting mempool transactions. + // + // Optional TODO: + // - add a `MempoolChange` type with an `async changed()` method (like `ChainTip`) + let Some((mempool_txs, mempool_tx_deps)) = + fetch_mempool_transactions(mempool.clone(), tip_hash) + .await? + // If the mempool and state responses are out of sync: + // - if we are not long polling, omit mempool transactions from the template, + // - if we are long polling, continue to the next iteration of the loop to make fresh state and mempool requests. + .or_else(|| client_long_poll_id.is_none().then(Default::default)) + else { + continue; + }; - break ( - server_long_poll_id, - chain_tip_and_local_time, - mempool_txs, - mempool_tx_deps, - submit_old, - ); + // - Long poll ID calculation + let server_long_poll_id = LongPollInput::new( + tip_height, + tip_hash, + max_time, + mempool_txs.iter().map(|tx| tx.transaction.id), + ) + .generate_id(); + + // The loop finishes if: + // - the client didn't pass a long poll ID, + // - the server long poll ID is different to the client long poll ID, or + // - the previous loop iteration waited until the max time. + if Some(&server_long_poll_id) != client_long_poll_id.as_ref() || max_time_reached { + let mut submit_old = client_long_poll_id + .as_ref() + .map(|old_long_poll_id| server_long_poll_id.submit_old(old_long_poll_id)); + + // On testnet, the max time changes the block difficulty, so old shares are + // invalid. On mainnet, this means there has been 90 minutes without a new + // block or mempool transaction, which is very unlikely. So the miner should + // probably reset anyway. + if max_time_reached { + submit_old = Some(false); } - // - Polling wait conditions - // - // TODO: when we're happy with this code, split it into a function. - // - // Periodically check the mempool for changes. - // - // Optional TODO: - // Remove this polling wait if we switch to using futures to detect sync status - // and mempool changes. - let wait_for_mempool_request = tokio::time::sleep(Duration::from_secs( - GET_BLOCK_TEMPLATE_MEMPOOL_LONG_POLL_INTERVAL, - )); + break ( + server_long_poll_id, + chain_tip_and_local_time, + mempool_txs, + mempool_tx_deps, + submit_old, + ); + } - // Return immediately if the chain tip has changed. - // The clone preserves the seen status of the chain tip. - let mut wait_for_best_tip_change = latest_chain_tip.clone(); - let wait_for_best_tip_change = wait_for_best_tip_change.best_tip_changed(); - - // Wait for the maximum block time to elapse. This can change the block header - // on testnet. (On mainnet it can happen due to a network disconnection, or a - // rapid drop in hash rate.) - // - // This duration might be slightly lower than the actual maximum, - // if cur_time was clamped to min_time. In that case the wait is very long, - // and it's ok to return early. - // - // It can also be zero if cur_time was clamped to max_time. In that case, - // we want to wait for another change, and ignore this timeout. So we use an - // `OptionFuture::None`. - let duration_until_max_time = max_time.saturating_duration_since(cur_time); - let wait_for_max_time: OptionFuture<_> = if duration_until_max_time.seconds() > 0 { - Some(tokio::time::sleep(duration_until_max_time.to_std())) - } else { - None + // - Polling wait conditions + // + // TODO: when we're happy with this code, split it into a function. + // + // Periodically check the mempool for changes. + // + // Optional TODO: + // Remove this polling wait if we switch to using futures to detect sync status + // and mempool changes. + let wait_for_mempool_request = tokio::time::sleep(Duration::from_secs( + GET_BLOCK_TEMPLATE_MEMPOOL_LONG_POLL_INTERVAL, + )); + + // Return immediately if the chain tip has changed. + // The clone preserves the seen status of the chain tip. + let mut wait_for_best_tip_change = latest_chain_tip.clone(); + let wait_for_best_tip_change = wait_for_best_tip_change.best_tip_changed(); + + // Wait for the maximum block time to elapse. This can change the block header + // on testnet. (On mainnet it can happen due to a network disconnection, or a + // rapid drop in hash rate.) + // + // This duration might be slightly lower than the actual maximum, + // if cur_time was clamped to min_time. In that case the wait is very long, + // and it's ok to return early. + // + // It can also be zero if cur_time was clamped to max_time. In that case, + // we want to wait for another change, and ignore this timeout. So we use an + // `OptionFuture::None`. + let duration_until_max_time = max_time.saturating_duration_since(cur_time); + let wait_for_max_time: OptionFuture<_> = if duration_until_max_time.seconds() > 0 { + Some(tokio::time::sleep(duration_until_max_time.to_std())) + } else { + None + } + .into(); + + // Optional TODO: + // `zcashd` generates the next coinbase transaction while waiting for changes. + // When Zebra supports shielded coinbase, we might want to do this in parallel. + // But the coinbase value depends on the selected transactions, so this needs + // further analysis to check if it actually saves us any time. + + tokio::select! { + // Poll the futures in the listed order, for efficiency. + // We put the most frequent conditions first. + biased; + + // This timer elapses every few seconds + _elapsed = wait_for_mempool_request => { + tracing::debug!( + ?max_time, + ?cur_time, + ?server_long_poll_id, + ?client_long_poll_id, + GET_BLOCK_TEMPLATE_MEMPOOL_LONG_POLL_INTERVAL, + "checking for a new mempool change after waiting a few seconds" + ); } - .into(); - - // Optional TODO: - // `zcashd` generates the next coinbase transaction while waiting for changes. - // When Zebra supports shielded coinbase, we might want to do this in parallel. - // But the coinbase value depends on the selected transactions, so this needs - // further analysis to check if it actually saves us any time. - - tokio::select! { - // Poll the futures in the listed order, for efficiency. - // We put the most frequent conditions first. - biased; - - // This timer elapses every few seconds - _elapsed = wait_for_mempool_request => { - tracing::debug!( - ?max_time, - ?cur_time, - ?server_long_poll_id, - ?client_long_poll_id, - GET_BLOCK_TEMPLATE_MEMPOOL_LONG_POLL_INTERVAL, - "checking for a new mempool change after waiting a few seconds" - ); - } - - // The state changes after around a target block interval (75s) - tip_changed_result = wait_for_best_tip_change => { - match tip_changed_result { - Ok(()) => { - // Spurious updates shouldn't happen in the state, because the - // difficulty and hash ordering is a stable total order. But - // since they could cause a busy-loop, guard against them here. - latest_chain_tip.mark_best_tip_seen(); - - let new_tip_hash = latest_chain_tip.best_tip_hash(); - if new_tip_hash == Some(tip_hash) { - tracing::debug!( - ?max_time, - ?cur_time, - ?server_long_poll_id, - ?client_long_poll_id, - ?tip_hash, - ?tip_height, - "ignoring spurious state change notification" - ); - - // Wait for the mempool interval, then check for any changes. - tokio::time::sleep(Duration::from_secs( - GET_BLOCK_TEMPLATE_MEMPOOL_LONG_POLL_INTERVAL, - )).await; - - continue; - } + // The state changes after around a target block interval (75s) + tip_changed_result = wait_for_best_tip_change => { + match tip_changed_result { + Ok(()) => { + // Spurious updates shouldn't happen in the state, because the + // difficulty and hash ordering is a stable total order. But + // since they could cause a busy-loop, guard against them here. + latest_chain_tip.mark_best_tip_seen(); + + let new_tip_hash = latest_chain_tip.best_tip_hash(); + if new_tip_hash == Some(tip_hash) { tracing::debug!( ?max_time, ?cur_time, ?server_long_poll_id, ?client_long_poll_id, - "returning from long poll because state has changed" + ?tip_hash, + ?tip_height, + "ignoring spurious state change notification" ); - } - Err(recv_error) => { - // This log is rare and helps with debugging, so it's ok to be info. - tracing::info!( - ?recv_error, - ?max_time, - ?cur_time, - ?server_long_poll_id, - ?client_long_poll_id, - "returning from long poll due to a state error.\ - Is Zebra shutting down?" - ); + // Wait for the mempool interval, then check for any changes. + tokio::time::sleep(Duration::from_secs( + GET_BLOCK_TEMPLATE_MEMPOOL_LONG_POLL_INTERVAL, + )).await; - return Err(recv_error).map_server_error(); + continue; } + + tracing::debug!( + ?max_time, + ?cur_time, + ?server_long_poll_id, + ?client_long_poll_id, + "returning from long poll because state has changed" + ); } - } - // The max time does not elapse during normal operation on mainnet, - // and it rarely elapses on testnet. - Some(_elapsed) = wait_for_max_time => { - // This log is very rare so it's ok to be info. - tracing::info!( - ?max_time, - ?cur_time, - ?server_long_poll_id, - ?client_long_poll_id, - "returning from long poll because max time was reached" - ); - - max_time_reached = true; + Err(recv_error) => { + // This log is rare and helps with debugging, so it's ok to be info. + tracing::info!( + ?recv_error, + ?max_time, + ?cur_time, + ?server_long_poll_id, + ?client_long_poll_id, + "returning from long poll due to a state error.\ + Is Zebra shutting down?" + ); + + return Err(recv_error).map_error(server::error::LegacyCode::default()); + } } } - }; - // - Processing fetched data to create a transaction template - // - // Apart from random weighted transaction selection, - // the template only depends on the previously fetched data. - // This processing never fails. - - // Calculate the next block height. - let next_block_height = - (chain_tip_and_local_time.tip_height + 1).expect("tip is far below Height::MAX"); - - tracing::debug!( - mempool_tx_hashes = ?mempool_txs - .iter() - .map(|tx| tx.transaction.id.mined_id()) - .collect::>(), - "selecting transactions for the template from the mempool" - ); + // The max time does not elapse during normal operation on mainnet, + // and it rarely elapses on testnet. + Some(_elapsed) = wait_for_max_time => { + // This log is very rare so it's ok to be info. + tracing::info!( + ?max_time, + ?cur_time, + ?server_long_poll_id, + ?client_long_poll_id, + "returning from long poll because max time was reached" + ); - // Randomly select some mempool transactions. - let mempool_txs = zip317::select_mempool_transactions( - &network, - next_block_height, - &miner_address, - mempool_txs, - mempool_tx_deps, - debug_like_zcashd, - extra_coinbase_data.clone(), - ); + max_time_reached = true; + } + } + }; + + // - Processing fetched data to create a transaction template + // + // Apart from random weighted transaction selection, + // the template only depends on the previously fetched data. + // This processing never fails. + + // Calculate the next block height. + let next_block_height = + (chain_tip_and_local_time.tip_height + 1).expect("tip is far below Height::MAX"); + + tracing::debug!( + mempool_tx_hashes = ?mempool_txs + .iter() + .map(|tx| tx.transaction.id.mined_id()) + .collect::>(), + "selecting transactions for the template from the mempool" + ); - tracing::debug!( - selected_mempool_tx_hashes = ?mempool_txs - .iter() - .map(|#[cfg(not(test))] tx, #[cfg(test)] (_, tx)| tx.transaction.id.mined_id()) - .collect::>(), - "selected transactions for the template from the mempool" - ); + // Randomly select some mempool transactions. + let mempool_txs = zip317::select_mempool_transactions( + &network, + next_block_height, + &miner_address, + mempool_txs, + mempool_tx_deps, + debug_like_zcashd, + extra_coinbase_data.clone(), + ); - // - After this point, the template only depends on the previously fetched data. - - let response = GetBlockTemplate::new( - &network, - &miner_address, - &chain_tip_and_local_time, - server_long_poll_id, - mempool_txs, - submit_old, - debug_like_zcashd, - extra_coinbase_data, - ); + tracing::debug!( + selected_mempool_tx_hashes = ?mempool_txs + .iter() + .map(|#[cfg(not(test))] tx, #[cfg(test)] (_, tx)| tx.transaction.id.mined_id()) + .collect::>(), + "selected transactions for the template from the mempool" + ); - Ok(response.into()) - } - .boxed() + // - After this point, the template only depends on the previously fetched data. + + let response = GetBlockTemplate::new( + &network, + &miner_address, + &chain_tip_and_local_time, + server_long_poll_id, + mempool_txs, + submit_old, + debug_like_zcashd, + extra_coinbase_data, + ); + + Ok(response.into()) } - fn submit_block( + async fn submit_block( &self, HexData(block_bytes): HexData, _parameters: Option, - ) -> BoxFuture> { + ) -> Result { let mut block_verifier_router = self.block_verifier_router.clone(); - async move { - let block: Block = match block_bytes.zcash_deserialize_into() { - Ok(block_bytes) => block_bytes, - Err(error) => { - tracing::info!(?error, "submit block failed: block bytes could not be deserialized into a structurally valid block"); + let block: Block = match block_bytes.zcash_deserialize_into() { + Ok(block_bytes) => block_bytes, + Err(error) => { + tracing::info!(?error, "submit block failed: block bytes could not be deserialized into a structurally valid block"); - return Ok(submit_block::ErrorResponse::Rejected.into()); - } - }; + return Ok(submit_block::ErrorResponse::Rejected.into()); + } + }; + + let block_height = block + .coinbase_height() + .map(|height| height.0.to_string()) + .unwrap_or_else(|| "invalid coinbase height".to_string()); + let block_hash = block.hash(); + + let block_verifier_router_response = block_verifier_router + .ready() + .await + .map_err(|error| ErrorObject::owned(0, error.to_string(), None::<()>))? + .call(zebra_consensus::Request::Commit(Arc::new(block))) + .await; + + let chain_error = match block_verifier_router_response { + // Currently, this match arm returns `null` (Accepted) for blocks committed + // to any chain, but Accepted is only for blocks in the best chain. + // + // TODO (#5487): + // - Inconclusive: check if the block is on a side-chain + // The difference is important to miners, because they want to mine on the best chain. + Ok(block_hash) => { + tracing::info!(?block_hash, ?block_height, "submit block accepted"); + return Ok(submit_block::Response::Accepted); + } - let block_height = block - .coinbase_height() - .map(|height| height.0.to_string()) - .unwrap_or_else(|| "invalid coinbase height".to_string()); - let block_hash = block.hash(); + // Turns BoxError into Result, + // by downcasting from Any to VerifyChainError. + Err(box_error) => { + let error = box_error + .downcast::() + .map(|boxed_chain_error| *boxed_chain_error); - let block_verifier_router_response = block_verifier_router - .ready() - .await - .map_err(|error| Error { - code: ErrorCode::ServerError(0), - message: error.to_string(), - data: None, - })? - .call(zebra_consensus::Request::Commit(Arc::new(block))) - .await; - - let chain_error = match block_verifier_router_response { - // Currently, this match arm returns `null` (Accepted) for blocks committed - // to any chain, but Accepted is only for blocks in the best chain. - // - // TODO (#5487): - // - Inconclusive: check if the block is on a side-chain - // The difference is important to miners, because they want to mine on the best chain. - Ok(block_hash) => { - tracing::info!(?block_hash, ?block_height, "submit block accepted"); - return Ok(submit_block::Response::Accepted); - } + tracing::info!( + ?error, + ?block_hash, + ?block_height, + "submit block failed verification" + ); - // Turns BoxError into Result, - // by downcasting from Any to VerifyChainError. - Err(box_error) => { - let error = box_error - .downcast::() - .map(|boxed_chain_error| *boxed_chain_error); + error + } + }; - tracing::info!(?error, ?block_hash, ?block_height, "submit block failed verification"); + let response = match chain_error { + Ok(source) if source.is_duplicate_request() => submit_block::ErrorResponse::Duplicate, - error - } - }; + // Currently, these match arms return Reject for the older duplicate in a queue, + // but queued duplicates should be DuplicateInconclusive. + // + // Optional TODO (#5487): + // - DuplicateInconclusive: turn these non-finalized state duplicate block errors + // into BlockError enum variants, and handle them as DuplicateInconclusive: + // - "block already sent to be committed to the state" + // - "replaced by newer request" + // - keep the older request in the queue, + // and return a duplicate error for the newer request immediately. + // This improves the speed of the RPC response. + // + // Checking the download queues and BlockVerifierRouter buffer for duplicates + // might require architectural changes to Zebra, so we should only do it + // if mining pools really need it. + Ok(_verify_chain_error) => submit_block::ErrorResponse::Rejected, - let response = match chain_error { - Ok(source) if source.is_duplicate_request() => { - submit_block::ErrorResponse::Duplicate - } + // This match arm is currently unreachable, but if future changes add extra error types, + // we want to turn them into `Rejected`. + Err(_unknown_error_type) => submit_block::ErrorResponse::Rejected, + }; - // Currently, these match arms return Reject for the older duplicate in a queue, - // but queued duplicates should be DuplicateInconclusive. - // - // Optional TODO (#5487): - // - DuplicateInconclusive: turn these non-finalized state duplicate block errors - // into BlockError enum variants, and handle them as DuplicateInconclusive: - // - "block already sent to be committed to the state" - // - "replaced by newer request" - // - keep the older request in the queue, - // and return a duplicate error for the newer request immediately. - // This improves the speed of the RPC response. - // - // Checking the download queues and BlockVerifierRouter buffer for duplicates - // might require architectural changes to Zebra, so we should only do it - // if mining pools really need it. - Ok(_verify_chain_error) => submit_block::ErrorResponse::Rejected, - - // This match arm is currently unreachable, but if future changes add extra error types, - // we want to turn them into `Rejected`. - Err(_unknown_error_type) => submit_block::ErrorResponse::Rejected, - }; - - Ok(response.into()) - } - .boxed() + Ok(response.into()) } - fn get_mining_info(&self) -> BoxFuture> { + async fn get_mining_info(&self) -> Result { let network = self.network.clone(); let mut state = self.state.clone(); @@ -1033,38 +1021,35 @@ where } let solution_rate_fut = self.get_network_sol_ps(None, None); - async move { - // Get the current block size. - let mut current_block_size = None; - if tip_height > 0 { - let request = zebra_state::ReadRequest::TipBlockSize; - let response: zebra_state::ReadResponse = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_server_error()?; - current_block_size = match response { - zebra_state::ReadResponse::TipBlockSize(Some(block_size)) => Some(block_size), - _ => None, - }; - } - - Ok(get_mining_info::Response::new( - tip_height, - current_block_size, - current_block_tx, - network, - solution_rate_fut.await?, - )) + // Get the current block size. + let mut current_block_size = None; + if tip_height > 0 { + let request = zebra_state::ReadRequest::TipBlockSize; + let response: zebra_state::ReadResponse = state + .ready() + .and_then(|service| service.call(request)) + .await + .map_error(server::error::LegacyCode::default())?; + current_block_size = match response { + zebra_state::ReadResponse::TipBlockSize(Some(block_size)) => Some(block_size), + _ => None, + }; } - .boxed() + + Ok(get_mining_info::Response::new( + tip_height, + current_block_size, + current_block_tx, + network, + solution_rate_fut.await?, + )) } - fn get_network_sol_ps( + async fn get_network_sol_ps( &self, num_blocks: Option, height: Option, - ) -> BoxFuture> { + ) -> Result { // Default number of blocks is 120 if not supplied. let mut num_blocks = num_blocks.unwrap_or(DEFAULT_SOLUTION_RATE_WINDOW_SIZE); // But if it is 0 or negative, it uses the proof of work averaging window. @@ -1080,341 +1065,296 @@ where let mut state = self.state.clone(); - async move { - let request = ReadRequest::SolutionRate { num_blocks, height }; + let request = ReadRequest::SolutionRate { num_blocks, height }; - let response = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_err(|error| Error { - code: ErrorCode::ServerError(0), - message: error.to_string(), - data: None, - })?; + let response = state + .ready() + .and_then(|service| service.call(request)) + .await + .map_err(|error| ErrorObject::owned(0, error.to_string(), None::<()>))?; - let solution_rate = match response { - // zcashd returns a 0 rate when the calculation is invalid - ReadResponse::SolutionRate(solution_rate) => solution_rate.unwrap_or(0), + let solution_rate = match response { + // zcashd returns a 0 rate when the calculation is invalid + ReadResponse::SolutionRate(solution_rate) => solution_rate.unwrap_or(0), - _ => unreachable!("unmatched response to a solution rate request"), - }; + _ => unreachable!("unmatched response to a solution rate request"), + }; - Ok(solution_rate - .try_into() - .expect("per-second solution rate always fits in u64")) - } - .boxed() + Ok(solution_rate + .try_into() + .expect("per-second solution rate always fits in u64")) } - fn get_peer_info(&self) -> BoxFuture>> { + async fn get_peer_info(&self) -> Result> { let address_book = self.address_book.clone(); - async move { - Ok(address_book - .recently_live_peers(chrono::Utc::now()) - .into_iter() - .map(PeerInfo::from) - .collect()) - } - .boxed() + Ok(address_book + .recently_live_peers(chrono::Utc::now()) + .into_iter() + .map(PeerInfo::from) + .collect()) } - fn validate_address( - &self, - raw_address: String, - ) -> BoxFuture> { + async fn validate_address(&self, raw_address: String) -> Result { let network = self.network.clone(); - async move { - let Ok(address) = raw_address - .parse::() else { - return Ok(validate_address::Response::invalid()); - }; - - let address = match address - .convert::() { - Ok(address) => address, - Err(err) => { - tracing::debug!(?err, "conversion error"); - return Ok(validate_address::Response::invalid()); - } - }; + let Ok(address) = raw_address.parse::() else { + return Ok(validate_address::Response::invalid()); + }; - // we want to match zcashd's behaviour - if !address.is_transparent() { + let address = match address.convert::() { + Ok(address) => address, + Err(err) => { + tracing::debug!(?err, "conversion error"); return Ok(validate_address::Response::invalid()); } + }; - if address.network() == network.kind() { - Ok(validate_address::Response { - address: Some(raw_address), - is_valid: true, - is_script: Some(address.is_script_hash()), - }) - } else { - tracing::info!( - ?network, - address_network = ?address.network(), - "invalid address in validateaddress RPC: Zebra's configured network must match address network" - ); + // we want to match zcashd's behaviour + if !address.is_transparent() { + return Ok(validate_address::Response::invalid()); + } - Ok(validate_address::Response::invalid()) - } + if address.network() == network.kind() { + Ok(validate_address::Response { + address: Some(raw_address), + is_valid: true, + is_script: Some(address.is_script_hash()), + }) + } else { + tracing::info!( + ?network, + address_network = ?address.network(), + "invalid address in validateaddress RPC: Zebra's configured network must match address network" + ); + + Ok(validate_address::Response::invalid()) } - .boxed() } - fn z_validate_address( + async fn z_validate_address( &self, raw_address: String, - ) -> BoxFuture> { + ) -> Result { let network = self.network.clone(); - async move { - let Ok(address) = raw_address - .parse::() else { - return Ok(z_validate_address::Response::invalid()); - }; - - let address = match address - .convert::() { - Ok(address) => address, - Err(err) => { - tracing::debug!(?err, "conversion error"); - return Ok(z_validate_address::Response::invalid()); - } - }; - - if address.network() == network.kind() { - Ok(z_validate_address::Response { - is_valid: true, - address: Some(raw_address), - address_type: Some(z_validate_address::AddressType::from(&address)), - is_mine: Some(false), - }) - } else { - tracing::info!( - ?network, - address_network = ?address.network(), - "invalid address network in z_validateaddress RPC: address is for {:?} but Zebra is on {:?}", - address.network(), - network - ); + let Ok(address) = raw_address.parse::() else { + return Ok(z_validate_address::Response::invalid()); + }; - Ok(z_validate_address::Response::invalid()) + let address = match address.convert::() { + Ok(address) => address, + Err(err) => { + tracing::debug!(?err, "conversion error"); + return Ok(z_validate_address::Response::invalid()); } + }; + + if address.network() == network.kind() { + Ok(z_validate_address::Response { + is_valid: true, + address: Some(raw_address), + address_type: Some(z_validate_address::AddressType::from(&address)), + is_mine: Some(false), + }) + } else { + tracing::info!( + ?network, + address_network = ?address.network(), + "invalid address network in z_validateaddress RPC: address is for {:?} but Zebra is on {:?}", + address.network(), + network + ); + + Ok(z_validate_address::Response::invalid()) } - .boxed() } - fn get_block_subsidy(&self, height: Option) -> BoxFuture> { + async fn get_block_subsidy(&self, height: Option) -> Result { let latest_chain_tip = self.latest_chain_tip.clone(); let network = self.network.clone(); - async move { - let height = if let Some(height) = height { - Height(height) - } else { - best_chain_tip_height(&latest_chain_tip)? - }; + let height = if let Some(height) = height { + Height(height) + } else { + best_chain_tip_height(&latest_chain_tip)? + }; + + if height < network.height_for_first_halving() { + return Err(ErrorObject::borrowed( + 0, + "Zebra does not support founders' reward subsidies, \ + use a block height that is after the first halving", + None, + )); + } - if height < network.height_for_first_halving() { - return Err(Error { - code: ErrorCode::ServerError(0), - message: "Zebra does not support founders' reward subsidies, \ - use a block height that is after the first halving" - .into(), - data: None, - }); - } + // Always zero for post-halving blocks + let founders = Amount::zero(); - // Always zero for post-halving blocks - let founders = Amount::zero(); + let total_block_subsidy = + block_subsidy(height, &network).map_error(server::error::LegacyCode::default())?; + let miner_subsidy = miner_subsidy(height, &network, total_block_subsidy) + .map_error(server::error::LegacyCode::default())?; - let total_block_subsidy = block_subsidy(height, &network).map_server_error()?; - let miner_subsidy = - miner_subsidy(height, &network, total_block_subsidy).map_server_error()?; + let (lockbox_streams, mut funding_streams): (Vec<_>, Vec<_>) = + funding_stream_values(height, &network, total_block_subsidy) + .map_error(server::error::LegacyCode::default())? + .into_iter() + // Separate the funding streams into deferred and non-deferred streams + .partition(|(receiver, _)| matches!(receiver, FundingStreamReceiver::Deferred)); + + let is_nu6 = NetworkUpgrade::current(&network, height) == NetworkUpgrade::Nu6; + + let [lockbox_total, funding_streams_total]: [std::result::Result< + Amount, + amount::Error, + >; 2] = [&lockbox_streams, &funding_streams] + .map(|streams| streams.iter().map(|&(_, amount)| amount).sum()); + + // Use the same funding stream order as zcashd + funding_streams.sort_by_key(|(receiver, _funding_stream)| { + ZCASHD_FUNDING_STREAM_ORDER + .iter() + .position(|zcashd_receiver| zcashd_receiver == receiver) + }); - let (lockbox_streams, mut funding_streams): (Vec<_>, Vec<_>) = - funding_stream_values(height, &network, total_block_subsidy) - .map_server_error()? + // Format the funding streams and lockbox streams + let [funding_streams, lockbox_streams]: [Vec<_>; 2] = [funding_streams, lockbox_streams] + .map(|streams| { + streams .into_iter() - // Separate the funding streams into deferred and non-deferred streams - .partition(|(receiver, _)| matches!(receiver, FundingStreamReceiver::Deferred)); - - let is_nu6 = NetworkUpgrade::current(&network, height) == NetworkUpgrade::Nu6; - - let [lockbox_total, funding_streams_total]: [std::result::Result< - Amount, - amount::Error, - >; 2] = [&lockbox_streams, &funding_streams] - .map(|streams| streams.iter().map(|&(_, amount)| amount).sum()); - - // Use the same funding stream order as zcashd - funding_streams.sort_by_key(|(receiver, _funding_stream)| { - ZCASHD_FUNDING_STREAM_ORDER - .iter() - .position(|zcashd_receiver| zcashd_receiver == receiver) + .map(|(receiver, value)| { + let address = funding_stream_address(height, &network, receiver); + FundingStream::new(is_nu6, receiver, value, address) + }) + .collect() }); - // Format the funding streams and lockbox streams - let [funding_streams, lockbox_streams]: [Vec<_>; 2] = - [funding_streams, lockbox_streams].map(|streams| { - streams - .into_iter() - .map(|(receiver, value)| { - let address = funding_stream_address(height, &network, receiver); - FundingStream::new(is_nu6, receiver, value, address) - }) - .collect() - }); - - Ok(BlockSubsidy { - miner: miner_subsidy.into(), - founders: founders.into(), - funding_streams, - lockbox_streams, - funding_streams_total: funding_streams_total.map_server_error()?.into(), - lockbox_total: lockbox_total.map_server_error()?.into(), - total_block_subsidy: total_block_subsidy.into(), - }) - } - .boxed() + Ok(BlockSubsidy { + miner: miner_subsidy.into(), + founders: founders.into(), + funding_streams, + lockbox_streams, + funding_streams_total: funding_streams_total + .map_error(server::error::LegacyCode::default())? + .into(), + lockbox_total: lockbox_total + .map_error(server::error::LegacyCode::default())? + .into(), + total_block_subsidy: total_block_subsidy.into(), + }) } - fn get_difficulty(&self) -> BoxFuture> { + async fn get_difficulty(&self) -> Result { let network = self.network.clone(); let mut state = self.state.clone(); - async move { - let request = ReadRequest::ChainInfo; - - // # TODO - // - add a separate request like BestChainNextMedianTimePast, but skipping the - // consistency check, because any block's difficulty is ok for display - // - return 1.0 for a "not enough blocks in the state" error, like `zcashd`: - // - let response = state - .ready() - .and_then(|service| service.call(request)) - .await - .map_err(|error| Error { - code: ErrorCode::ServerError(0), - message: error.to_string(), - data: None, - })?; - - let chain_info = match response { - ReadResponse::ChainInfo(info) => info, - _ => unreachable!("unmatched response to a chain info request"), - }; - - // This RPC is typically used for display purposes, so it is not consensus-critical. - // But it uses the difficulty consensus rules for its calculations. - // - // Consensus: - // https://zips.z.cash/protocol/protocol.pdf#nbits - // - // The zcashd implementation performs to_expanded() on f64, - // and then does an inverse division: - // https://github.com/zcash/zcash/blob/d6e2fada844373a8554ee085418e68de4b593a6c/src/rpc/blockchain.cpp#L46-L73 - // - // But in Zebra we divide the high 128 bits of each expanded difficulty. This gives - // a similar result, because the lower 128 bits are insignificant after conversion - // to `f64` with a 53-bit mantissa. - // - // `pow_limit >> 128 / difficulty >> 128` is the same as the work calculation - // `(2^256 / pow_limit) / (2^256 / difficulty)`, but it's a bit more accurate. - // - // To simplify the calculation, we don't scale for leading zeroes. (Bitcoin's - // difficulty currently uses 68 bits, so even it would still have full precision - // using this calculation.) - - // Get expanded difficulties (256 bits), these are the inverse of the work - let pow_limit: U256 = network.target_difficulty_limit().into(); - let difficulty: U256 = chain_info - .expected_difficulty - .to_expanded() - .expect("valid blocks have valid difficulties") - .into(); - - // Shift out the lower 128 bits (256 bits, but the top 128 are all zeroes) - let pow_limit = pow_limit >> 128; - let difficulty = difficulty >> 128; - - // Convert to u128 then f64. - // We could also convert U256 to String, then parse as f64, but that's slower. - let pow_limit = pow_limit.as_u128() as f64; - let difficulty = difficulty.as_u128() as f64; - - // Invert the division to give approximately: `work(difficulty) / work(pow_limit)` - Ok(pow_limit / difficulty) - } - .boxed() + let request = ReadRequest::ChainInfo; + + // # TODO + // - add a separate request like BestChainNextMedianTimePast, but skipping the + // consistency check, because any block's difficulty is ok for display + // - return 1.0 for a "not enough blocks in the state" error, like `zcashd`: + // + let response = state + .ready() + .and_then(|service| service.call(request)) + .await + .map_err(|error| ErrorObject::owned(0, error.to_string(), None::<()>))?; + + let chain_info = match response { + ReadResponse::ChainInfo(info) => info, + _ => unreachable!("unmatched response to a chain info request"), + }; + + // This RPC is typically used for display purposes, so it is not consensus-critical. + // But it uses the difficulty consensus rules for its calculations. + // + // Consensus: + // https://zips.z.cash/protocol/protocol.pdf#nbits + // + // The zcashd implementation performs to_expanded() on f64, + // and then does an inverse division: + // https://github.com/zcash/zcash/blob/d6e2fada844373a8554ee085418e68de4b593a6c/src/rpc/blockchain.cpp#L46-L73 + // + // But in Zebra we divide the high 128 bits of each expanded difficulty. This gives + // a similar result, because the lower 128 bits are insignificant after conversion + // to `f64` with a 53-bit mantissa. + // + // `pow_limit >> 128 / difficulty >> 128` is the same as the work calculation + // `(2^256 / pow_limit) / (2^256 / difficulty)`, but it's a bit more accurate. + // + // To simplify the calculation, we don't scale for leading zeroes. (Bitcoin's + // difficulty currently uses 68 bits, so even it would still have full precision + // using this calculation.) + + // Get expanded difficulties (256 bits), these are the inverse of the work + let pow_limit: U256 = network.target_difficulty_limit().into(); + let difficulty: U256 = chain_info + .expected_difficulty + .to_expanded() + .expect("valid blocks have valid difficulties") + .into(); + + // Shift out the lower 128 bits (256 bits, but the top 128 are all zeroes) + let pow_limit = pow_limit >> 128; + let difficulty = difficulty >> 128; + + // Convert to u128 then f64. + // We could also convert U256 to String, then parse as f64, but that's slower. + let pow_limit = pow_limit.as_u128() as f64; + let difficulty = difficulty.as_u128() as f64; + + // Invert the division to give approximately: `work(difficulty) / work(pow_limit)` + Ok(pow_limit / difficulty) } - fn z_list_unified_receivers( - &self, - address: String, - ) -> BoxFuture> { + async fn z_list_unified_receivers(&self, address: String) -> Result { use zcash_address::unified::Container; - async move { - let (network, unified_address): ( - zcash_address::Network, - zcash_address::unified::Address, - ) = zcash_address::unified::Encoding::decode(address.clone().as_str()).map_err( - |error| Error { - code: ErrorCode::ServerError(0), - message: error.to_string(), - data: None, - }, - )?; + let (network, unified_address): (zcash_address::Network, zcash_address::unified::Address) = + zcash_address::unified::Encoding::decode(address.clone().as_str()) + .map_err(|error| ErrorObject::owned(0, error.to_string(), None::<()>))?; - let mut p2pkh = String::new(); - let mut p2sh = String::new(); - let mut orchard = String::new(); - let mut sapling = String::new(); + let mut p2pkh = String::new(); + let mut p2sh = String::new(); + let mut orchard = String::new(); + let mut sapling = String::new(); - for item in unified_address.items() { - match item { - zcash_address::unified::Receiver::Orchard(_data) => { - let addr = zcash_address::unified::Address::try_from_items(vec![item]) - .expect("using data already decoded as valid"); - orchard = addr.encode(&network); - } - zcash_address::unified::Receiver::Sapling(data) => { - let addr = - zebra_chain::primitives::Address::try_from_sapling(network, data) - .expect("using data already decoded as valid"); - sapling = addr.payment_address().unwrap_or_default(); - } - zcash_address::unified::Receiver::P2pkh(data) => { - let addr = zebra_chain::primitives::Address::try_from_transparent_p2pkh( - network, data, - ) + for item in unified_address.items() { + match item { + zcash_address::unified::Receiver::Orchard(_data) => { + let addr = zcash_address::unified::Address::try_from_items(vec![item]) .expect("using data already decoded as valid"); - p2pkh = addr.payment_address().unwrap_or_default(); - } - zcash_address::unified::Receiver::P2sh(data) => { - let addr = zebra_chain::primitives::Address::try_from_transparent_p2sh( - network, data, - ) + orchard = addr.encode(&network); + } + zcash_address::unified::Receiver::Sapling(data) => { + let addr = zebra_chain::primitives::Address::try_from_sapling(network, data) .expect("using data already decoded as valid"); - p2sh = addr.payment_address().unwrap_or_default(); - } - _ => (), + sapling = addr.payment_address().unwrap_or_default(); + } + zcash_address::unified::Receiver::P2pkh(data) => { + let addr = + zebra_chain::primitives::Address::try_from_transparent_p2pkh(network, data) + .expect("using data already decoded as valid"); + p2pkh = addr.payment_address().unwrap_or_default(); + } + zcash_address::unified::Receiver::P2sh(data) => { + let addr = + zebra_chain::primitives::Address::try_from_transparent_p2sh(network, data) + .expect("using data already decoded as valid"); + p2sh = addr.payment_address().unwrap_or_default(); } + _ => (), } - - Ok(unified_address::Response::new( - orchard, sapling, p2pkh, p2sh, - )) } - .boxed() + + Ok(unified_address::Response::new( + orchard, sapling, p2pkh, p2sh, + )) } - fn generate(&self, num_blocks: u32) -> BoxFuture>> { + async fn generate(&self, num_blocks: u32) -> Result> { let rpc: GetBlockTemplateRpcImpl< Mempool, State, @@ -1425,48 +1365,50 @@ where > = self.clone(); let network = self.network.clone(); - async move { - if !network.is_regtest() { - return Err(Error { - code: ErrorCode::ServerError(0), - message: "generate is only supported on regtest".to_string(), - data: None, - }); - } + if !network.is_regtest() { + return Err(ErrorObject::borrowed( + 0, + "generate is only supported on regtest", + None, + )); + } - let mut block_hashes = Vec::new(); - for _ in 0..num_blocks { - let block_template = rpc.get_block_template(None).await.map_server_error()?; - - let get_block_template::Response::TemplateMode(block_template) = block_template - else { - return Err(Error { - code: ErrorCode::ServerError(0), - message: "error generating block template".to_string(), - data: None, - }); - }; - - let proposal_block = proposal_block_from_template( - &block_template, - TimeSource::CurTime, - NetworkUpgrade::current(&network, Height(block_template.height)), - ) - .map_server_error()?; - let hex_proposal_block = - HexData(proposal_block.zcash_serialize_to_vec().map_server_error()?); - - let _submit = rpc - .submit_block(hex_proposal_block, None) - .await - .map_server_error()?; - - block_hashes.push(GetBlockHash(proposal_block.hash())); - } + let mut block_hashes = Vec::new(); + for _ in 0..num_blocks { + let block_template = rpc + .get_block_template(None) + .await + .map_error(server::error::LegacyCode::default())?; + + let get_block_template::Response::TemplateMode(block_template) = block_template else { + return Err(ErrorObject::borrowed( + 0, + "error generating block template", + None, + )); + }; + + let proposal_block = proposal_block_from_template( + &block_template, + TimeSource::CurTime, + NetworkUpgrade::current(&network, Height(block_template.height)), + ) + .map_error(server::error::LegacyCode::default())?; + let hex_proposal_block = HexData( + proposal_block + .zcash_serialize_to_vec() + .map_error(server::error::LegacyCode::default())?, + ); - Ok(block_hashes) + let _submit = rpc + .submit_block(hex_proposal_block, None) + .await + .map_error(server::error::LegacyCode::default())?; + + block_hashes.push(GetBlockHash(proposal_block.hash())); } - .boxed() + + Ok(block_hashes) } } diff --git a/zebra-rpc/src/methods/get_block_template_rpcs/constants.rs b/zebra-rpc/src/methods/get_block_template_rpcs/constants.rs index 3fd4696980d..950dff5db5d 100644 --- a/zebra-rpc/src/methods/get_block_template_rpcs/constants.rs +++ b/zebra-rpc/src/methods/get_block_template_rpcs/constants.rs @@ -1,6 +1,6 @@ //! Constant values used in mining rpcs methods. -use jsonrpc_core::ErrorCode; +use jsonrpsee_types::ErrorCode; use zebra_chain::{ block, diff --git a/zebra-rpc/src/methods/get_block_template_rpcs/get_block_template.rs b/zebra-rpc/src/methods/get_block_template_rpcs/get_block_template.rs index 7ab1a48e20a..baa0200db1f 100644 --- a/zebra-rpc/src/methods/get_block_template_rpcs/get_block_template.rs +++ b/zebra-rpc/src/methods/get_block_template_rpcs/get_block_template.rs @@ -2,7 +2,8 @@ use std::{collections::HashMap, iter, sync::Arc}; -use jsonrpc_core::{Error, ErrorCode, Result}; +use jsonrpsee::core::RpcResult as Result; +use jsonrpsee_types::{ErrorCode, ErrorObject}; use tower::{Service, ServiceExt}; use zebra_chain::{ @@ -25,12 +26,12 @@ use zebra_consensus::{ use zebra_node_services::mempool::{self, TransactionDependencies}; use zebra_state::GetBlockTemplateChainInfo; -use crate::methods::{ - errors::OkOrServerError, - get_block_template_rpcs::{ +use crate::{ + methods::get_block_template_rpcs::{ constants::{MAX_ESTIMATED_DISTANCE_TO_NETWORK_CHAIN_TIP, NOT_SYNCED_ERROR_CODE}, types::{default_roots::DefaultRoots, transaction::TransactionTemplate}, }, + server::error::OkOrError, }; pub use crate::methods::get_block_template_rpcs::types::get_block_template::*; @@ -61,25 +62,23 @@ pub fn check_parameters(parameters: &Option) -> Result<()> { mode: GetBlockTemplateRequestMode::Proposal, data: None, .. - } => Err(Error { - code: ErrorCode::InvalidParams, - message: "\"data\" parameter must be \ - provided in \"proposal\" mode" - .to_string(), - data: None, - }), + } => Err(ErrorObject::borrowed( + ErrorCode::InvalidParams.code(), + "\"data\" parameter must be \ + provided in \"proposal\" mode", + None, + )), JsonParameters { mode: GetBlockTemplateRequestMode::Template, data: Some(_), .. - } => Err(Error { - code: ErrorCode::InvalidParams, - message: "\"data\" parameter must be \ - omitted in \"template\" mode" - .to_string(), - data: None, - }), + } => Err(ErrorObject::borrowed( + ErrorCode::InvalidParams.code(), + "\"data\" parameter must be \ + omitted in \"template\" mode", + None, + )), } } @@ -87,13 +86,9 @@ pub fn check_parameters(parameters: &Option) -> Result<()> { pub fn check_miner_address( miner_address: Option, ) -> Result { - miner_address.ok_or_else(|| Error { - code: ErrorCode::ServerError(0), - message: "configure mining.miner_address in zebrad.toml \ - with a transparent address" - .to_string(), - data: None, - }) + miner_address.ok_or_misc_error( + "set `mining.miner_address` in `zebrad.toml` to a transparent address".to_string(), + ) } /// Attempts to validate block proposal against all of the server's @@ -135,11 +130,7 @@ where let block_verifier_router_response = block_verifier_router .ready() .await - .map_err(|error| Error { - code: ErrorCode::ServerError(0), - message: error.to_string(), - data: None, - })? + .map_err(|error| ErrorObject::owned(0, error.to_string(), None::<()>))? .call(zebra_consensus::Request::CheckProposal(Arc::new(block))) .await; @@ -181,7 +172,7 @@ where // but this is ok for an estimate let (estimated_distance_to_chain_tip, local_tip_height) = latest_chain_tip .estimate_distance_to_network_chain_tip(network) - .ok_or_server_error("no chain tip available yet")?; + .ok_or_misc_error("no chain tip available yet")?; if !sync_status.is_close_to_tip() || estimated_distance_to_chain_tip > MAX_ESTIMATED_DISTANCE_TO_NETWORK_CHAIN_TIP @@ -193,16 +184,14 @@ where Hint: check your network connection, clock, and time zone settings." ); - return Err(Error { - code: NOT_SYNCED_ERROR_CODE, - message: format!( - "Zebra has not synced to the chain tip, \ + return Err(ErrorObject::borrowed( + NOT_SYNCED_ERROR_CODE.code(), + "Zebra has not synced to the chain tip, \ estimated distance: {estimated_distance_to_chain_tip:?}, \ local tip: {local_tip_height:?}. \ - Hint: check your network connection, clock, and time zone settings." - ), - data: None, - }); + Hint: check your network connection, clock, and time zone settings.", + None, + )); } Ok(()) @@ -231,11 +220,7 @@ where let response = state .oneshot(request.clone()) .await - .map_err(|error| Error { - code: ErrorCode::ServerError(0), - message: error.to_string(), - data: None, - })?; + .map_err(|error| ErrorObject::owned(0, error.to_string(), None::<()>))?; let chain_info = match response { zebra_state::ReadResponse::ChainInfo(chain_info) => chain_info, @@ -265,11 +250,7 @@ where let response = mempool .oneshot(mempool::Request::FullTransactions) .await - .map_err(|error| Error { - code: ErrorCode::ServerError(0), - message: error.to_string(), - data: None, - })?; + .map_err(|error| ErrorObject::owned(0, error.to_string(), None::<()>))?; // TODO: Order transactions in block templates based on their dependencies diff --git a/zebra-rpc/src/methods/get_block_template_rpcs/types/get_mining_info.rs b/zebra-rpc/src/methods/get_block_template_rpcs/types/get_mining_info.rs index 21627d509db..1caa1593c27 100644 --- a/zebra-rpc/src/methods/get_block_template_rpcs/types/get_mining_info.rs +++ b/zebra-rpc/src/methods/get_block_template_rpcs/types/get_mining_info.rs @@ -3,7 +3,7 @@ use zebra_chain::parameters::Network; /// Response to a `getmininginfo` RPC request. -#[derive(Debug, Default, PartialEq, Eq, serde::Serialize)] +#[derive(Debug, Default, Clone, PartialEq, Eq, serde::Serialize)] pub struct Response { /// The current tip height. #[serde(rename = "blocks")] diff --git a/zebra-rpc/src/methods/get_block_template_rpcs/types/submit_block.rs b/zebra-rpc/src/methods/get_block_template_rpcs/types/submit_block.rs index 2513af85aa6..cec806901bb 100644 --- a/zebra-rpc/src/methods/get_block_template_rpcs/types/submit_block.rs +++ b/zebra-rpc/src/methods/get_block_template_rpcs/types/submit_block.rs @@ -2,11 +2,11 @@ // Allow doc links to these imports. #[allow(unused_imports)] -use crate::methods::get_block_template_rpcs::GetBlockTemplateRpc; +use crate::methods::get_block_template_rpcs::GetBlockTemplate; /// Optional argument `jsonparametersobject` for `submitblock` RPC request /// -/// See notes for [`GetBlockTemplateRpc::submit_block`] method +/// See notes for [`crate::methods::GetBlockTemplateRpcServer::submit_block`] method #[derive(Clone, Debug, PartialEq, Eq, serde::Deserialize)] pub struct JsonParameters { /// The workid for the block template. Currently unused. @@ -28,7 +28,7 @@ pub struct JsonParameters { /// Response to a `submitblock` RPC request. /// /// Zebra never returns "duplicate-invalid", because it does not store invalid blocks. -#[derive(Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize)] +#[derive(Clone, Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize)] #[serde(rename_all = "kebab-case")] pub enum ErrorResponse { /// Block was already committed to the non-finalized or finalized state @@ -44,7 +44,7 @@ pub enum ErrorResponse { /// Response to a `submitblock` RPC request. /// /// Zebra never returns "duplicate-invalid", because it does not store invalid blocks. -#[derive(Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize)] +#[derive(Clone, Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize)] #[serde(untagged)] pub enum Response { /// Block was not successfully submitted, return error diff --git a/zebra-rpc/src/methods/tests/prop.rs b/zebra-rpc/src/methods/tests/prop.rs index 9435a68ac7e..8753d514c23 100644 --- a/zebra-rpc/src/methods/tests/prop.rs +++ b/zebra-rpc/src/methods/tests/prop.rs @@ -1,10 +1,10 @@ //! Randomised property tests for RPC methods. -use std::{collections::HashSet, sync::Arc}; +use std::{collections::HashSet, fmt::Debug, sync::Arc}; use futures::{join, FutureExt, TryFutureExt}; -use hex::ToHex; -use jsonrpc_core::{Error, ErrorCode}; +use hex::{FromHex, ToHex}; +use jsonrpsee_types::{ErrorCode, ErrorObject}; use proptest::{collection::vec, prelude::*}; use thiserror::Error; use tokio::sync::oneshot; @@ -13,11 +13,8 @@ use tower::buffer::Buffer; use zebra_chain::{ amount::{Amount, NonNegative}, block::{self, Block, Height}, - chain_tip::{mock::MockChainTip, NoChainTip}, - parameters::{ - Network::{self, *}, - NetworkUpgrade, - }, + chain_tip::{mock::MockChainTip, ChainTip, NoChainTip}, + parameters::{Network, NetworkUpgrade}, serialization::{ZcashDeserialize, ZcashSerialize}, transaction::{self, Transaction, UnminedTx, VerifiedUnminedTx}, transparent, @@ -28,37 +25,31 @@ use zebra_state::BoxError; use zebra_test::mock_service::MockService; +use crate::methods; + use super::super::{ - AddressBalance, AddressStrings, NetworkUpgradeStatus, Rpc, RpcImpl, SentTransactionHash, + AddressBalance, AddressStrings, NetworkUpgradeStatus, RpcImpl, RpcServer, SentTransactionHash, }; proptest! { /// Test that when sending a raw transaction, it is received by the mempool service. #[test] - fn mempool_receives_raw_transaction(transaction in any::()) { + fn mempool_receives_raw_tx(transaction in any::(), network in any::()) { let (runtime, _init_guard) = zebra_test::init_async(); + let _guard = runtime.enter(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, NoChainTip); + + // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. + tokio::time::pause(); runtime.block_on(async move { - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - Mainnet, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); let hash = SentTransactionHash(transaction.hash()); - let transaction_bytes = transaction - .zcash_serialize_to_vec() - .expect("Transaction serializes successfully"); + let transaction_bytes = transaction.zcash_serialize_to_vec()?; + let transaction_hex = hex::encode(&transaction_bytes); - let send_task = tokio::spawn(rpc.send_raw_transaction(transaction_hex)); + let send_task = tokio::spawn(async move { rpc.send_raw_transaction(transaction_hex).await }); let unmined_transaction = UnminedTx::from(transaction); let expected_request = mempool::Request::Queue(vec![unmined_transaction.into()]); @@ -73,17 +64,14 @@ proptest! { state.expect_no_requests().await?; - let result = send_task - .await - .expect("Sending raw transactions should not panic"); + let result = send_task.await.expect("send_raw_transaction should not panic"); prop_assert_eq!(result, Ok(hash)); // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); + prop_assert!(mempool_tx_queue.now_or_never().is_none()); - Ok::<_, TestCaseError>(()) + Ok(()) })?; } @@ -91,30 +79,21 @@ proptest! { /// /// Mempool service errors should become server errors. #[test] - fn mempool_errors_are_forwarded(transaction in any::()) { + fn mempool_errors_are_forwarded(transaction in any::(), network in any::()) { let (runtime, _init_guard) = zebra_test::init_async(); + let _guard = runtime.enter(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, NoChainTip); - runtime.block_on(async move { - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - Mainnet, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); + // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. + tokio::time::pause(); - let transaction_bytes = transaction - .zcash_serialize_to_vec() - .expect("Transaction serializes successfully"); + runtime.block_on(async move { + let transaction_bytes = transaction.zcash_serialize_to_vec()?; let transaction_hex = hex::encode(&transaction_bytes); - let send_task = tokio::spawn(rpc.send_raw_transaction(transaction_hex.clone())); + let _rpc = rpc.clone(); + let _transaction_hex = transaction_hex.clone(); + let send_task = tokio::spawn(async move { _rpc.send_raw_transaction(_transaction_hex).await }); let unmined_transaction = UnminedTx::from(transaction); let expected_request = mempool::Request::Queue(vec![unmined_transaction.clone().into()]); @@ -126,22 +105,11 @@ proptest! { state.expect_no_requests().await?; - let result = send_task - .await - .expect("Sending raw transactions should not panic"); + let result = send_task.await.expect("send_raw_transaction should not panic"); - prop_assert!( - matches!( - result, - Err(Error { - code: ErrorCode::ServerError(_), - .. - }) - ), - "Result is not a server error: {result:?}" - ); + check_err_code(result, ErrorCode::ServerError(-1))?; - let send_task = tokio::spawn(rpc.send_raw_transaction(transaction_hex)); + let send_task = tokio::spawn(async move { rpc.send_raw_transaction(transaction_hex.clone()).await }); let expected_request = mempool::Request::Queue(vec![unmined_transaction.clone().into()]); @@ -152,87 +120,44 @@ proptest! { .await? .respond(Ok::<_, BoxError>(mempool::Response::Queued(vec![Ok(rsp_rx)]))); - let result = send_task - .await - .expect("Sending raw transactions should not panic"); + let result = send_task.await.expect("send_raw_transaction should not panic"); - prop_assert!( - matches!( - result, - Err(Error { - code: ErrorCode::ServerError(_), - .. - }) - ), - "Result is not a server error: {result:?}" - ); + check_err_code(result, ErrorCode::ServerError(-25))?; // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); + prop_assert!(mempool_tx_queue.now_or_never().is_none()); - Ok::<_, TestCaseError>(()) + Ok(()) })?; } /// Test that when the mempool rejects a transaction the caller receives an error. #[test] - fn rejected_transactions_are_reported(transaction in any::()) { + fn rejected_txs_are_reported(transaction in any::(), network in any::()) { let (runtime, _init_guard) = zebra_test::init_async(); + let _guard = runtime.enter(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, NoChainTip); - runtime.block_on(async move { - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - Mainnet, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); - - let transaction_bytes = transaction - .zcash_serialize_to_vec() - .expect("Transaction serializes successfully"); - let transaction_hex = hex::encode(&transaction_bytes); + // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. + tokio::time::pause(); - let send_task = tokio::spawn(rpc.send_raw_transaction(transaction_hex)); + runtime.block_on(async move { + let tx = hex::encode(&transaction.zcash_serialize_to_vec()?); + let req = mempool::Request::Queue(vec![UnminedTx::from(transaction).into()]); + let rsp = mempool::Response::Queued(vec![Err(DummyError.into())]); + let mempool_query = mempool.expect_request(req).map_ok(|r| r.respond(rsp)); - let unmined_transaction = UnminedTx::from(transaction); - let expected_request = mempool::Request::Queue(vec![unmined_transaction.into()]); - let response = mempool::Response::Queued(vec![Err(DummyError.into())]); + let (rpc_rsp, _) = tokio::join!(rpc.send_raw_transaction(tx), mempool_query); - mempool - .expect_request(expected_request) - .await? - .respond(response); + check_err_code(rpc_rsp, ErrorCode::ServerError(-1))?; + // Check that no state request was made. state.expect_no_requests().await?; - let result = send_task - .await - .expect("Sending raw transactions should not panic"); - - prop_assert!( - matches!( - result, - Err(Error { - code: ErrorCode::ServerError(_), - .. - }) - ), - "Result is not a server error: {result:?}" - ); - // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); + prop_assert!(mempool_tx_queue.now_or_never().is_none()); - Ok::<_, TestCaseError>(()) + Ok(()) })?; } @@ -241,53 +166,27 @@ proptest! { /// Try to call `send_raw_transaction` using a string parameter that has at least one /// non-hexadecimal character, and check that it fails with an expected error. #[test] - fn non_hexadecimal_string_results_in_an_error(non_hex_string in ".*[^0-9A-Fa-f].*") { + fn non_hex_string_is_error(non_hex_string in ".*[^0-9A-Fa-f].*", network in any::()) { let (runtime, _init_guard) = zebra_test::init_async(); let _guard = runtime.enter(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, NoChainTip); // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. tokio::time::pause(); runtime.block_on(async move { - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - Mainnet, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); - - let send_task = tokio::spawn(rpc.send_raw_transaction(non_hex_string)); + let send_task = rpc.send_raw_transaction(non_hex_string); + // Check that there are no further requests. mempool.expect_no_requests().await?; state.expect_no_requests().await?; - let result = send_task - .await - .expect("Sending raw transactions should not panic"); - - prop_assert!( - matches!( - result, - Err(Error { - code: ErrorCode::InvalidParams, - .. - }) - ), - "Result is not an invalid parameters error: {result:?}" - ); + check_err_code(send_task.await, ErrorCode::ServerError(-22))?; // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); + prop_assert!(mempool_tx_queue.now_or_never().is_none()); - Ok::<_, TestCaseError>(()) + Ok(()) })?; } @@ -296,9 +195,10 @@ proptest! { /// Try to call `send_raw_transaction` using random bytes that fail to deserialize as a /// transaction, and check that it fails with an expected error. #[test] - fn invalid_transaction_results_in_an_error(random_bytes in any::>()) { + fn invalid_tx_results_in_an_error(random_bytes in any::>(), network in any::()) { let (runtime, _init_guard) = zebra_test::init_async(); let _guard = runtime.enter(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, NoChainTip); // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. tokio::time::pause(); @@ -306,45 +206,17 @@ proptest! { prop_assume!(Transaction::zcash_deserialize(&*random_bytes).is_err()); runtime.block_on(async move { - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - Mainnet, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); - - let send_task = tokio::spawn(rpc.send_raw_transaction(hex::encode(random_bytes))); + let send_task = rpc.send_raw_transaction(hex::encode(random_bytes)); mempool.expect_no_requests().await?; state.expect_no_requests().await?; - let result = send_task - .await - .expect("Sending raw transactions should not panic"); - - prop_assert!( - matches!( - result, - Err(Error { - code: ErrorCode::InvalidParams, - .. - }) - ), - "Result is not an invalid parameters error: {result:?}" - ); + check_err_code(send_task.await, ErrorCode::ServerError(-22))?; // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); + prop_assert!(mempool_tx_queue.now_or_never().is_none()); - Ok::<_, TestCaseError>(()) + Ok(()) })?; } @@ -353,33 +225,18 @@ proptest! { /// Make the mock mempool service return a list of transaction IDs, and check that the RPC call /// returns those IDs as hexadecimal strings. #[test] - fn mempool_transactions_are_sent_to_caller(transactions in any::>()) { + fn mempool_transactions_are_sent_to_caller(transactions in any::>(), + network in any::()) { let (runtime, _init_guard) = zebra_test::init_async(); let _guard = runtime.enter(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, NoChainTip); // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. tokio::time::pause(); runtime.block_on(async move { - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - Mainnet, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); - - let call_task = tokio::spawn(rpc.get_raw_mempool()); - - #[cfg(not(feature = "getblocktemplate-rpcs"))] - let expected_response = { + let (expected_response, mempool_query) = { let transaction_ids: HashSet<_> = transactions .iter() .map(|tx| tx.transaction.id) @@ -391,18 +248,18 @@ proptest! { .collect(); expected_response.sort(); - mempool + let mempool_query = mempool .expect_request(mempool::Request::TransactionIds) - .await? - .respond(mempool::Response::TransactionIds(transaction_ids)); + .map_ok(|r|r.respond(mempool::Response::TransactionIds(transaction_ids))); - expected_response + (expected_response, mempool_query) }; // Note: this depends on `SHOULD_USE_ZCASHD_ORDER` being true. #[cfg(feature = "getblocktemplate-rpcs")] - let expected_response = { + let (expected_response, mempool_query) = { let mut expected_response = transactions.clone(); + expected_response.sort_by_cached_key(|tx| { // zcashd uses modified fee here but Zebra doesn't currently // support prioritizing transactions @@ -416,151 +273,80 @@ proptest! { let expected_response = expected_response .iter() - .map(|tx| tx.transaction.id.mined_id().encode_hex()) - .collect(); + .map(|tx| tx.transaction.id.mined_id().encode_hex::()) + .collect::>(); - mempool + let mempool_query = mempool .expect_request(mempool::Request::FullTransactions) - .await? - .respond(mempool::Response::FullTransactions { + .map_ok(|r| r.respond(mempool::Response::FullTransactions { transactions, transaction_dependencies: Default::default(), last_seen_tip_hash: [0; 32].into(), - }); + })); - expected_response + (expected_response, mempool_query) }; - mempool.expect_no_requests().await?; - state.expect_no_requests().await?; + let (rpc_rsp, _) = tokio::join!(rpc.get_raw_mempool(), mempool_query); - let result = call_task - .await - .expect("Sending raw transactions should not panic"); + prop_assert_eq!(rpc_rsp?, expected_response); - prop_assert_eq!(result, Ok(expected_response)); + mempool.expect_no_requests().await?; + state.expect_no_requests().await?; // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); + prop_assert!(mempool_tx_queue.now_or_never().is_none()); - Ok::<_, TestCaseError>(()) + Ok(()) })?; } - /// Test that the method rejects non-hexadecimal characters. + /// Calls `get_raw_transaction` with: /// - /// Try to call `get_raw_transaction` using a string parameter that has at least one - /// non-hexadecimal character, and check that it fails with an expected error. + /// 1. an invalid TXID that won't deserialize; + /// 2. a valid TXID that is not in the mempool nor in the state; + /// + /// and checks that the RPC returns the right error code. #[test] - fn get_raw_transaction_non_hexadecimal_string_results_in_an_error( - non_hex_string in ".*[^0-9A-Fa-f].*", - ) { + fn check_err_for_get_raw_transaction(unknown_txid: transaction::Hash, + invalid_txid in invalid_txid(), + network: Network) { let (runtime, _init_guard) = zebra_test::init_async(); let _guard = runtime.enter(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, NoChainTip); // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. tokio::time::pause(); runtime.block_on(async move { - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - Mainnet, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); + // Check the invalid TXID first. + let rpc_rsp = rpc.get_raw_transaction(invalid_txid, Some(1)).await; - let send_task = tokio::spawn(rpc.get_raw_transaction(non_hex_string, Some(0))); + check_err_code(rpc_rsp, ErrorCode::ServerError(-5))?; + // Check that no further requests were made. mempool.expect_no_requests().await?; state.expect_no_requests().await?; - let result = send_task - .await - .expect("Sending raw transactions should not panic"); - - prop_assert!( - matches!( - result, - Err(Error { - code: ErrorCode::InvalidParams, - .. - }) - ), - "Result is not an invalid parameters error: {result:?}" - ); - - // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); - - Ok::<_, TestCaseError>(()) - })?; - } - - /// Test that the method rejects an input that's not a transaction. - /// - /// Try to call `get_raw_transaction` using random bytes that fail to deserialize as a - /// transaction, and check that it fails with an expected error. - #[test] - fn get_raw_transaction_invalid_transaction_results_in_an_error( - random_bytes in any::>(), - ) { - let (runtime, _init_guard) = zebra_test::init_async(); - let _guard = runtime.enter(); - - // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. - tokio::time::pause(); - - prop_assume!(transaction::Hash::zcash_deserialize(&*random_bytes).is_err()); + // Now check the unknown TXID. + let mempool_query = mempool + .expect_request(mempool::Request::TransactionsByMinedId([unknown_txid].into())) + .map_ok(|r| r.respond(mempool::Response::Transactions(vec![]))); - runtime.block_on(async move { - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - Mainnet, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); + let state_query = state + .expect_request(zebra_state::ReadRequest::Transaction(unknown_txid)) + .map_ok(|r| r.respond(zebra_state::ReadResponse::Transaction(None))); - let send_task = tokio::spawn(rpc.get_raw_transaction(hex::encode(random_bytes), Some(0))); + let rpc_query = rpc.get_raw_transaction(unknown_txid.encode_hex(), Some(1)); - mempool.expect_no_requests().await?; - state.expect_no_requests().await?; + let (rpc_rsp, _, _) = tokio::join!(rpc_query, mempool_query, state_query); - let result = send_task - .await - .expect("Sending raw transactions should not panic"); - - prop_assert!( - matches!( - result, - Err(Error { - code: ErrorCode::InvalidParams, - .. - }) - ), - "Result is not an invalid parameters error: {result:?}" - ); + check_err_code(rpc_rsp, ErrorCode::ServerError(-5))?; // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); + prop_assert!(mempool_tx_queue.now_or_never().is_none()); - Ok::<_, TestCaseError>(()) + Ok(()) })?; } @@ -569,21 +355,10 @@ proptest! { fn get_blockchain_info_response_without_a_chain_tip(network in any::()) { let (runtime, _init_guard) = zebra_test::init_async(); let _guard = runtime.enter(); - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - // look for an error with a `NoChainTip` - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - network, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, NoChainTip); + // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. + tokio::time::pause(); runtime.block_on(async move { let response_fut = rpc.get_blockchain_info(); @@ -601,18 +376,17 @@ proptest! { let (response, _) = tokio::join!(response_fut, mock_state_handler); prop_assert_eq!( - &response.err().unwrap().message, - "no chain tip available yet" + response.err().unwrap().message().to_string(), + "no chain tip available yet".to_string() ); - // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); - mempool.expect_no_requests().await?; state.expect_no_requests().await?; - Ok::<_, TestCaseError>(()) + // The queue task should continue without errors or panics + prop_assert!(mempool_tx_queue.now_or_never().is_none()); + + Ok(()) })?; } @@ -624,26 +398,17 @@ proptest! { ) { let (runtime, _init_guard) = zebra_test::init_async(); let _guard = runtime.enter(); - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = + mock_services(network.clone(), NoChainTip); + + // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. + tokio::time::pause(); // get arbitrary chain tip data let block_height = block.coinbase_height().unwrap(); let block_hash = block.hash(); let block_time = block.header.time; - // Start RPC with the mocked `ChainTip` - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - network.clone(), - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); - // check no requests were made during this test runtime.block_on(async move { let response_fut = rpc.get_blockchain_info(); @@ -718,14 +483,13 @@ proptest! { } }; - // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); - mempool.expect_no_requests().await?; state.expect_no_requests().await?; - Ok::<_, TestCaseError>(()) + // The queue task should continue without errors or panics + prop_assert!(mempool_tx_queue.now_or_never().is_none()); + + Ok(()) })?; } @@ -738,12 +502,11 @@ proptest! { ) { let (runtime, _init_guard) = zebra_test::init_async(); let _guard = runtime.enter(); - - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - // Create a mocked `ChainTip` let (chain_tip, _mock_chain_tip_sender) = MockChainTip::new(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, chain_tip); + + // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. + tokio::time::pause(); // Prepare the list of addresses. let address_strings = AddressStrings { @@ -753,21 +516,8 @@ proptest! { .collect(), }; - tokio::time::pause(); - // Start RPC with the mocked `ChainTip` runtime.block_on(async move { - let (rpc, _rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - network, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - chain_tip, - ); - // Build the future to call the RPC let call = rpc.get_address_balance(address_strings); @@ -792,7 +542,10 @@ proptest! { mempool.expect_no_requests().await?; state.expect_no_requests().await?; - Ok::<_, TestCaseError>(()) + // The queue task should continue without errors or panics + prop_assert!(mempool_tx_queue.now_or_never().is_none()); + + Ok(()) })?; } @@ -806,31 +559,17 @@ proptest! { ) { let (runtime, _init_guard) = zebra_test::init_async(); let _guard = runtime.enter(); + let (chain_tip, _mock_chain_tip_sender) = MockChainTip::new(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, chain_tip); + + // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. + tokio::time::pause(); prop_assume!(at_least_one_invalid_address .iter() .any(|string| string.parse::().is_err())); - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - // Create a mocked `ChainTip` - let (chain_tip, _mock_chain_tip_sender) = MockChainTip::new(); - - tokio::time::pause(); - - // Start RPC with the mocked `ChainTip` runtime.block_on(async move { - let (rpc, _rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - network, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - chain_tip, - ); let address_strings = AddressStrings { addresses: at_least_one_invalid_address, @@ -839,58 +578,37 @@ proptest! { // Build the future to call the RPC let result = rpc.get_address_balance(address_strings).await; - // Check that the invalid addresses lead to an error - prop_assert!( - matches!( - result, - Err(Error { - code: ErrorCode::InvalidParams, - .. - }) - ), - "Result is not a server error: {result:?}" - ); + check_err_code(result, ErrorCode::ServerError(-5))?; // Check no requests were made during this test mempool.expect_no_requests().await?; state.expect_no_requests().await?; - Ok::<_, TestCaseError>(()) + // The queue task should continue without errors or panics + prop_assert!(mempool_tx_queue.now_or_never().is_none()); + + Ok(()) })?; } /// Test the queue functionality using `send_raw_transaction` #[test] - fn rpc_queue_main_loop(tx in any::()) { + fn rpc_queue_main_loop(tx in any::(), network in any::()) { let (runtime, _init_guard) = zebra_test::init_async(); let _guard = runtime.enter(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, NoChainTip); - let transaction_hash = tx.hash(); + // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. + tokio::time::pause(); runtime.block_on(async move { - tokio::time::pause(); - - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - Mainnet, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); - - // send a transaction - let tx_bytes = tx - .zcash_serialize_to_vec() - .expect("Transaction serializes successfully"); + let transaction_hash = tx.hash(); + let tx_bytes = tx.zcash_serialize_to_vec()?; let tx_hex = hex::encode(&tx_bytes); - let send_task = tokio::spawn(rpc.send_raw_transaction(tx_hex)); - + let send_task = { + let rpc = rpc.clone(); + tokio::task::spawn(async move { rpc.send_raw_transaction(tx_hex).await }) + }; let tx_unmined = UnminedTx::from(tx); let expected_request = mempool::Request::Queue(vec![tx_unmined.clone().into()]); @@ -901,9 +619,7 @@ proptest! { .unwrap() .respond(Err(DummyError)); - let _ = send_task - .await - .expect("Sending raw transactions should not panic"); + let _ = send_task.await?; // advance enough time to have a new runner iteration let spacing = chrono::Duration::seconds(150); @@ -946,44 +662,31 @@ proptest! { state.expect_no_requests().await?; // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); + prop_assert!(mempool_tx_queue.now_or_never().is_none()); - Ok::<_, TestCaseError>(()) + Ok(()) })?; } /// Test we receive all transactions that are sent in a channel #[test] - fn rpc_queue_receives_all_transactions_from_channel(txs in any::<[Transaction; 2]>()) { + fn rpc_queue_receives_all_txs_from_channel(txs in any::<[Transaction; 2]>(), + network in any::()) { let (runtime, _init_guard) = zebra_test::init_async(); let _guard = runtime.enter(); + let (mut mempool, mut state, rpc, mempool_tx_queue) = mock_services(network, NoChainTip); - runtime.block_on(async move { - tokio::time::pause(); - - let mut mempool = MockService::build().for_prop_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_prop_tests(); - - let (rpc, rpc_tx_queue_task_handle) = RpcImpl::new( - "RPC test", - "RPC test", - Mainnet, - false, - true, - mempool.clone(), - Buffer::new(state.clone(), 1), - NoChainTip, - ); + // CORRECTNESS: Nothing in this test depends on real time, so we can speed it up. + tokio::time::pause(); + runtime.block_on(async move { let mut transactions_hash_set = HashSet::new(); for tx in txs.clone() { + let rpc_clone = rpc.clone(); // send a transaction - let tx_bytes = tx - .zcash_serialize_to_vec() - .expect("Transaction serializes successfully"); + let tx_bytes = tx.zcash_serialize_to_vec()?; let tx_hex = hex::encode(&tx_bytes); - let send_task = tokio::spawn(rpc.send_raw_transaction(tx_hex)); + let send_task = tokio::task::spawn(async move { rpc_clone.send_raw_transaction(tx_hex).await }); let tx_unmined = UnminedTx::from(tx.clone()); let expected_request = mempool::Request::Queue(vec![tx_unmined.clone().into()]); @@ -999,9 +702,7 @@ proptest! { .unwrap() .respond(Err(DummyError)); - let _ = send_task - .await - .expect("Sending raw transactions should not panic"); + let _ = send_task.await?; } // advance enough time to have a new runner iteration @@ -1049,10 +750,9 @@ proptest! { state.expect_no_requests().await?; // The queue task should continue without errors or panics - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - prop_assert!(rpc_tx_queue_task_result.is_none()); + prop_assert!(mempool_tx_queue.now_or_never().is_none()); - Ok::<_, TestCaseError>(()) + Ok(()) })?; } } @@ -1060,3 +760,94 @@ proptest! { #[derive(Clone, Copy, Debug, Error)] #[error("a dummy error type")] pub struct DummyError; + +// Helper functions + +/// Creates [`String`]s that won't deserialize into [`transaction::Hash`]. +fn invalid_txid() -> BoxedStrategy { + any::() + .prop_filter("string must not deserialize into TXID", |s| { + transaction::Hash::from_hex(s).is_err() + }) + .boxed() +} + +/// Checks that the given RPC response contains the given error code. +fn check_err_code( + rsp: Result, + error_code: ErrorCode, +) -> Result<(), TestCaseError> { + match rsp { + Err(e) => { + prop_assert!( + e.code() == error_code.code(), + "the RPC response must match the error code: {:?}", + error_code.code() + ); + } + Ok(_) => { + prop_assert!(false, "expected an error response, but got Ok"); + } + } + + Ok(()) +} + +/// Creates mocked: +/// +/// 1. mempool service, +/// 2. state service, +/// 3. rpc service, +/// +/// and a handle to the mempool tx queue. +fn mock_services( + network: Network, + chain_tip: Tip, +) -> ( + zebra_test::mock_service::MockService< + zebra_node_services::mempool::Request, + zebra_node_services::mempool::Response, + zebra_test::mock_service::PropTestAssertion, + >, + zebra_test::mock_service::MockService< + zebra_state::ReadRequest, + zebra_state::ReadResponse, + zebra_test::mock_service::PropTestAssertion, + >, + methods::RpcImpl< + zebra_test::mock_service::MockService< + zebra_node_services::mempool::Request, + zebra_node_services::mempool::Response, + zebra_test::mock_service::PropTestAssertion, + >, + tower::buffer::Buffer< + zebra_test::mock_service::MockService< + zebra_state::ReadRequest, + zebra_state::ReadResponse, + zebra_test::mock_service::PropTestAssertion, + >, + zebra_state::ReadRequest, + >, + Tip, + >, + tokio::task::JoinHandle<()>, +) +where + Tip: ChainTip + Clone + Send + Sync + 'static, +{ + let mempool = MockService::build().for_prop_tests(); + let state = MockService::build().for_prop_tests(); + + let (rpc, mempool_tx_queue) = RpcImpl::new( + "RPC test", + "RPC test", + network, + false, + true, + mempool.clone(), + Buffer::new(state.clone(), 1), + chain_tip, + ); + + (mempool, state, rpc, mempool_tx_queue) +} diff --git a/zebra-rpc/src/methods/tests/snapshot.rs b/zebra-rpc/src/methods/tests/snapshot.rs index 02c633af260..89ee464c70a 100644 --- a/zebra-rpc/src/methods/tests/snapshot.rs +++ b/zebra-rpc/src/methods/tests/snapshot.rs @@ -5,9 +5,11 @@ //! cargo insta test --review --release -p zebra-rpc --lib -- test_rpc_response_data //! ``` -use std::collections::BTreeMap; +use std::{collections::BTreeMap, sync::Arc}; +use futures::FutureExt; use insta::dynamic_redaction; +use jsonrpsee::core::RpcResult as Result; use tower::buffer::Buffer; use zebra_chain::{ @@ -229,12 +231,10 @@ async fn test_rpc_response_data_for_network(network: &Network) { snapshot_rpc_getblockchaininfo("", get_blockchain_info, &settings); // get the first transaction of the first block which is not the genesis - let first_block_first_transaction = &blocks[1].transactions[0]; + let first_block_first_tx = &blocks[1].transactions[0]; // build addresses - let address = &first_block_first_transaction.outputs()[1] - .address(network) - .unwrap(); + let address = &first_block_first_tx.outputs()[1].address(network).unwrap(); let addresses = vec![address.to_string()]; // `getaddressbalance` @@ -407,8 +407,9 @@ async fn test_rpc_response_data_for_network(network: &Network) { // `getrawtransaction` verbosity=0 // - // - similar to `getrawmempool` described above, a mempool request will be made to get the requested - // transaction from the mempool, response will be empty as we have this transaction in state + // - Similarly to `getrawmempool` described above, a mempool request will be made to get the + // requested transaction from the mempool. Response will be empty as we have this transaction + // in the state. let mempool_req = mempool .expect_request_that(|request| { matches!(request, mempool::Request::TransactionsByMinedId(_)) @@ -417,13 +418,12 @@ async fn test_rpc_response_data_for_network(network: &Network) { responder.respond(mempool::Response::Transactions(vec![])); }); - // make the api call - let get_raw_transaction = - rpc.get_raw_transaction(first_block_first_transaction.hash().encode_hex(), Some(0u8)); - let (response, _) = futures::join!(get_raw_transaction, mempool_req); - let get_raw_transaction = response.expect("We should have a GetRawTransaction struct"); + let txid = first_block_first_tx.hash().encode_hex::(); - snapshot_rpc_getrawtransaction("verbosity_0", get_raw_transaction, &settings); + let rpc_req = rpc.get_raw_transaction(txid.clone(), Some(0u8)); + let (rsp, _) = futures::join!(rpc_req, mempool_req); + settings.bind(|| insta::assert_json_snapshot!(format!("getrawtransaction_verbosity=0"), rsp)); + mempool.expect_no_requests().await; // `getrawtransaction` verbosity=1 let mempool_req = mempool @@ -434,13 +434,31 @@ async fn test_rpc_response_data_for_network(network: &Network) { responder.respond(mempool::Response::Transactions(vec![])); }); - // make the api call - let get_raw_transaction = - rpc.get_raw_transaction(first_block_first_transaction.hash().encode_hex(), Some(1u8)); - let (response, _) = futures::join!(get_raw_transaction, mempool_req); - let get_raw_transaction = response.expect("We should have a GetRawTransaction struct"); + let rpc_req = rpc.get_raw_transaction(txid, Some(1u8)); + let (rsp, _) = futures::join!(rpc_req, mempool_req); + settings.bind(|| insta::assert_json_snapshot!(format!("getrawtransaction_verbosity=1"), rsp)); + mempool.expect_no_requests().await; + + // `getrawtransaction` with unknown txid + let mempool_req = mempool + .expect_request_that(|request| { + matches!(request, mempool::Request::TransactionsByMinedId(_)) + }) + .map(|responder| { + responder.respond(mempool::Response::Transactions(vec![])); + }); - snapshot_rpc_getrawtransaction("verbosity_1", get_raw_transaction, &settings); + let rpc_req = rpc.get_raw_transaction(transaction::Hash::from([0; 32]).encode_hex(), Some(1)); + let (rsp, _) = futures::join!(rpc_req, mempool_req); + settings.bind(|| insta::assert_json_snapshot!(format!("getrawtransaction_unknown_txid"), rsp)); + mempool.expect_no_requests().await; + + // `getrawtransaction` with an invalid TXID + let rsp = rpc + .get_raw_transaction("aBadC0de".to_owned(), Some(1)) + .await; + settings.bind(|| insta::assert_json_snapshot!(format!("getrawtransaction_invalid_txid"), rsp)); + mempool.expect_no_requests().await; // `getaddresstxids` let get_address_tx_ids = rpc @@ -666,17 +684,6 @@ fn snapshot_rpc_getrawmempool(raw_mempool: Vec, settings: &insta::Settin settings.bind(|| insta::assert_json_snapshot!("get_raw_mempool", raw_mempool)); } -/// Snapshot `getrawtransaction` response, using `cargo insta` and JSON serialization. -fn snapshot_rpc_getrawtransaction( - variant: &'static str, - raw_transaction: GetRawTransaction, - settings: &insta::Settings, -) { - settings.bind(|| { - insta::assert_json_snapshot!(format!("get_raw_transaction_{variant}"), raw_transaction) - }); -} - /// Snapshot valid `getaddressbalance` response, using `cargo insta` and JSON serialization. fn snapshot_rpc_getaddresstxids_valid( variant: &'static str, diff --git a/zebra-rpc/src/methods/tests/snapshot/get_block_template_rpcs.rs b/zebra-rpc/src/methods/tests/snapshot/get_block_template_rpcs.rs index b2e012c7bcd..a512faf7cfc 100644 --- a/zebra-rpc/src/methods/tests/snapshot/get_block_template_rpcs.rs +++ b/zebra-rpc/src/methods/tests/snapshot/get_block_template_rpcs.rs @@ -12,7 +12,7 @@ use std::{ use hex::FromHex; use insta::Settings; -use jsonrpc_core::Result; +use jsonrpsee::core::RpcResult as Result; use tower::{buffer::Buffer, Service}; use zebra_chain::{ @@ -47,7 +47,7 @@ use crate::methods::{ }, hex_data::HexData, tests::{snapshot::EXCESSIVE_BLOCK_HEIGHT, utils::fake_history_tree}, - GetBlockHash, GetBlockTemplateRpc, GetBlockTemplateRpcImpl, + GetBlockHash, GetBlockTemplateRpcImpl, GetBlockTemplateRpcServer, }; pub async fn test_responses( @@ -105,7 +105,8 @@ pub async fn test_responses( extra_coinbase_data: None, debug_like_zcashd: true, // TODO: Use default field values when optional features are enabled in tests #8183 - ..Default::default() + #[cfg(feature = "internal-miner")] + internal_miner: true, }; // nu5 block height @@ -487,20 +488,18 @@ pub async fn test_responses( // `z_listunifiedreceivers` let ua1 = String::from("u1l8xunezsvhq8fgzfl7404m450nwnd76zshscn6nfys7vyz2ywyh4cc5daaq0c7q2su5lqfh23sp7fkf3kt27ve5948mzpfdvckzaect2jtte308mkwlycj2u0eac077wu70vqcetkxf"); - let z_list_unified_receivers = - tokio::spawn(get_block_template_rpc.z_list_unified_receivers(ua1)) - .await - .expect("unexpected panic in z_list_unified_receivers RPC task") - .expect("unexpected error in z_list_unified_receivers RPC call"); + let z_list_unified_receivers = get_block_template_rpc + .z_list_unified_receivers(ua1) + .await + .expect("unexpected error in z_list_unified_receivers RPC call"); snapshot_rpc_z_listunifiedreceivers("ua1", z_list_unified_receivers, &settings); let ua2 = String::from("u1uf4qsmh037x2jp6k042h9d2w22wfp39y9cqdf8kcg0gqnkma2gf4g80nucnfeyde8ev7a6kf0029gnwqsgadvaye9740gzzpmr67nfkjjvzef7rkwqunqga4u4jges4tgptcju5ysd0"); - let z_list_unified_receivers = - tokio::spawn(get_block_template_rpc.z_list_unified_receivers(ua2)) - .await - .expect("unexpected panic in z_list_unified_receivers RPC task") - .expect("unexpected error in z_list_unified_receivers RPC call"); + let z_list_unified_receivers = get_block_template_rpc + .z_list_unified_receivers(ua2) + .await + .expect("unexpected error in z_list_unified_receivers RPC call"); snapshot_rpc_z_listunifiedreceivers("ua2", z_list_unified_receivers, &settings); } diff --git a/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_hash_verbosity_2@mainnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_hash_verbosity_2@mainnet_10.snap index 93010ad42d4..51729b13293 100644 --- a/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_hash_verbosity_2@mainnet_10.snap +++ b/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_hash_verbosity_2@mainnet_10.snap @@ -10,7 +10,11 @@ expression: block "merkleroot": "851bf6fbf7a976327817c738c489d7fa657752445430922d94c983c0b9ed4609", "finalsaplingroot": "0000000000000000000000000000000000000000000000000000000000000000", "tx": [ - "851bf6fbf7a976327817c738c489d7fa657752445430922d94c983c0b9ed4609" + { + "hex": "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff025100ffffffff0250c30000000000002321027a46eb513588b01b37ea24303f4b628afd12cc20df789fede0921e43cad3e875acd43000000000000017a9147d46a730d31f97b1930d3368a967c309bd4d136a8700000000", + "height": 1, + "confirmations": 10 + } ], "time": 1477671596, "nonce": "9057977ea6d4ae867decc96359fcf2db8cdebcbfb3bd549de4f21f16cfe83475", diff --git a/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_hash_verbosity_2@testnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_hash_verbosity_2@testnet_10.snap index 5bd22590f1b..51bbfc72f05 100644 --- a/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_hash_verbosity_2@testnet_10.snap +++ b/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_hash_verbosity_2@testnet_10.snap @@ -10,7 +10,11 @@ expression: block "merkleroot": "f37e9f691fffb635de0999491d906ee85ba40cd36dae9f6e5911a8277d7c5f75", "finalsaplingroot": "0000000000000000000000000000000000000000000000000000000000000000", "tx": [ - "f37e9f691fffb635de0999491d906ee85ba40cd36dae9f6e5911a8277d7c5f75" + { + "hex": "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff03510101ffffffff0250c30000000000002321025229e1240a21004cf8338db05679fa34753706e84f6aebba086ba04317fd8f99acd43000000000000017a914ef775f1f997f122a062fff1a2d7443abd1f9c6428700000000", + "height": 1, + "confirmations": 10 + } ], "time": 1477674473, "nonce": "0000e5739438a096ca89cde16bcf6001e0c5a7ce6f7c591d26314c26c2560000", diff --git a/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_height_verbosity_2@mainnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_height_verbosity_2@mainnet_10.snap index 93010ad42d4..51729b13293 100644 --- a/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_height_verbosity_2@mainnet_10.snap +++ b/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_height_verbosity_2@mainnet_10.snap @@ -10,7 +10,11 @@ expression: block "merkleroot": "851bf6fbf7a976327817c738c489d7fa657752445430922d94c983c0b9ed4609", "finalsaplingroot": "0000000000000000000000000000000000000000000000000000000000000000", "tx": [ - "851bf6fbf7a976327817c738c489d7fa657752445430922d94c983c0b9ed4609" + { + "hex": "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff025100ffffffff0250c30000000000002321027a46eb513588b01b37ea24303f4b628afd12cc20df789fede0921e43cad3e875acd43000000000000017a9147d46a730d31f97b1930d3368a967c309bd4d136a8700000000", + "height": 1, + "confirmations": 10 + } ], "time": 1477671596, "nonce": "9057977ea6d4ae867decc96359fcf2db8cdebcbfb3bd549de4f21f16cfe83475", diff --git a/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_height_verbosity_2@testnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_height_verbosity_2@testnet_10.snap index 5bd22590f1b..51bbfc72f05 100644 --- a/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_height_verbosity_2@testnet_10.snap +++ b/zebra-rpc/src/methods/tests/snapshots/get_block_verbose_height_verbosity_2@testnet_10.snap @@ -10,7 +10,11 @@ expression: block "merkleroot": "f37e9f691fffb635de0999491d906ee85ba40cd36dae9f6e5911a8277d7c5f75", "finalsaplingroot": "0000000000000000000000000000000000000000000000000000000000000000", "tx": [ - "f37e9f691fffb635de0999491d906ee85ba40cd36dae9f6e5911a8277d7c5f75" + { + "hex": "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff03510101ffffffff0250c30000000000002321025229e1240a21004cf8338db05679fa34753706e84f6aebba086ba04317fd8f99acd43000000000000017a914ef775f1f997f122a062fff1a2d7443abd1f9c6428700000000", + "height": 1, + "confirmations": 10 + } ], "time": 1477674473, "nonce": "0000e5739438a096ca89cde16bcf6001e0c5a7ce6f7c591d26314c26c2560000", diff --git a/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_0@mainnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_0@mainnet_10.snap deleted file mode 100644 index fe57f682126..00000000000 --- a/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_0@mainnet_10.snap +++ /dev/null @@ -1,5 +0,0 @@ ---- -source: zebra-rpc/src/methods/tests/snapshot.rs -expression: raw_transaction ---- -"01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff025100ffffffff0250c30000000000002321027a46eb513588b01b37ea24303f4b628afd12cc20df789fede0921e43cad3e875acd43000000000000017a9147d46a730d31f97b1930d3368a967c309bd4d136a8700000000" diff --git a/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_0@testnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_0@testnet_10.snap deleted file mode 100644 index 6f7145404de..00000000000 --- a/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_0@testnet_10.snap +++ /dev/null @@ -1,5 +0,0 @@ ---- -source: zebra-rpc/src/methods/tests/snapshot.rs -expression: raw_transaction ---- -"01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff03510101ffffffff0250c30000000000002321025229e1240a21004cf8338db05679fa34753706e84f6aebba086ba04317fd8f99acd43000000000000017a914ef775f1f997f122a062fff1a2d7443abd1f9c6428700000000" diff --git a/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_1@mainnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_1@mainnet_10.snap deleted file mode 100644 index 25091fe3fb5..00000000000 --- a/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_1@mainnet_10.snap +++ /dev/null @@ -1,9 +0,0 @@ ---- -source: zebra-rpc/src/methods/tests/snapshot.rs -expression: raw_transaction ---- -{ - "hex": "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff025100ffffffff0250c30000000000002321027a46eb513588b01b37ea24303f4b628afd12cc20df789fede0921e43cad3e875acd43000000000000017a9147d46a730d31f97b1930d3368a967c309bd4d136a8700000000", - "height": 1, - "confirmations": 10 -} diff --git a/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_1@testnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_1@testnet_10.snap deleted file mode 100644 index 61499b2e880..00000000000 --- a/zebra-rpc/src/methods/tests/snapshots/get_raw_transaction_verbosity_1@testnet_10.snap +++ /dev/null @@ -1,9 +0,0 @@ ---- -source: zebra-rpc/src/methods/tests/snapshot.rs -expression: raw_transaction ---- -{ - "hex": "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff03510101ffffffff0250c30000000000002321025229e1240a21004cf8338db05679fa34753706e84f6aebba086ba04317fd8f99acd43000000000000017a914ef775f1f997f122a062fff1a2d7443abd1f9c6428700000000", - "height": 1, - "confirmations": 10 -} diff --git a/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_invalid_txid@mainnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_invalid_txid@mainnet_10.snap new file mode 100644 index 00000000000..e048f7ad516 --- /dev/null +++ b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_invalid_txid@mainnet_10.snap @@ -0,0 +1,11 @@ +--- +source: zebra-rpc/src/methods/tests/snapshot.rs +expression: rsp +snapshot_kind: text +--- +{ + "Err": { + "code": -5, + "message": "Invalid string length" + } +} diff --git a/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_invalid_txid@testnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_invalid_txid@testnet_10.snap new file mode 100644 index 00000000000..e048f7ad516 --- /dev/null +++ b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_invalid_txid@testnet_10.snap @@ -0,0 +1,11 @@ +--- +source: zebra-rpc/src/methods/tests/snapshot.rs +expression: rsp +snapshot_kind: text +--- +{ + "Err": { + "code": -5, + "message": "Invalid string length" + } +} diff --git a/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_unknown_txid@mainnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_unknown_txid@mainnet_10.snap new file mode 100644 index 00000000000..878c8505a19 --- /dev/null +++ b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_unknown_txid@mainnet_10.snap @@ -0,0 +1,11 @@ +--- +source: zebra-rpc/src/methods/tests/snapshot.rs +expression: rsp +snapshot_kind: text +--- +{ + "Err": { + "code": -5, + "message": "No such mempool or main chain transaction" + } +} diff --git a/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_unknown_txid@testnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_unknown_txid@testnet_10.snap new file mode 100644 index 00000000000..878c8505a19 --- /dev/null +++ b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_unknown_txid@testnet_10.snap @@ -0,0 +1,11 @@ +--- +source: zebra-rpc/src/methods/tests/snapshot.rs +expression: rsp +snapshot_kind: text +--- +{ + "Err": { + "code": -5, + "message": "No such mempool or main chain transaction" + } +} diff --git a/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=0@mainnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=0@mainnet_10.snap new file mode 100644 index 00000000000..90fa5021b56 --- /dev/null +++ b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=0@mainnet_10.snap @@ -0,0 +1,8 @@ +--- +source: zebra-rpc/src/methods/tests/snapshot.rs +expression: rsp +snapshot_kind: text +--- +{ + "Ok": "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff025100ffffffff0250c30000000000002321027a46eb513588b01b37ea24303f4b628afd12cc20df789fede0921e43cad3e875acd43000000000000017a9147d46a730d31f97b1930d3368a967c309bd4d136a8700000000" +} diff --git a/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=0@testnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=0@testnet_10.snap new file mode 100644 index 00000000000..673c9f7ce89 --- /dev/null +++ b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=0@testnet_10.snap @@ -0,0 +1,8 @@ +--- +source: zebra-rpc/src/methods/tests/snapshot.rs +expression: rsp +snapshot_kind: text +--- +{ + "Ok": "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff03510101ffffffff0250c30000000000002321025229e1240a21004cf8338db05679fa34753706e84f6aebba086ba04317fd8f99acd43000000000000017a914ef775f1f997f122a062fff1a2d7443abd1f9c6428700000000" +} diff --git a/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=1@mainnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=1@mainnet_10.snap new file mode 100644 index 00000000000..b78a6686336 --- /dev/null +++ b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=1@mainnet_10.snap @@ -0,0 +1,12 @@ +--- +source: zebra-rpc/src/methods/tests/snapshot.rs +expression: rsp +snapshot_kind: text +--- +{ + "Ok": { + "hex": "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff025100ffffffff0250c30000000000002321027a46eb513588b01b37ea24303f4b628afd12cc20df789fede0921e43cad3e875acd43000000000000017a9147d46a730d31f97b1930d3368a967c309bd4d136a8700000000", + "height": 1, + "confirmations": 10 + } +} diff --git a/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=1@testnet_10.snap b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=1@testnet_10.snap new file mode 100644 index 00000000000..ab133db9b1a --- /dev/null +++ b/zebra-rpc/src/methods/tests/snapshots/getrawtransaction_verbosity=1@testnet_10.snap @@ -0,0 +1,12 @@ +--- +source: zebra-rpc/src/methods/tests/snapshot.rs +expression: rsp +snapshot_kind: text +--- +{ + "Ok": { + "hex": "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff03510101ffffffff0250c30000000000002321025229e1240a21004cf8338db05679fa34753706e84f6aebba086ba04317fd8f99acd43000000000000017a914ef775f1f997f122a062fff1a2d7443abd1f9c6428700000000", + "height": 1, + "confirmations": 10 + } +} diff --git a/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_by_non_existent_hash@custom_testnet.snap b/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_by_non_existent_hash@custom_testnet.snap index d0013994ab0..7801c859a27 100644 --- a/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_by_non_existent_hash@custom_testnet.snap +++ b/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_by_non_existent_hash@custom_testnet.snap @@ -1,10 +1,11 @@ --- source: zebra-rpc/src/methods/tests/snapshot.rs expression: treestate +snapshot_kind: text --- { "Err": { "code": -8, - "message": "the requested block was not found" + "message": "the requested block is not in the main chain" } } diff --git a/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_excessive_block_height@custom_testnet.snap b/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_excessive_block_height@custom_testnet.snap index d0013994ab0..7801c859a27 100644 --- a/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_excessive_block_height@custom_testnet.snap +++ b/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_excessive_block_height@custom_testnet.snap @@ -1,10 +1,11 @@ --- source: zebra-rpc/src/methods/tests/snapshot.rs expression: treestate +snapshot_kind: text --- { "Err": { "code": -8, - "message": "the requested block was not found" + "message": "the requested block is not in the main chain" } } diff --git a/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_unparsable_hash_or_height@custom_testnet.snap b/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_unparsable_hash_or_height@custom_testnet.snap index a45d7e298dc..d7b3c2b1ff0 100644 --- a/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_unparsable_hash_or_height@custom_testnet.snap +++ b/zebra-rpc/src/methods/tests/snapshots/z_get_treestate_unparsable_hash_or_height@custom_testnet.snap @@ -1,10 +1,11 @@ --- source: zebra-rpc/src/methods/tests/snapshot.rs expression: treestate +snapshot_kind: text --- { "Err": { - "code": 0, + "code": -8, "message": "parse error: could not convert the input string to a hash or height" } } diff --git a/zebra-rpc/src/methods/tests/vectors.rs b/zebra-rpc/src/methods/tests/vectors.rs index 007a89ee893..e1f559b8e4f 100644 --- a/zebra-rpc/src/methods/tests/vectors.rs +++ b/zebra-rpc/src/methods/tests/vectors.rs @@ -1,15 +1,18 @@ //! Fixed test vectors for RPC methods. use std::ops::RangeInclusive; +use std::sync::Arc; +use futures::FutureExt; use tower::buffer::Buffer; +use zebra_chain::serialization::ZcashSerialize; use zebra_chain::{ amount::Amount, block::Block, chain_tip::{mock::MockChainTip, NoChainTip}, parameters::Network::*, - serialization::{ZcashDeserializeInto, ZcashSerialize}, + serialization::ZcashDeserializeInto, transaction::UnminedTxId, }; use zebra_node_services::BoxError; @@ -176,7 +179,7 @@ async fn rpc_getblock() { tx: block .transactions .iter() - .map(|tx| tx.hash().encode_hex()) + .map(|tx| GetBlockTransaction::Hash(tx.hash())) .collect(), trees, size: None, @@ -219,7 +222,7 @@ async fn rpc_getblock() { tx: block .transactions .iter() - .map(|tx| tx.hash().encode_hex()) + .map(|tx| GetBlockTransaction::Hash(tx.hash())) .collect(), trees, size: None, @@ -262,7 +265,11 @@ async fn rpc_getblock() { tx: block .transactions .iter() - .map(|tx| tx.hash().encode_hex()) + .map(|tx| GetBlockTransaction::Object(TransactionObject { + hex: (*tx).clone().into(), + height: Some(i.try_into().expect("valid u32")), + confirmations: Some((blocks.len() - i).try_into().expect("valid i64")) + })) .collect(), trees, size: None, @@ -305,7 +312,11 @@ async fn rpc_getblock() { tx: block .transactions .iter() - .map(|tx| tx.hash().encode_hex()) + .map(|tx| GetBlockTransaction::Object(TransactionObject { + hex: (*tx).clone().into(), + height: Some(i.try_into().expect("valid u32")), + confirmations: Some((blocks.len() - i).try_into().expect("valid i64")) + })) .collect(), trees, size: None, @@ -348,7 +359,7 @@ async fn rpc_getblock() { tx: block .transactions .iter() - .map(|tx| tx.hash().encode_hex()) + .map(|tx| GetBlockTransaction::Hash(tx.hash())) .collect(), trees, size: None, @@ -391,7 +402,7 @@ async fn rpc_getblock() { tx: block .transactions .iter() - .map(|tx| tx.hash().encode_hex()) + .map(|tx| GetBlockTransaction::Hash(tx.hash())) .collect(), trees, size: None, @@ -485,7 +496,7 @@ async fn rpc_getblock_missing_error() { // Make sure Zebra returns the correct error code `-8` for missing blocks // https://github.com/zcash/lightwalletd/blob/v0.4.16/common/common.go#L287-L290 - let block_future = tokio::spawn(rpc.get_block("0".to_string(), Some(0u8))); + let block_future = tokio::spawn(async move { rpc.get_block("0".to_string(), Some(0u8)).await }); // Make the mock service respond with no block let response_handler = state @@ -493,11 +504,10 @@ async fn rpc_getblock_missing_error() { .await; response_handler.respond(zebra_state::ReadResponse::Block(None)); - let block_response = block_future.await; - let block_response = block_response - .expect("unexpected panic in spawned request future") - .expect_err("unexpected success from missing block state response"); - assert_eq!(block_response.code, ErrorCode::ServerError(-8),); + let block_response = block_future.await.expect("block future should not panic"); + let block_response = + block_response.expect_err("unexpected success from missing block state response"); + assert_eq!(block_response.code(), ErrorCode::ServerError(-8).code()); // Now check the error string the way `lightwalletd` checks it assert_eq!( @@ -722,9 +732,12 @@ async fn rpc_getrawtransaction() { conventional_fee: Amount::zero(), }])); }); - let get_tx_req = rpc.get_raw_transaction(tx.hash().encode_hex(), Some(0u8)); - let (response, _) = futures::join!(get_tx_req, mempool_req); - let get_tx = response.expect("We should have a GetRawTransaction struct"); + + let rpc_req = rpc.get_raw_transaction(tx.hash().encode_hex(), Some(0u8)); + + let (rsp, _) = futures::join!(rpc_req, mempool_req); + let get_tx = rsp.expect("we should have a `GetRawTransaction` struct"); + if let GetRawTransaction::Raw(raw_tx) = get_tx { assert_eq!(raw_tx.as_ref(), tx.zcash_serialize_to_vec().unwrap()); } else { @@ -752,12 +765,14 @@ async fn rpc_getrawtransaction() { let run_state_test_case = |block_idx: usize, block: Arc, tx: Arc| { let read_state = read_state.clone(); - let tx_hash = tx.hash(); - let get_tx_verbose_0_req = rpc.get_raw_transaction(tx_hash.encode_hex(), Some(0u8)); - let get_tx_verbose_1_req = rpc.get_raw_transaction(tx_hash.encode_hex(), Some(1u8)); + let txid = tx.hash(); + let hex_txid = txid.encode_hex::(); + + let get_tx_verbose_0_req = rpc.get_raw_transaction(hex_txid.clone(), Some(0u8)); + let get_tx_verbose_1_req = rpc.get_raw_transaction(hex_txid, Some(1u8)); async move { - let (response, _) = futures::join!(get_tx_verbose_0_req, make_mempool_req(tx_hash)); + let (response, _) = futures::join!(get_tx_verbose_0_req, make_mempool_req(txid)); let get_tx = response.expect("We should have a GetRawTransaction struct"); if let GetRawTransaction::Raw(raw_tx) = get_tx { assert_eq!(raw_tx.as_ref(), tx.zcash_serialize_to_vec().unwrap()); @@ -765,18 +780,22 @@ async fn rpc_getrawtransaction() { unreachable!("Should return a Raw enum") } - let (response, _) = futures::join!(get_tx_verbose_1_req, make_mempool_req(tx_hash)); - let GetRawTransaction::Object { + let (response, _) = futures::join!(get_tx_verbose_1_req, make_mempool_req(txid)); + + let GetRawTransaction::Object(TransactionObject { hex, height, confirmations, - } = response.expect("We should have a GetRawTransaction struct") + }) = response.expect("We should have a GetRawTransaction struct") else { unreachable!("Should return a Raw enum") }; + let height = height.expect("state requests should have height"); + let confirmations = confirmations.expect("state requests should have confirmations"); + assert_eq!(hex.as_ref(), tx.zcash_serialize_to_vec().unwrap()); - assert_eq!(height, block_idx as i32); + assert_eq!(height, block_idx as u32); let depth_response = read_state .oneshot(zebra_state::ReadRequest::Depth(block.hash())) @@ -870,25 +889,18 @@ async fn rpc_getaddresstxids_invalid_arguments() { ); // call the method with an invalid address string - let address = "11111111".to_string(); - let addresses = vec![address.clone()]; - let start: u32 = 1; - let end: u32 = 2; - let error = rpc + let rpc_rsp = rpc .get_address_tx_ids(GetAddressTxIdsRequest { - addresses: addresses.clone(), - start, - end, + addresses: vec!["t1invalidaddress".to_owned()], + start: 1, + end: 2, }) .await .unwrap_err(); - assert_eq!( - error.message, - format!( - "invalid address \"{}\": parse error: t-addr decoding error", - address.clone() - ) - ); + + assert_eq!(rpc_rsp.code(), ErrorCode::ServerError(-5).code()); + + mempool.expect_no_requests().await; // create a valid address let address = "t3Vz22vK5z2LcKEdg16Yv4FFneEL1zg9ojd".to_string(); @@ -906,7 +918,7 @@ async fn rpc_getaddresstxids_invalid_arguments() { .await .unwrap_err(); assert_eq!( - error.message, + error.message(), "start Height(2) must be less than or equal to end Height(1)".to_string() ); @@ -922,7 +934,7 @@ async fn rpc_getaddresstxids_invalid_arguments() { .await .unwrap_err(); assert_eq!( - error.message, + error.message(), "start Height(0) and end Height(1) must both be greater than zero".to_string() ); @@ -938,7 +950,7 @@ async fn rpc_getaddresstxids_invalid_arguments() { .await .unwrap_err(); assert_eq!( - error.message, + error.message(), "start Height(1) and end Height(11) must both be less than or equal to the chain tip Height(10)".to_string() ); @@ -1078,17 +1090,13 @@ async fn rpc_getaddressutxos_invalid_arguments() { ); // call the method with an invalid address string - let address = "11111111".to_string(); - let addresses = vec![address.clone()]; let error = rpc .0 - .get_address_utxos(AddressStrings::new(addresses)) + .get_address_utxos(AddressStrings::new(vec!["t1invalidaddress".to_owned()])) .await .unwrap_err(); - assert_eq!( - error.message, - format!("invalid address \"{address}\": parse error: t-addr decoding error") - ); + + assert_eq!(error.code(), ErrorCode::ServerError(-5).code()); mempool.expect_no_requests().await; state.expect_no_requests().await; @@ -1245,7 +1253,10 @@ async fn rpc_getblockcount_empty_state() { assert!(get_block_count.is_err()); // Check the error we got is the correct one - assert_eq!(get_block_count.err().unwrap().message, "No blocks in state"); + assert_eq!( + get_block_count.err().unwrap().message(), + "No blocks in state" + ); mempool.expect_no_requests().await; } @@ -1552,7 +1563,8 @@ async fn rpc_getblocktemplate_mining_address(use_p2pkh: bool) { extra_coinbase_data: None, debug_like_zcashd: true, // TODO: Use default field values when optional features are enabled in tests #8183 - ..Default::default() + #[cfg(feature = "internal-miner")] + internal_miner: true, }; // nu5 block height @@ -1688,8 +1700,8 @@ async fn rpc_getblocktemplate_mining_address(use_p2pkh: bool) { .expect_err("needs an error when estimated distance to network chain tip is far"); assert_eq!( - get_block_template_sync_error.code, - ErrorCode::ServerError(-10) + get_block_template_sync_error.code(), + ErrorCode::ServerError(-10).code() ); mock_sync_status.set_is_close_to_tip(false); @@ -1701,8 +1713,8 @@ async fn rpc_getblocktemplate_mining_address(use_p2pkh: bool) { .expect_err("needs an error when syncer is not close to tip"); assert_eq!( - get_block_template_sync_error.code, - ErrorCode::ServerError(-10) + get_block_template_sync_error.code(), + ErrorCode::ServerError(-10).code() ); mock_chain_tip_sender.send_estimated_distance_to_network_chain_tip(Some(200)); @@ -1712,8 +1724,8 @@ async fn rpc_getblocktemplate_mining_address(use_p2pkh: bool) { .expect_err("needs an error when syncer is not close to tip or estimated distance to network chain tip is far"); assert_eq!( - get_block_template_sync_error.code, - ErrorCode::ServerError(-10) + get_block_template_sync_error.code(), + ErrorCode::ServerError(-10).code() ); let get_block_template_sync_error = get_block_template_rpc @@ -1724,7 +1736,10 @@ async fn rpc_getblocktemplate_mining_address(use_p2pkh: bool) { .await .expect_err("needs an error when called in proposal mode without data"); - assert_eq!(get_block_template_sync_error.code, ErrorCode::InvalidParams); + assert_eq!( + get_block_template_sync_error.code(), + ErrorCode::InvalidParams.code() + ); let get_block_template_sync_error = get_block_template_rpc .get_block_template(Some(get_block_template::JsonParameters { @@ -1734,7 +1749,10 @@ async fn rpc_getblocktemplate_mining_address(use_p2pkh: bool) { .await .expect_err("needs an error when passing in block data in template mode"); - assert_eq!(get_block_template_sync_error.code, ErrorCode::InvalidParams); + assert_eq!( + get_block_template_sync_error.code(), + ErrorCode::InvalidParams.code() + ); // The long poll id is valid, so it returns a state error instead let get_block_template_sync_error = get_block_template_rpc @@ -1752,8 +1770,8 @@ async fn rpc_getblocktemplate_mining_address(use_p2pkh: bool) { .expect_err("needs an error when the state is empty"); assert_eq!( - get_block_template_sync_error.code, - ErrorCode::ServerError(-10) + get_block_template_sync_error.code(), + ErrorCode::ServerError(-10).code() ); // Try getting mempool transactions with a different tip hash @@ -2006,7 +2024,8 @@ async fn rpc_getdifficulty() { extra_coinbase_data: None, debug_like_zcashd: true, // TODO: Use default field values when optional features are enabled in tests #8183 - ..Default::default() + #[cfg(feature = "internal-miner")] + internal_miner: true, }; // nu5 block height diff --git a/zebra-rpc/src/server.rs b/zebra-rpc/src/server.rs index 73fcde65f6b..c787071d74d 100644 --- a/zebra-rpc/src/server.rs +++ b/zebra-rpc/src/server.rs @@ -7,12 +7,11 @@ //! See the full list of //! [Differences between JSON-RPC 1.0 and 2.0.](https://www.simple-is-better.org/rpc/#differences-between-1-0-and-2-0) -use std::{fmt, panic, thread::available_parallelism}; +use std::{fmt, panic}; use cookie::Cookie; -use http_request_compatibility::With; -use jsonrpc_core::{Compatibility, MetaIoHandler}; -use jsonrpc_http_server::{CloseHandle, ServerBuilder}; +use jsonrpsee::server::middleware::rpc::RpcServiceBuilder; +use jsonrpsee::server::{Server, ServerHandle}; use tokio::task::JoinHandle; use tower::Service; use tracing::*; @@ -25,17 +24,18 @@ use zebra_node_services::mempool; use crate::{ config::Config, - methods::{Rpc, RpcImpl}, + methods::{RpcImpl, RpcServer as _}, server::{ - http_request_compatibility::HttpRequestMiddleware, + http_request_compatibility::HttpRequestMiddlewareLayer, rpc_call_compatibility::FixRpcResponseMiddleware, }, }; #[cfg(feature = "getblocktemplate-rpcs")] -use crate::methods::{GetBlockTemplateRpc, GetBlockTemplateRpcImpl}; +use crate::methods::{GetBlockTemplateRpcImpl, GetBlockTemplateRpcServer}; pub mod cookie; +pub mod error; pub mod http_request_compatibility; pub mod rpc_call_compatibility; @@ -54,8 +54,8 @@ pub struct RpcServer { /// Zebra's application version, with build metadata. build_version: String, - /// A handle that shuts down the RPC server. - close_handle: CloseHandle, + /// A server handle used to shuts down the RPC server. + close_handle: ServerHandle, } impl fmt::Debug for RpcServer { @@ -67,7 +67,7 @@ impl fmt::Debug for RpcServer { .field( "close_handle", // TODO: when it stabilises, use std::any::type_name_of_val(&self.close_handle) - &"CloseHandle", + &"ServerHandle", ) .finish() } @@ -76,6 +76,8 @@ impl fmt::Debug for RpcServer { /// The message to log when logging the RPC server's listen address pub const OPENED_RPC_ENDPOINT_MSG: &str = "Opened RPC endpoint at "; +type ServerTask = JoinHandle>; + impl RpcServer { /// Start a new RPC server endpoint using the supplied configs and services. /// @@ -89,7 +91,7 @@ impl RpcServer { // - put some of the configs or services in their own struct? // - replace VersionString with semver::Version, and update the tests to provide valid versions #[allow(clippy::too_many_arguments)] - pub fn spawn< + pub async fn spawn< VersionString, UserAgentString, Mempool, @@ -114,7 +116,7 @@ impl RpcServer { address_book: AddressBook, latest_chain_tip: Tip, network: Network, - ) -> (JoinHandle<()>, JoinHandle<()>, Option) + ) -> Result<(ServerTask, JoinHandle<()>), tower::BoxError> where VersionString: ToString + Clone + Send + 'static, UserAgentString: ToString + Clone + Send + 'static, @@ -149,136 +151,79 @@ impl RpcServer { SyncStatus: ChainSyncStatus + Clone + Send + Sync + 'static, AddressBook: AddressBookPeers + Clone + Send + Sync + 'static, { - if let Some(listen_addr) = config.listen_addr { - info!("Trying to open RPC endpoint at {}...", listen_addr,); - - // Create handler compatible with V1 and V2 RPC protocols - let mut io: MetaIoHandler<(), _> = - MetaIoHandler::new(Compatibility::Both, FixRpcResponseMiddleware); - + let listen_addr = config + .listen_addr + .expect("caller should make sure listen_addr is set"); + + #[cfg(feature = "getblocktemplate-rpcs")] + // Initialize the getblocktemplate rpc method handler + let get_block_template_rpc_impl = GetBlockTemplateRpcImpl::new( + &network, + mining_config.clone(), + mempool.clone(), + state.clone(), + latest_chain_tip.clone(), + block_verifier_router, + sync_status, + address_book, + ); + + // Initialize the rpc methods with the zebra version + let (rpc_impl, rpc_tx_queue_task_handle) = RpcImpl::new( + build_version.clone(), + user_agent, + network.clone(), + config.debug_force_finished_sync, #[cfg(feature = "getblocktemplate-rpcs")] - { - // Initialize the getblocktemplate rpc method handler - let get_block_template_rpc_impl = GetBlockTemplateRpcImpl::new( - &network, - mining_config.clone(), - mempool.clone(), - state.clone(), - latest_chain_tip.clone(), - block_verifier_router, - sync_status, - address_book, - ); - - io.extend_with(get_block_template_rpc_impl.to_delegate()); - } - - // Initialize the rpc methods with the zebra version - let (rpc_impl, rpc_tx_queue_task_handle) = RpcImpl::new( - build_version.clone(), - user_agent, - network.clone(), - config.debug_force_finished_sync, - #[cfg(feature = "getblocktemplate-rpcs")] - mining_config.debug_like_zcashd, - #[cfg(not(feature = "getblocktemplate-rpcs"))] - true, - mempool, - state, - latest_chain_tip, - ); - - io.extend_with(rpc_impl.to_delegate()); - - // If zero, automatically scale threads to the number of CPU cores - let mut parallel_cpu_threads = config.parallel_cpu_threads; - if parallel_cpu_threads == 0 { - parallel_cpu_threads = available_parallelism().map(usize::from).unwrap_or(1); - } - - // The server is a blocking task, which blocks on executor shutdown. - // So we need to start it in a std::thread. - // (Otherwise tokio panics on RPC port conflict, which shuts down the RPC server.) - let span = Span::current(); - let start_server = move || { - span.in_scope(|| { - let middleware = if config.enable_cookie_auth { - let cookie = Cookie::default(); - cookie::write_to_disk(&cookie, &config.cookie_dir) - .expect("Zebra must be able to write the auth cookie to the disk"); - HttpRequestMiddleware::default().with(cookie) - } else { - HttpRequestMiddleware::default() - }; - - // Use a different tokio executor from the rest of Zebra, - // so that large RPCs and any task handling bugs don't impact Zebra. - let server_instance = ServerBuilder::new(io) - .threads(parallel_cpu_threads) - // TODO: disable this security check if we see errors from lightwalletd - //.allowed_hosts(DomainsValidation::Disabled) - .request_middleware(middleware) - .start_http(&listen_addr) - .expect("Unable to start RPC server"); - - info!("{OPENED_RPC_ENDPOINT_MSG}{}", server_instance.address()); - - let close_handle = server_instance.close_handle(); - - let rpc_server_handle = RpcServer { - config, - network, - build_version: build_version.to_string(), - close_handle, - }; - - (server_instance, rpc_server_handle) - }) - }; - - // Propagate panics from the std::thread - let (server_instance, rpc_server_handle) = match std::thread::spawn(start_server).join() - { - Ok(rpc_server) => rpc_server, - Err(panic_object) => panic::resume_unwind(panic_object), - }; - - // The server is a blocking task, which blocks on executor shutdown. - // So we need to wait on it on a std::thread, inside a tokio blocking task. - // (Otherwise tokio panics when we shut down the RPC server.) - let span = Span::current(); - let wait_on_server = move || { - span.in_scope(|| { - server_instance.wait(); - - info!("Stopped RPC endpoint"); - }) - }; - - let span = Span::current(); - let rpc_server_task_handle = tokio::task::spawn_blocking(move || { - let thread_handle = std::thread::spawn(wait_on_server); - - // Propagate panics from the inner std::thread to the outer tokio blocking task - span.in_scope(|| match thread_handle.join() { - Ok(()) => (), - Err(panic_object) => panic::resume_unwind(panic_object), - }) - }); - - ( - rpc_server_task_handle, - rpc_tx_queue_task_handle, - Some(rpc_server_handle), - ) + mining_config.debug_like_zcashd, + #[cfg(not(feature = "getblocktemplate-rpcs"))] + true, + mempool, + state, + latest_chain_tip, + ); + + let http_middleware_layer = if config.enable_cookie_auth { + let cookie = Cookie::default(); + cookie::write_to_disk(&cookie, &config.cookie_dir) + .expect("Zebra must be able to write the auth cookie to the disk"); + HttpRequestMiddlewareLayer::new(Some(cookie)) } else { - // There is no RPC port, so the RPC tasks do nothing. - ( - tokio::task::spawn(futures::future::pending().in_current_span()), - tokio::task::spawn(futures::future::pending().in_current_span()), - None, - ) - } + HttpRequestMiddlewareLayer::new(None) + }; + + let http_middleware = tower::ServiceBuilder::new().layer(http_middleware_layer); + + let rpc_middleware = RpcServiceBuilder::new() + .rpc_logger(1024) + .layer_fn(FixRpcResponseMiddleware::new); + + let server_instance = Server::builder() + .http_only() + .set_http_middleware(http_middleware) + .set_rpc_middleware(rpc_middleware) + .build(listen_addr) + .await + .expect("Unable to start RPC server"); + let addr = server_instance + .local_addr() + .expect("Unable to get local address"); + info!("{OPENED_RPC_ENDPOINT_MSG}{}", addr); + + #[cfg(feature = "getblocktemplate-rpcs")] + let mut rpc_module = rpc_impl.into_rpc(); + #[cfg(not(feature = "getblocktemplate-rpcs"))] + let rpc_module = rpc_impl.into_rpc(); + #[cfg(feature = "getblocktemplate-rpcs")] + rpc_module + .merge(get_block_template_rpc_impl.into_rpc()) + .unwrap(); + + let server_task: JoinHandle> = tokio::spawn(async move { + server_instance.start(rpc_module).stopped().await; + Ok(()) + }); + Ok((server_task, rpc_tx_queue_task_handle)) } /// Shut down this RPC server, blocking the current thread. @@ -304,7 +249,7 @@ impl RpcServer { /// Shuts down this RPC server using its `close_handle`. /// /// See `shutdown_blocking()` for details. - fn shutdown_blocking_inner(close_handle: CloseHandle, config: Config) { + fn shutdown_blocking_inner(close_handle: ServerHandle, config: Config) { // The server is a blocking task, so it can't run inside a tokio thread. // See the note at wait_on_server. let span = Span::current(); @@ -320,7 +265,7 @@ impl RpcServer { } info!("Stopping RPC server"); - close_handle.clone().close(); + let _ = close_handle.stop(); debug!("Stopped RPC server"); }) }; diff --git a/zebra-rpc/src/server/error.rs b/zebra-rpc/src/server/error.rs new file mode 100644 index 00000000000..5130a16d533 --- /dev/null +++ b/zebra-rpc/src/server/error.rs @@ -0,0 +1,115 @@ +//! RPC error codes & their handling. +use jsonrpsee_types::{ErrorCode, ErrorObject, ErrorObjectOwned}; + +/// Bitcoin RPC error codes +/// +/// Drawn from . +/// +/// ## Notes +/// +/// - All explicit discriminants fit within `i64`. +#[derive(Default)] +pub enum LegacyCode { + // General application defined errors + /// `std::exception` thrown in command handling + #[default] + Misc = -1, + /// Server is in safe mode, and command is not allowed in safe mode + ForbiddenBySafeMode = -2, + /// Unexpected type was passed as parameter + Type = -3, + /// Invalid address or key + InvalidAddressOrKey = -5, + /// Ran out of memory during operation + OutOfMemory = -7, + /// Invalid, missing or duplicate parameter + InvalidParameter = -8, + /// Database error + Database = -20, + /// Error parsing or validating structure in raw format + Deserialization = -22, + /// General error during transaction or block submission + Verify = -25, + /// Transaction or block was rejected by network rules + VerifyRejected = -26, + /// Transaction already in chain + VerifyAlreadyInChain = -27, + /// Client still warming up + InWarmup = -28, + + // P2P client errors + /// Bitcoin is not connected + ClientNotConnected = -9, + /// Still downloading initial blocks + ClientInInitialDownload = -10, + /// Node is already added + ClientNodeAlreadyAdded = -23, + /// Node has not been added before + ClientNodeNotAdded = -24, + /// Node to disconnect not found in connected nodes + ClientNodeNotConnected = -29, + /// Invalid IP/Subnet + ClientInvalidIpOrSubnet = -30, +} + +impl From for ErrorCode { + fn from(code: LegacyCode) -> Self { + Self::ServerError(code as i32) + } +} + +impl From for i32 { + fn from(code: LegacyCode) -> Self { + code as i32 + } +} + +/// A trait for mapping errors to [`jsonrpsee_types::ErrorObjectOwned`]. +pub(crate) trait MapError: Sized { + /// Maps errors to [`jsonrpsee_types::ErrorObjectOwned`] with a specific error code. + fn map_error(self, code: impl Into) -> std::result::Result; + + /// Maps errors to [`jsonrpsee_types::ErrorObjectOwned`] with a [`LegacyCode::Misc`] error code. + fn map_misc_error(self) -> std::result::Result { + self.map_error(LegacyCode::Misc) + } +} + +/// A trait for conditionally converting a value into a `Result`. +pub(crate) trait OkOrError: Sized { + /// Converts the implementing type to `Result`, using an error code and + /// message if conversion is to `Err`. + fn ok_or_error( + self, + code: impl Into, + message: impl ToString, + ) -> std::result::Result; + + /// Converts the implementing type to `Result`, using a [`LegacyCode::Misc`] error code. + fn ok_or_misc_error(self, message: impl ToString) -> std::result::Result { + self.ok_or_error(LegacyCode::Misc, message) + } +} + +impl MapError for Result +where + E: ToString, +{ + fn map_error(self, code: impl Into) -> Result { + self.map_err(|error| ErrorObject::owned(code.into().code(), error.to_string(), None::<()>)) + } +} + +impl OkOrError for Option { + fn ok_or_error( + self, + code: impl Into, + message: impl ToString, + ) -> Result { + self.ok_or(ErrorObject::owned( + code.into().code(), + message.to_string(), + None::<()>, + )) + } +} diff --git a/zebra-rpc/src/server/http_request_compatibility.rs b/zebra-rpc/src/server/http_request_compatibility.rs index 89925c229b8..ebbf49c05d3 100644 --- a/zebra-rpc/src/server/http_request_compatibility.rs +++ b/zebra-rpc/src/server/http_request_compatibility.rs @@ -2,16 +2,25 @@ //! //! These fixes are applied at the HTTP level, before the RPC request is parsed. -use base64::{engine::general_purpose::URL_SAFE, Engine as _}; -use futures::TryStreamExt; -use jsonrpc_http_server::{ - hyper::{body::Bytes, header, Body, Request}, - RequestMiddleware, RequestMiddlewareAction, +use std::future::Future; + +use std::pin::Pin; + +use futures::{future, FutureExt}; +use http_body_util::BodyExt; +use hyper::{body::Bytes, header}; +use jsonrpsee::{ + core::BoxError, + server::{HttpBody, HttpRequest, HttpResponse}, }; +use jsonrpsee_types::ErrorObject; +use tower::Service; use super::cookie::Cookie; -/// HTTP [`RequestMiddleware`] with compatibility workarounds. +use base64::{engine::general_purpose::URL_SAFE, Engine as _}; + +/// HTTP [`HttpRequestMiddleware`] with compatibility workarounds. /// /// This middleware makes the following changes to HTTP requests: /// @@ -25,7 +34,7 @@ use super::cookie::Cookie; /// ### Add missing `content-type` HTTP header /// /// Some RPC clients don't include a `content-type` HTTP header. -/// But unlike web browsers, [`jsonrpc_http_server`] does not do content sniffing. +/// But unlike web browsers, [`jsonrpsee`] does not do content sniffing. /// /// If there is no `content-type` header, we assume the content is JSON, /// and let the parser error if we are incorrect. @@ -42,103 +51,30 @@ use super::cookie::Cookie; /// Any user-specified data in RPC requests is hex or base58check encoded. /// We assume lightwalletd validates data encodings before sending it on to Zebra. /// So any fixes Zebra performs won't change user-specified data. -#[derive(Clone, Debug, Default)] -pub struct HttpRequestMiddleware { +#[derive(Clone, Debug)] +pub struct HttpRequestMiddleware { + service: S, cookie: Option, } -/// A trait for updating an object, consuming it and returning the updated version. -pub trait With { - /// Updates `self` with an instance of type `T` and returns the updated version of `self`. - fn with(self, _: T) -> Self; -} - -impl With for HttpRequestMiddleware { - fn with(mut self, cookie: Cookie) -> Self { - self.cookie = Some(cookie); - self - } -} - -impl RequestMiddleware for HttpRequestMiddleware { - fn on_request(&self, mut request: Request) -> RequestMiddlewareAction { - tracing::trace!(?request, "original HTTP request"); - - // Check if the request is authenticated - if !self.check_credentials(request.headers_mut()) { - let error = jsonrpc_core::Error { - code: jsonrpc_core::ErrorCode::ServerError(401), - message: "unauthenticated method".to_string(), - data: None, - }; - return jsonrpc_http_server::Response { - code: jsonrpc_http_server::hyper::StatusCode::from_u16(401) - .expect("hard-coded status code should be valid"), - content_type: header::HeaderValue::from_static("application/json; charset=utf-8"), - content: serde_json::to_string(&jsonrpc_core::Response::from(error, None)) - .expect("hard-coded result should serialize"), - } - .into(); - } - - // Fix the request headers if needed and we can do so. - HttpRequestMiddleware::insert_or_replace_content_type_header(request.headers_mut()); - - // Fix the request body - let request = request.map(|body| { - let body = body.map_ok(|data| { - // To simplify data handling, we assume that any search strings won't be split - // across multiple `Bytes` data buffers. - // - // To simplify error handling, Zebra only supports valid UTF-8 requests, - // and uses lossy UTF-8 conversion. - // - // JSON-RPC requires all requests to be valid UTF-8. - // The lower layers should reject invalid requests with lossy changes. - // But if they accept some lossy changes, that's ok, - // because the request was non-standard anyway. - // - // We're not concerned about performance here, so we just clone the Cow - let data = String::from_utf8_lossy(data.as_ref()).to_string(); - - // Fix up the request. - let data = Self::remove_json_1_fields(data); - - Bytes::from(data) - }); - - Body::wrap_stream(body) - }); - - tracing::trace!(?request, "modified HTTP request"); - - RequestMiddlewareAction::Proceed { - // TODO: disable this security check if we see errors from lightwalletd. - should_continue_on_invalid_cors: false, - request, - } +impl HttpRequestMiddleware { + /// Create a new `HttpRequestMiddleware` with the given service and cookie. + pub fn new(service: S, cookie: Option) -> Self { + Self { service, cookie } } -} -impl HttpRequestMiddleware { - /// Remove any "jsonrpc: 1.0" fields in `data`, and return the resulting string. - pub fn remove_json_1_fields(data: String) -> String { - // Replace "jsonrpc = 1.0": - // - at the start or middle of a list, and - // - at the end of a list; - // with no spaces (lightwalletd format), and spaces after separators (example format). - // - // TODO: if we see errors from lightwalletd, make this replacement more accurate: - // - use a partial JSON fragment parser - // - combine the whole request into a single buffer, and use a JSON parser - // - use a regular expression - // - // We could also just handle the exact lightwalletd format, - // by replacing `{"jsonrpc":"1.0",` with `{`. - data.replace("\"jsonrpc\":\"1.0\",", "") - .replace("\"jsonrpc\": \"1.0\",", "") - .replace(",\"jsonrpc\":\"1.0\"", "") - .replace(", \"jsonrpc\": \"1.0\"", "") + /// Check if the request is authenticated. + pub fn check_credentials(&self, headers: &header::HeaderMap) -> bool { + self.cookie.as_ref().map_or(true, |internal_cookie| { + headers + .get(header::AUTHORIZATION) + .and_then(|auth_header| auth_header.to_str().ok()) + .and_then(|auth_header| auth_header.split_whitespace().nth(1)) + .and_then(|encoded| URL_SAFE.decode(encoded).ok()) + .and_then(|decoded| String::from_utf8(decoded).ok()) + .and_then(|request_cookie| request_cookie.split(':').nth(1).map(String::from)) + .map_or(false, |passwd| internal_cookie.authenticate(passwd)) + }) } /// Insert or replace client supplied `content-type` HTTP header to `application/json` in the following cases: @@ -182,17 +118,110 @@ impl HttpRequestMiddleware { } } - /// Check if the request is authenticated. - pub fn check_credentials(&self, headers: &header::HeaderMap) -> bool { - self.cookie.as_ref().map_or(true, |internal_cookie| { - headers - .get(header::AUTHORIZATION) - .and_then(|auth_header| auth_header.to_str().ok()) - .and_then(|auth_header| auth_header.split_whitespace().nth(1)) - .and_then(|encoded| URL_SAFE.decode(encoded).ok()) - .and_then(|decoded| String::from_utf8(decoded).ok()) - .and_then(|request_cookie| request_cookie.split(':').nth(1).map(String::from)) - .map_or(false, |passwd| internal_cookie.authenticate(passwd)) - }) + /// Remove any "jsonrpc: 1.0" fields in `data`, and return the resulting string. + pub fn remove_json_1_fields(data: String) -> String { + // Replace "jsonrpc = 1.0": + // - at the start or middle of a list, and + // - at the end of a list; + // with no spaces (lightwalletd format), and spaces after separators (example format). + // + // TODO: if we see errors from lightwalletd, make this replacement more accurate: + // - use a partial JSON fragment parser + // - combine the whole request into a single buffer, and use a JSON parser + // - use a regular expression + // + // We could also just handle the exact lightwalletd format, + // by replacing `{"jsonrpc":"1.0",` with `{"jsonrpc":"2.0`. + data.replace("\"jsonrpc\":\"1.0\",", "\"jsonrpc\":\"2.0\",") + .replace("\"jsonrpc\": \"1.0\",", "\"jsonrpc\": \"2.0\",") + .replace(",\"jsonrpc\":\"1.0\"", ",\"jsonrpc\":\"2.0\"") + .replace(", \"jsonrpc\": \"1.0\"", ", \"jsonrpc\": \"2.0\"") + } +} + +/// Implement the Layer for HttpRequestMiddleware to allow injecting the cookie +#[derive(Clone)] +pub struct HttpRequestMiddlewareLayer { + cookie: Option, +} + +impl HttpRequestMiddlewareLayer { + /// Create a new `HttpRequestMiddlewareLayer` with the given cookie. + pub fn new(cookie: Option) -> Self { + Self { cookie } + } +} + +impl tower::Layer for HttpRequestMiddlewareLayer { + type Service = HttpRequestMiddleware; + + fn layer(&self, service: S) -> Self::Service { + HttpRequestMiddleware::new(service, self.cookie.clone()) + } +} + +/// A trait for updating an object, consuming it and returning the updated version. +pub trait With { + /// Updates `self` with an instance of type `T` and returns the updated version of `self`. + fn with(self, _: T) -> Self; +} + +impl With for HttpRequestMiddleware { + fn with(mut self, cookie: Cookie) -> Self { + self.cookie = Some(cookie); + self + } +} + +impl Service> for HttpRequestMiddleware +where + S: Service + std::clone::Clone + Send + 'static, + S::Error: Into + 'static, + S::Future: Send + 'static, +{ + type Response = S::Response; + type Error = BoxError; + type Future = + Pin> + Send + 'static>>; + + fn poll_ready( + &mut self, + cx: &mut std::task::Context<'_>, + ) -> std::task::Poll> { + self.service.poll_ready(cx).map_err(Into::into) + } + + fn call(&mut self, mut request: HttpRequest) -> Self::Future { + // Check if the request is authenticated + if !self.check_credentials(request.headers_mut()) { + let error = ErrorObject::borrowed(401, "unauthenticated method", None); + // TODO: Error object is not being returned to the user but an empty response. + return future::err(BoxError::from(error)).boxed(); + } + + // Fix the request headers. + Self::insert_or_replace_content_type_header(request.headers_mut()); + + let mut service = self.service.clone(); + let (parts, body) = request.into_parts(); + + async move { + let bytes = body + .collect() + .await + .expect("Failed to collect body data") + .to_bytes(); + + let data = String::from_utf8_lossy(bytes.as_ref()).to_string(); + + // Fix JSON-RPC 1.0 requests. + let data = Self::remove_json_1_fields(data); + let body = HttpBody::from(Bytes::from(data).as_ref().to_vec()); + + let request = HttpRequest::from_parts(parts, body); + + service.call(request).await.map_err(Into::into) + } + .boxed() } } diff --git a/zebra-rpc/src/server/rpc_call_compatibility.rs b/zebra-rpc/src/server/rpc_call_compatibility.rs index c3974ac3cf8..2bd22b72924 100644 --- a/zebra-rpc/src/server/rpc_call_compatibility.rs +++ b/zebra-rpc/src/server/rpc_call_compatibility.rs @@ -3,106 +3,66 @@ //! These fixes are applied at the JSON-RPC call level, //! after the RPC request is parsed and split into calls. -use std::future::Future; - -use futures::future::{Either, FutureExt}; -use jsonrpc_core::{ - middleware::Middleware, - types::{Call, Failure, Output, Response}, - BoxFuture, ErrorCode, Metadata, MethodCall, Notification, +use jsonrpsee::{ + server::middleware::rpc::{layer::ResponseFuture, RpcService, RpcServiceT}, + MethodResponse, }; +use jsonrpsee_types::ErrorObject; -use crate::constants::{INVALID_PARAMETERS_ERROR_CODE, MAX_PARAMS_LOG_LENGTH}; - -/// JSON-RPC [`Middleware`] with compatibility workarounds. +/// JSON-RPC [`FixRpcResponseMiddleware`] with compatibility workarounds. /// /// This middleware makes the following changes to JSON-RPC calls: /// /// ## Make RPC framework response codes match `zcashd` /// -/// [`jsonrpc_core`] returns specific error codes while parsing requests: -/// +/// [`jsonrpsee_types`] returns specific error codes while parsing requests: +/// /// /// But these codes are different from `zcashd`, and some RPC clients rely on the exact code. -/// -/// ## Read-Only Functionality -/// -/// This middleware also logs unrecognized RPC requests. -pub struct FixRpcResponseMiddleware; - -impl Middleware for FixRpcResponseMiddleware { - type Future = BoxFuture>; - type CallFuture = BoxFuture>; - - fn on_call( - &self, - call: Call, - meta: M, - next: Next, - ) -> Either - where - Next: Fn(Call, M) -> NextFuture + Send + Sync, - NextFuture: Future> + Send + 'static, - { - Either::Left( - next(call.clone(), meta) - .map(|mut output| { - Self::fix_error_codes(&mut output); - output - }) - .inspect(|output| Self::log_if_error(output, call)) - .boxed(), - ) - } +/// Specifically, the [`jsonrpsee_types::error::INVALID_PARAMS_CODE`] is different: +/// +pub struct FixRpcResponseMiddleware { + service: RpcService, } impl FixRpcResponseMiddleware { - /// Replace [`jsonrpc_core`] server error codes in `output` with the `zcashd` equivalents. - fn fix_error_codes(output: &mut Option) { - if let Some(Output::Failure(Failure { ref mut error, .. })) = output { - if matches!(error.code, ErrorCode::InvalidParams) { - let original_code = error.code.clone(); - - error.code = INVALID_PARAMETERS_ERROR_CODE; - tracing::debug!("Replacing RPC error: {original_code:?} with {error}"); - } - } + /// Create a new `FixRpcResponseMiddleware` with the given `service`. + pub fn new(service: RpcService) -> Self { + Self { service } } +} - /// Obtain a description string for a received request. - /// - /// Prints out only the method name and the received parameters. - fn call_description(call: &Call) -> String { - match call { - Call::MethodCall(MethodCall { method, params, .. }) => { - let mut params = format!("{params:?}"); - if params.len() >= MAX_PARAMS_LOG_LENGTH { - params.truncate(MAX_PARAMS_LOG_LENGTH); - params.push_str("..."); - } +impl<'a> RpcServiceT<'a> for FixRpcResponseMiddleware { + type Future = ResponseFuture>; - format!(r#"method = {method:?}, params = {params}"#) - } - Call::Notification(Notification { method, params, .. }) => { - let mut params = format!("{params:?}"); - if params.len() >= MAX_PARAMS_LOG_LENGTH { - params.truncate(MAX_PARAMS_LOG_LENGTH); - params.push_str("..."); - } + fn call(&self, request: jsonrpsee::types::Request<'a>) -> Self::Future { + let service = self.service.clone(); + ResponseFuture::future(Box::pin(async move { + let response = service.call(request).await; + if response.is_error() { + let original_error_code = response + .as_error_code() + .expect("response should have an error code"); + if original_error_code == jsonrpsee_types::ErrorCode::InvalidParams.code() { + let new_error_code = crate::server::error::LegacyCode::Misc.into(); + tracing::debug!( + "Replacing RPC error: {original_error_code} with {new_error_code}" + ); + let json: serde_json::Value = + serde_json::from_str(response.into_parts().0.as_str()) + .expect("response string should be valid json"); + let id = json["id"] + .as_str() + .expect("response json should have an id") + .to_string(); - format!(r#"notification = {method:?}, params = {params}"#) + return MethodResponse::error( + jsonrpsee_types::Id::Str(id.into()), + ErrorObject::borrowed(new_error_code, "Invalid params", None), + ); + } } - Call::Invalid { .. } => "invalid request".to_owned(), - } - } - - /// Check RPC output and log any errors. - // - // TODO: do we want to ignore ErrorCode::ServerError(_), or log it at debug? - fn log_if_error(output: &Option, call: Call) { - if let Some(Output::Failure(Failure { error, .. })) = output { - let call_description = Self::call_description(&call); - tracing::info!("RPC error: {error} in call: {call_description}"); - } + response + })) } } diff --git a/zebra-rpc/src/server/tests/vectors.rs b/zebra-rpc/src/server/tests/vectors.rs index 8ffc3386a0d..bf850661a09 100644 --- a/zebra-rpc/src/server/tests/vectors.rs +++ b/zebra-rpc/src/server/tests/vectors.rs @@ -3,12 +3,8 @@ // These tests call functions which can take unit arguments if some features aren't enabled. #![allow(clippy::unit_arg)] -use std::{ - net::{Ipv4Addr, SocketAddrV4}, - time::Duration, -}; +use std::net::{Ipv4Addr, SocketAddrV4}; -use futures::FutureExt; use tower::buffer::Buffer; use zebra_chain::{ @@ -21,111 +17,71 @@ use zebra_test::mock_service::MockService; use super::super::*; -/// Test that the JSON-RPC server spawns when configured with a single thread. -#[test] -fn rpc_server_spawn_single_thread() { - rpc_server_spawn(false) -} - -/// Test that the JSON-RPC server spawns when configured with multiple threads. -#[test] -#[cfg(not(target_os = "windows"))] -fn rpc_server_spawn_parallel_threads() { - rpc_server_spawn(true) +/// Test that the JSON-RPC server spawns. +#[tokio::test] +async fn rpc_server_spawn_test() { + rpc_server_spawn().await } /// Test if the RPC server will spawn on a randomly generated port. -/// -/// Set `parallel_cpu_threads` to true to auto-configure based on the number of CPU cores. #[tracing::instrument] -fn rpc_server_spawn(parallel_cpu_threads: bool) { +async fn rpc_server_spawn() { let _init_guard = zebra_test::init(); let config = Config { listen_addr: Some(SocketAddrV4::new(Ipv4Addr::LOCALHOST, 0).into()), indexer_listen_addr: None, - parallel_cpu_threads: if parallel_cpu_threads { 2 } else { 1 }, + parallel_cpu_threads: 0, debug_force_finished_sync: false, cookie_dir: Default::default(), enable_cookie_auth: false, }; - let rt = tokio::runtime::Runtime::new().unwrap(); - - rt.block_on(async { - let mut mempool: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); - let mut block_verifier_router: MockService<_, _, _, BoxError> = - MockService::build().for_unit_tests(); - - info!("spawning RPC server..."); - - let (rpc_server_task_handle, rpc_tx_queue_task_handle, _rpc_server) = RpcServer::spawn( - config, - Default::default(), - "RPC server test", - "RPC server test", - Buffer::new(mempool.clone(), 1), - Buffer::new(state.clone(), 1), - Buffer::new(block_verifier_router.clone(), 1), - MockSyncStatus::default(), - MockAddressBookPeers::default(), - NoChainTip, - Mainnet, - ); - - info!("spawned RPC server, checking services..."); - - mempool.expect_no_requests().await; - state.expect_no_requests().await; - block_verifier_router.expect_no_requests().await; - - // The server and queue tasks should continue without errors or panics - let rpc_server_task_result = rpc_server_task_handle.now_or_never(); - assert!(rpc_server_task_result.is_none()); - - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - assert!(rpc_tx_queue_task_result.is_none()); - }); - - info!("waiting for RPC server to shut down..."); - rt.shutdown_timeout(Duration::from_secs(1)); -} - -/// Test that the JSON-RPC server spawns when configured with a single thread, -/// on an OS-assigned unallocated port. -#[test] -fn rpc_server_spawn_unallocated_port_single_thread() { - rpc_server_spawn_unallocated_port(false, false) -} - -/// Test that the JSON-RPC server spawns and shuts down when configured with a single thread, -/// on an OS-assigned unallocated port. -#[test] -fn rpc_server_spawn_unallocated_port_single_thread_shutdown() { - rpc_server_spawn_unallocated_port(false, true) + let mut mempool: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); + let mut state: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); + let mut block_verifier_router: MockService<_, _, _, BoxError> = + MockService::build().for_unit_tests(); + + info!("spawning RPC server..."); + + let _rpc_server_task_handle = RpcServer::spawn( + config, + Default::default(), + "RPC server test", + "RPC server test", + Buffer::new(mempool.clone(), 1), + Buffer::new(state.clone(), 1), + Buffer::new(block_verifier_router.clone(), 1), + MockSyncStatus::default(), + MockAddressBookPeers::default(), + NoChainTip, + Mainnet, + ); + + info!("spawned RPC server, checking services..."); + + mempool.expect_no_requests().await; + state.expect_no_requests().await; + block_verifier_router.expect_no_requests().await; } -/// Test that the JSON-RPC server spawns when configured with multiple threads, -/// on an OS-assigned unallocated port. -#[test] -fn rpc_sever_spawn_unallocated_port_parallel_threads() { - rpc_server_spawn_unallocated_port(true, false) +/// Test that the JSON-RPC server spawns on an OS-assigned unallocated port. +#[tokio::test] +async fn rpc_server_spawn_unallocated_port() { + rpc_spawn_unallocated_port(false).await } -/// Test that the JSON-RPC server spawns and shuts down when configured with multiple threads, -/// on an OS-assigned unallocated port. -#[test] -fn rpc_sever_spawn_unallocated_port_parallel_threads_shutdown() { - rpc_server_spawn_unallocated_port(true, true) +/// Test that the JSON-RPC server spawns and shuts down on an OS-assigned unallocated port. +#[tokio::test] +async fn rpc_server_spawn_unallocated_port_shutdown() { + rpc_spawn_unallocated_port(true).await } /// Test if the RPC server will spawn on an OS-assigned unallocated port. /// -/// Set `parallel_cpu_threads` to true to auto-configure based on the number of CPU cores, -/// and `do_shutdown` to true to close the server using the close handle. +/// Set `do_shutdown` to true to close the server using the close handle. #[tracing::instrument] -fn rpc_server_spawn_unallocated_port(parallel_cpu_threads: bool, do_shutdown: bool) { +async fn rpc_spawn_unallocated_port(do_shutdown: bool) { let _init_guard = zebra_test::init(); let port = zebra_test::net::random_unallocated_port(); @@ -134,300 +90,111 @@ fn rpc_server_spawn_unallocated_port(parallel_cpu_threads: bool, do_shutdown: bo let config = Config { listen_addr: Some(SocketAddrV4::new(Ipv4Addr::LOCALHOST, port).into()), indexer_listen_addr: None, - parallel_cpu_threads: if parallel_cpu_threads { 0 } else { 1 }, + parallel_cpu_threads: 0, debug_force_finished_sync: false, cookie_dir: Default::default(), enable_cookie_auth: false, }; - let rt = tokio::runtime::Runtime::new().unwrap(); - - rt.block_on(async { - let mut mempool: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); - let mut block_verifier_router: MockService<_, _, _, BoxError> = - MockService::build().for_unit_tests(); - - info!("spawning RPC server..."); - - let (rpc_server_task_handle, rpc_tx_queue_task_handle, rpc_server) = RpcServer::spawn( - config, - Default::default(), - "RPC server test", - "RPC server test", - Buffer::new(mempool.clone(), 1), - Buffer::new(state.clone(), 1), - Buffer::new(block_verifier_router.clone(), 1), - MockSyncStatus::default(), - MockAddressBookPeers::default(), - NoChainTip, - Mainnet, - ); - - info!("spawned RPC server, checking services..."); - - mempool.expect_no_requests().await; - state.expect_no_requests().await; - block_verifier_router.expect_no_requests().await; - - if do_shutdown { - rpc_server - .expect("unexpected missing RpcServer for configured RPC port") - .shutdown() - .await - .expect("unexpected panic during RpcServer shutdown"); - - // The server and queue tasks should shut down without errors or panics - let rpc_server_task_result = rpc_server_task_handle.await; - assert!( - matches!(rpc_server_task_result, Ok(())), - "unexpected server task panic during shutdown: {rpc_server_task_result:?}" - ); - - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.await; - assert!( - matches!(rpc_tx_queue_task_result, Ok(())), - "unexpected queue task panic during shutdown: {rpc_tx_queue_task_result:?}" - ); - } else { - // The server and queue tasks should continue without errors or panics - let rpc_server_task_result = rpc_server_task_handle.now_or_never(); - assert!(rpc_server_task_result.is_none()); - - let rpc_tx_queue_task_result = rpc_tx_queue_task_handle.now_or_never(); - assert!(rpc_tx_queue_task_result.is_none()); - } - }); - - info!("waiting for RPC server to shut down..."); - rt.shutdown_timeout(Duration::from_secs(1)); + let mut mempool: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); + let mut state: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); + let mut block_verifier_router: MockService<_, _, _, BoxError> = + MockService::build().for_unit_tests(); + + info!("spawning RPC server..."); + + let rpc_server_task_handle = RpcServer::spawn( + config, + Default::default(), + "RPC server test", + "RPC server test", + Buffer::new(mempool.clone(), 1), + Buffer::new(state.clone(), 1), + Buffer::new(block_verifier_router.clone(), 1), + MockSyncStatus::default(), + MockAddressBookPeers::default(), + NoChainTip, + Mainnet, + ) + .await + .expect(""); + + info!("spawned RPC server, checking services..."); + + mempool.expect_no_requests().await; + state.expect_no_requests().await; + block_verifier_router.expect_no_requests().await; + + if do_shutdown { + rpc_server_task_handle.0.abort(); + } } /// Test if the RPC server will panic correctly when there is a port conflict. /// /// This test is sometimes unreliable on Windows, and hangs on macOS. /// We believe this is a CI infrastructure issue, not a platform-specific issue. -#[test] +#[tokio::test] #[should_panic(expected = "Unable to start RPC server")] #[cfg(not(any(target_os = "windows", target_os = "macos")))] -fn rpc_server_spawn_port_conflict() { +async fn rpc_server_spawn_port_conflict() { + use std::time::Duration; let _init_guard = zebra_test::init(); let port = zebra_test::net::random_known_port(); let config = Config { listen_addr: Some(SocketAddrV4::new(Ipv4Addr::LOCALHOST, port).into()), indexer_listen_addr: None, - parallel_cpu_threads: 1, debug_force_finished_sync: false, + parallel_cpu_threads: 0, cookie_dir: Default::default(), enable_cookie_auth: false, }; - let rt = tokio::runtime::Runtime::new().unwrap(); - - let test_task_handle = rt.spawn(async { - let mut mempool: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); - let mut block_verifier_router: MockService<_, _, _, BoxError> = - MockService::build().for_unit_tests(); - - info!("spawning RPC server 1..."); - - let (_rpc_server_1_task_handle, _rpc_tx_queue_1_task_handle, _rpc_server) = - RpcServer::spawn( - config.clone(), - Default::default(), - "RPC server 1 test", - "RPC server 1 test", - Buffer::new(mempool.clone(), 1), - Buffer::new(state.clone(), 1), - Buffer::new(block_verifier_router.clone(), 1), - MockSyncStatus::default(), - MockAddressBookPeers::default(), - NoChainTip, - Mainnet, - ); - - tokio::time::sleep(Duration::from_secs(3)).await; - - info!("spawning conflicted RPC server 2..."); - - let (rpc_server_2_task_handle, _rpc_tx_queue_2_task_handle, _rpc_server) = RpcServer::spawn( - config, - Default::default(), - "RPC server 2 conflict test", - "RPC server 2 conflict test", - Buffer::new(mempool.clone(), 1), - Buffer::new(state.clone(), 1), - Buffer::new(block_verifier_router.clone(), 1), - MockSyncStatus::default(), - MockAddressBookPeers::default(), - NoChainTip, - Mainnet, - ); - - info!("spawned RPC servers, checking services..."); - - mempool.expect_no_requests().await; - state.expect_no_requests().await; - block_verifier_router.expect_no_requests().await; - - // Because there is a panic inside a multi-threaded executor, - // we can't depend on the exact behaviour of the other tasks, - // particularly across different machines and OSes. - - // The second server should panic, so its task handle should return the panic - let rpc_server_2_task_result = rpc_server_2_task_handle.await; - match rpc_server_2_task_result { - Ok(()) => panic!( - "RPC server with conflicting port should exit with an error: \ - unexpected Ok result" - ), - Err(join_error) => match join_error.try_into_panic() { - Ok(panic_object) => panic::resume_unwind(panic_object), - Err(cancelled_error) => panic!( - "RPC server with conflicting port should exit with an error: \ - unexpected JoinError: {cancelled_error:?}" - ), - }, - } - - // Ignore the queue task result - }); - - // Wait until the spawned task finishes - std::thread::sleep(Duration::from_secs(10)); - - info!("waiting for RPC server to shut down..."); - rt.shutdown_timeout(Duration::from_secs(3)); - - match test_task_handle.now_or_never() { - Some(Ok(_never)) => unreachable!("test task always panics"), - None => panic!("unexpected test task hang"), - Some(Err(join_error)) => match join_error.try_into_panic() { - Ok(panic_object) => panic::resume_unwind(panic_object), - Err(cancelled_error) => panic!( - "test task should exit with a RPC server panic: \ - unexpected non-panic JoinError: {cancelled_error:?}" - ), - }, - } -} - -/// Check if the RPC server detects a port conflict when running parallel threads. -/// -/// If this test fails, that's great! -/// We can make parallel the default, and remove the warnings in the config docs. -/// -/// This test is sometimes unreliable on Windows, and hangs on macOS. -/// We believe this is a CI infrastructure issue, not a platform-specific issue. -#[test] -#[cfg(not(any(target_os = "windows", target_os = "macos")))] -fn rpc_server_spawn_port_conflict_parallel_auto() { - let _init_guard = zebra_test::init(); - - let port = zebra_test::net::random_known_port(); - let config = Config { - listen_addr: Some(SocketAddrV4::new(Ipv4Addr::LOCALHOST, port).into()), - indexer_listen_addr: None, - parallel_cpu_threads: 2, - debug_force_finished_sync: false, - cookie_dir: Default::default(), - enable_cookie_auth: false, - }; - - let rt = tokio::runtime::Runtime::new().unwrap(); - - let test_task_handle = rt.spawn(async { - let mut mempool: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); - let mut state: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); - let mut block_verifier_router: MockService<_, _, _, BoxError> = - MockService::build().for_unit_tests(); - - info!("spawning parallel RPC server 1..."); - - let (_rpc_server_1_task_handle, _rpc_tx_queue_1_task_handle, _rpc_server) = - RpcServer::spawn( - config.clone(), - Default::default(), - "RPC server 1 test", - "RPC server 1 test", - Buffer::new(mempool.clone(), 1), - Buffer::new(state.clone(), 1), - Buffer::new(block_verifier_router.clone(), 1), - MockSyncStatus::default(), - MockAddressBookPeers::default(), - NoChainTip, - Mainnet, - ); - - tokio::time::sleep(Duration::from_secs(3)).await; - - info!("spawning parallel conflicted RPC server 2..."); - - let (rpc_server_2_task_handle, _rpc_tx_queue_2_task_handle, _rpc_server) = RpcServer::spawn( - config, - Default::default(), - "RPC server 2 conflict test", - "RPC server 2 conflict test", - Buffer::new(mempool.clone(), 1), - Buffer::new(state.clone(), 1), - Buffer::new(block_verifier_router.clone(), 1), - MockSyncStatus::default(), - MockAddressBookPeers::default(), - NoChainTip, - Mainnet, - ); - - info!("spawned RPC servers, checking services..."); - - mempool.expect_no_requests().await; - state.expect_no_requests().await; - block_verifier_router.expect_no_requests().await; - - // Because there might be a panic inside a multi-threaded executor, - // we can't depend on the exact behaviour of the other tasks, - // particularly across different machines and OSes. - - // The second server doesn't panic, but we'd like it to. - // (See the function docs for details.) - let rpc_server_2_task_result = rpc_server_2_task_handle.await; - match rpc_server_2_task_result { - Ok(()) => info!( - "Parallel RPC server with conflicting port should exit with an error: \ - but we're ok with it ignoring the conflict for now" - ), - Err(join_error) => match join_error.try_into_panic() { - Ok(panic_object) => panic::resume_unwind(panic_object), - Err(cancelled_error) => info!( - "Parallel RPC server with conflicting port should exit with an error: \ - but we're ok with it ignoring the conflict for now: \ - unexpected JoinError: {cancelled_error:?}" - ), - }, - } - - // Ignore the queue task result - }); - - // Wait until the spawned task finishes - std::thread::sleep(Duration::from_secs(10)); - - info!("waiting for parallel RPC server to shut down..."); - rt.shutdown_timeout(Duration::from_secs(3)); - - match test_task_handle.now_or_never() { - Some(Ok(())) => { - info!("parallel RPC server task successfully exited"); - } - None => panic!("unexpected test task hang"), - Some(Err(join_error)) => match join_error.try_into_panic() { - Ok(panic_object) => panic::resume_unwind(panic_object), - Err(cancelled_error) => info!( - "Parallel RPC server with conflicting port should exit with an error: \ - but we're ok with it ignoring the conflict for now: \ - unexpected JoinError: {cancelled_error:?}" - ), - }, - } + let mut mempool: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); + let mut state: MockService<_, _, _, BoxError> = MockService::build().for_unit_tests(); + let mut block_verifier_router: MockService<_, _, _, BoxError> = + MockService::build().for_unit_tests(); + + info!("spawning RPC server 1..."); + + let _rpc_server_1_task_handle = RpcServer::spawn( + config.clone(), + Default::default(), + "RPC server 1 test", + "RPC server 1 test", + Buffer::new(mempool.clone(), 1), + Buffer::new(state.clone(), 1), + Buffer::new(block_verifier_router.clone(), 1), + MockSyncStatus::default(), + MockAddressBookPeers::default(), + NoChainTip, + Mainnet, + ) + .await; + + tokio::time::sleep(Duration::from_secs(3)).await; + + info!("spawning conflicted RPC server 2..."); + + let _rpc_server_2_task_handle = RpcServer::spawn( + config, + Default::default(), + "RPC server 2 conflict test", + "RPC server 2 conflict test", + Buffer::new(mempool.clone(), 1), + Buffer::new(state.clone(), 1), + Buffer::new(block_verifier_router.clone(), 1), + MockSyncStatus::default(), + MockAddressBookPeers::default(), + NoChainTip, + Mainnet, + ) + .await; + + info!("spawned RPC servers, checking services..."); + + mempool.expect_no_requests().await; + state.expect_no_requests().await; + block_verifier_router.expect_no_requests().await; } diff --git a/zebra-rpc/src/sync.rs b/zebra-rpc/src/sync.rs index fd323ef64bb..c678f580b4a 100644 --- a/zebra-rpc/src/sync.rs +++ b/zebra-rpc/src/sync.rs @@ -21,8 +21,8 @@ use zebra_state::{ use zebra_chain::diagnostic::task::WaitForPanics; use crate::{ - constants::MISSING_BLOCK_ERROR_CODE, methods::{hex_data::HexData, GetBlockHeightAndHash}, + server, }; /// How long to wait between calls to `getbestblockheightandhash` when it: @@ -382,8 +382,11 @@ impl SyncerRpcMethods for RpcRequestClient { } Err(err) if err - .downcast_ref::() - .is_some_and(|err| err.code == MISSING_BLOCK_ERROR_CODE) => + .downcast_ref::() + .is_some_and(|err| { + let code: i32 = server::error::LegacyCode::InvalidParameter.into(); + err.code() == code + }) => { Ok(None) } diff --git a/zebra-rpc/src/tests/vectors.rs b/zebra-rpc/src/tests/vectors.rs index 84ed937d6ef..0ca221c2cf5 100644 --- a/zebra-rpc/src/tests/vectors.rs +++ b/zebra-rpc/src/tests/vectors.rs @@ -1,24 +1,31 @@ //! Fixed Zebra RPC serialization test vectors. -use crate::methods::{GetBlock, GetRawTransaction}; +use crate::methods::{GetBlock, GetRawTransaction, TransactionObject}; #[test] pub fn test_transaction_serialization() { - let expected_tx = GetRawTransaction::Raw(vec![0x42].into()); - let expected_json = r#""42""#; - let j = serde_json::to_string(&expected_tx).unwrap(); + let tx = GetRawTransaction::Raw(vec![0x42].into()); - assert_eq!(j, expected_json); + assert_eq!(serde_json::to_string(&tx).unwrap(), r#""42""#); - let expected_tx = GetRawTransaction::Object { + let tx = GetRawTransaction::Object(TransactionObject { hex: vec![0x42].into(), - height: 1, - confirmations: 0, - }; - let expected_json = r#"{"hex":"42","height":1,"confirmations":0}"#; - let j = serde_json::to_string(&expected_tx).unwrap(); + height: Some(1), + confirmations: Some(0), + }); - assert_eq!(j, expected_json); + assert_eq!( + serde_json::to_string(&tx).unwrap(), + r#"{"hex":"42","height":1,"confirmations":0}"# + ); + + let tx = GetRawTransaction::Object(TransactionObject { + hex: vec![0x42].into(), + height: None, + confirmations: None, + }); + + assert_eq!(serde_json::to_string(&tx).unwrap(), r#"{"hex":"42"}"#); } #[test] diff --git a/zebra-state/src/service/check/tests/utxo.rs b/zebra-state/src/service/check/tests/utxo.rs index acdc2d399a7..57d087c552d 100644 --- a/zebra-state/src/service/check/tests/utxo.rs +++ b/zebra-state/src/service/check/tests/utxo.rs @@ -48,7 +48,7 @@ fn accept_shielded_mature_coinbase_utxo_spend() { let ordered_utxo = transparent::OrderedUtxo::new(output, created_height, 0); let min_spend_height = Height(created_height.0 + MIN_TRANSPARENT_COINBASE_MATURITY); - let spend_restriction = transparent::CoinbaseSpendRestriction::OnlyShieldedOutputs { + let spend_restriction = transparent::CoinbaseSpendRestriction::CheckCoinbaseMaturity { spend_height: min_spend_height, }; @@ -78,7 +78,7 @@ fn reject_unshielded_coinbase_utxo_spend() { }; let ordered_utxo = transparent::OrderedUtxo::new(output, created_height, 0); - let spend_restriction = transparent::CoinbaseSpendRestriction::SomeTransparentOutputs; + let spend_restriction = transparent::CoinbaseSpendRestriction::DisallowCoinbaseSpend; let result = check::utxo::transparent_coinbase_spend(outpoint, spend_restriction, ordered_utxo.as_ref()); @@ -104,7 +104,7 @@ fn reject_immature_coinbase_utxo_spend() { let min_spend_height = Height(created_height.0 + MIN_TRANSPARENT_COINBASE_MATURITY); let spend_height = Height(min_spend_height.0 - 1); let spend_restriction = - transparent::CoinbaseSpendRestriction::OnlyShieldedOutputs { spend_height }; + transparent::CoinbaseSpendRestriction::CheckCoinbaseMaturity { spend_height }; let result = check::utxo::transparent_coinbase_spend(outpoint, spend_restriction, ordered_utxo.as_ref()); diff --git a/zebra-state/src/service/check/utxo.rs b/zebra-state/src/service/check/utxo.rs index 324efa3c035..df3981ec0b8 100644 --- a/zebra-state/src/service/check/utxo.rs +++ b/zebra-state/src/service/check/utxo.rs @@ -72,8 +72,10 @@ pub fn transparent_spend( // We don't want to use UTXOs from invalid pending blocks, // so we check transparent coinbase maturity and shielding // using known valid UTXOs during non-finalized chain validation. - let spend_restriction = - transaction.coinbase_spend_restriction(semantically_verified.height); + let spend_restriction = transaction.coinbase_spend_restriction( + &finalized_state.network(), + semantically_verified.height, + ); transparent_coinbase_spend(spend, spend_restriction, utxo.as_ref())?; // We don't delete the UTXOs until the block is committed, @@ -195,7 +197,7 @@ pub fn transparent_coinbase_spend( } match spend_restriction { - OnlyShieldedOutputs { spend_height } => { + CheckCoinbaseMaturity { spend_height } => { let min_spend_height = utxo.height + MIN_TRANSPARENT_COINBASE_MATURITY.into(); let min_spend_height = min_spend_height.expect("valid UTXOs have coinbase heights far below Height::MAX"); @@ -210,7 +212,7 @@ pub fn transparent_coinbase_spend( }) } } - SomeTransparentOutputs => Err(UnshieldedTransparentCoinbaseSpend { outpoint }), + DisallowCoinbaseSpend => Err(UnshieldedTransparentCoinbaseSpend { outpoint }), } } diff --git a/zebrad/Cargo.toml b/zebrad/Cargo.toml index cb3e417d0cf..9cf2d1e4095 100644 --- a/zebrad/Cargo.toml +++ b/zebrad/Cargo.toml @@ -254,7 +254,7 @@ tonic-build = { version = "0.12.3", optional = true } abscissa_core = { version = "0.7.0", features = ["testing"] } hex = "0.4.3" hex-literal = "0.4.1" -jsonrpc-core = "18.0.0" +jsonrpsee-types = "0.24.7" once_cell = "1.20.2" regex = "1.11.0" insta = { version = "1.41.1", features = ["json"] } diff --git a/zebrad/src/commands/start.rs b/zebrad/src/commands/start.rs index 2f8a1563b8a..ab06e546fc8 100644 --- a/zebrad/src/commands/start.rs +++ b/zebrad/src/commands/start.rs @@ -243,20 +243,31 @@ impl StartCmd { } // Launch RPC server - info!("spawning RPC server"); - let (rpc_task_handle, rpc_tx_queue_task_handle, rpc_server) = RpcServer::spawn( - config.rpc.clone(), - config.mining.clone(), - build_version(), - user_agent(), - mempool.clone(), - read_only_state_service.clone(), - block_verifier_router.clone(), - sync_status.clone(), - address_book.clone(), - latest_chain_tip.clone(), - config.network.network.clone(), - ); + let (rpc_task_handle, mut rpc_tx_queue_task_handle) = + if let Some(listen_addr) = config.rpc.listen_addr { + info!("spawning RPC server"); + info!("Trying to open RPC endpoint at {}...", listen_addr,); + let rpc_task_handle = RpcServer::spawn( + config.rpc.clone(), + config.mining.clone(), + build_version(), + user_agent(), + mempool.clone(), + read_only_state_service.clone(), + block_verifier_router.clone(), + sync_status.clone(), + address_book.clone(), + latest_chain_tip.clone(), + config.network.network.clone(), + ); + rpc_task_handle.await.unwrap() + } else { + warn!("configure an listen_addr to start the RPC server"); + ( + tokio::spawn(std::future::pending().in_current_span()), + tokio::spawn(std::future::pending().in_current_span()), + ) + }; // TODO: Add a shutdown signal and start the server with `serve_with_incoming_shutdown()` if // any related unit tests sometimes crash with memory errors @@ -399,7 +410,6 @@ impl StartCmd { // ongoing tasks pin!(rpc_task_handle); pin!(indexer_rpc_task_handle); - pin!(rpc_tx_queue_task_handle); pin!(syncer_task_handle); pin!(block_gossip_task_handle); pin!(mempool_crawler_task_handle); @@ -425,17 +435,10 @@ impl StartCmd { let mut exit_when_task_finishes = true; let result = select! { - rpc_result = &mut rpc_task_handle => { - rpc_result + rpc_join_result = &mut rpc_task_handle => { + let rpc_server_result = rpc_join_result .expect("unexpected panic in the rpc task"); - info!("rpc task exited"); - Ok(()) - } - - indexer_rpc_join_result = &mut indexer_rpc_task_handle => { - let indexer_rpc_server_result = indexer_rpc_join_result - .expect("unexpected panic in the rpc task"); - info!(?indexer_rpc_server_result, "indexer rpc task exited"); + info!(?rpc_server_result, "rpc task exited"); Ok(()) } @@ -446,6 +449,13 @@ impl StartCmd { Ok(()) } + indexer_rpc_join_result = &mut indexer_rpc_task_handle => { + let indexer_rpc_server_result = indexer_rpc_join_result + .expect("unexpected panic in the indexer task"); + info!(?indexer_rpc_server_result, "indexer rpc task exited"); + Ok(()) + } + sync_result = &mut syncer_task_handle => sync_result .expect("unexpected panic in the syncer task") .map(|_| info!("syncer task exited")), @@ -536,15 +546,6 @@ impl StartCmd { state_checkpoint_verify_handle.abort(); old_databases_task_handle.abort(); - // Wait until the RPC server shuts down. - // This can take around 150 seconds. - // - // Without this shutdown, Zebra's RPC unit tests sometimes crashed with memory errors. - if let Some(rpc_server) = rpc_server { - info!("waiting for RPC server to shut down"); - rpc_server.shutdown_blocking(); - } - info!("exiting Zebra: all tasks have been asked to stop, waiting for remaining tasks to finish"); exit_status diff --git a/zebrad/src/components/mempool.rs b/zebrad/src/components/mempool.rs index 6986f601e9c..0d76b778d87 100644 --- a/zebrad/src/components/mempool.rs +++ b/zebrad/src/components/mempool.rs @@ -737,6 +737,24 @@ impl Service for Mempool { async move { Ok(Response::Transactions(res)) }.boxed() } + Request::TransactionWithDepsByMinedId(tx_id) => { + trace!(?req, "got mempool request"); + + let res = if let Some((transaction, dependencies)) = + storage.transaction_with_deps(tx_id) + { + Ok(Response::TransactionWithDeps { + transaction, + dependencies, + }) + } else { + Err("transaction not found in mempool".into()) + }; + + trace!(?req, ?res, "answered mempool request"); + + async move { res }.boxed() + } Request::AwaitOutput(outpoint) => { trace!(?req, "got mempool request"); @@ -832,7 +850,7 @@ impl Service for Mempool { Request::TransactionsById(_) => Response::Transactions(Default::default()), Request::TransactionsByMinedId(_) => Response::Transactions(Default::default()), - Request::AwaitOutput(_) => { + Request::TransactionWithDepsByMinedId(_) | Request::AwaitOutput(_) => { return async move { Err("mempool is not active: wait for Zebra to sync to the tip".into()) } diff --git a/zebrad/src/components/mempool/storage.rs b/zebrad/src/components/mempool/storage.rs index be7cbc9593f..cee0845ba2b 100644 --- a/zebrad/src/components/mempool/storage.rs +++ b/zebrad/src/components/mempool/storage.rs @@ -513,6 +513,23 @@ impl Storage { .map(|(_, tx)| &tx.transaction) } + /// Returns a transaction and the transaction ids of its dependencies, if it is in the verified set. + pub fn transaction_with_deps( + &self, + tx_id: transaction::Hash, + ) -> Option<(VerifiedUnminedTx, HashSet)> { + let tx = self.verified.transactions().get(&tx_id).cloned()?; + let deps = self + .verified + .transaction_dependencies() + .dependencies() + .get(&tx_id) + .cloned() + .unwrap_or_default(); + + Some((tx, deps)) + } + /// Returns `true` if a transaction exactly matching an [`UnminedTxId`] is in /// the mempool. /// diff --git a/zebrad/src/components/miner.rs b/zebrad/src/components/miner.rs index cb32cc91981..ee4960a5d03 100644 --- a/zebrad/src/components/miner.rs +++ b/zebrad/src/components/miner.rs @@ -35,7 +35,7 @@ use zebra_rpc::{ GetBlockTemplateCapability::*, GetBlockTemplateRequestMode::*, }, hex_data::HexData, - GetBlockTemplateRpc, GetBlockTemplateRpcImpl, + GetBlockTemplateRpcImpl, GetBlockTemplateRpcServer, }, }; use zebra_state::WatchReceiver; diff --git a/zebrad/tests/acceptance.rs b/zebrad/tests/acceptance.rs index 3dfc959eb58..ef2de55dc83 100644 --- a/zebrad/tests/acceptance.rs +++ b/zebrad/tests/acceptance.rs @@ -3270,7 +3270,7 @@ async fn nu6_funding_streams_and_coinbase_balance() -> Result<()> { types::submit_block, }, hex_data::HexData, - GetBlockTemplateRpc, GetBlockTemplateRpcImpl, + GetBlockTemplateRpcImpl, GetBlockTemplateRpcServer, }; use zebra_test::mock_service::MockService; let _init_guard = zebra_test::init(); diff --git a/zebrad/tests/common/regtest.rs b/zebrad/tests/common/regtest.rs index bf1cba697de..efd3c08875b 100644 --- a/zebrad/tests/common/regtest.rs +++ b/zebrad/tests/common/regtest.rs @@ -17,7 +17,6 @@ use zebra_chain::{ }; use zebra_node_services::rpc_client::RpcRequestClient; use zebra_rpc::{ - constants::MISSING_BLOCK_ERROR_CODE, methods::{ get_block_template_rpcs::{ get_block_template::{ @@ -27,7 +26,7 @@ use zebra_rpc::{ }, hex_data::HexData, }, - server::OPENED_RPC_ENDPOINT_MSG, + server::{self, OPENED_RPC_ENDPOINT_MSG}, }; use zebra_test::args; @@ -162,8 +161,11 @@ impl MiningRpcMethods for RpcRequestClient { } Err(err) if err - .downcast_ref::() - .is_some_and(|err| err.code == MISSING_BLOCK_ERROR_CODE) => + .downcast_ref::() + .is_some_and(|err| { + let error: i32 = server::error::LegacyCode::InvalidParameter.into(); + err.code() == error + }) => { Ok(None) } diff --git a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_0_1.snap b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_0_1.snap index 9e830f19e61..d277043f701 100644 --- a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_0_1.snap +++ b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_0_1.snap @@ -4,6 +4,7 @@ expression: parsed --- { "jsonrpc": "2.0", + "id": 123, "result": { "pool": "orchard", "start_index": 0, @@ -13,6 +14,5 @@ expression: parsed "end_height": 1707429 } ] - }, - "id": 123 + } } diff --git a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_338_1.snap b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_338_1.snap index bcaa36d61fd..c683839781e 100644 --- a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_338_1.snap +++ b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_338_1.snap @@ -4,6 +4,7 @@ expression: parsed --- { "jsonrpc": "2.0", + "id": 123, "result": { "pool": "orchard", "start_index": 338, @@ -13,6 +14,5 @@ expression: parsed "end_height": 1888929 } ] - }, - "id": 123 + } } diff --git a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_585_1.snap b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_585_1.snap index 945af42ca5f..ec880d7df6f 100644 --- a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_585_1.snap +++ b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_orchard_585_1.snap @@ -4,6 +4,7 @@ expression: parsed --- { "jsonrpc": "2.0", + "id": 123, "result": { "pool": "orchard", "start_index": 585, @@ -13,6 +14,5 @@ expression: parsed "end_height": 2000126 } ] - }, - "id": 123 + } } diff --git a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_0_1.snap b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_0_1.snap index 2cf43dd6098..08f8744fadd 100644 --- a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_0_1.snap +++ b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_0_1.snap @@ -4,6 +4,7 @@ expression: parsed --- { "jsonrpc": "2.0", + "id": 123, "result": { "pool": "sapling", "start_index": 0, @@ -13,6 +14,5 @@ expression: parsed "end_height": 558822 } ] - }, - "id": 123 + } } diff --git a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_0_11.snap b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_0_11.snap index d709a53f0c6..f76f202706c 100644 --- a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_0_11.snap +++ b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_0_11.snap @@ -4,6 +4,7 @@ expression: parsed --- { "jsonrpc": "2.0", + "id": 123, "result": { "pool": "sapling", "start_index": 0, @@ -53,6 +54,5 @@ expression: parsed "end_height": 1363036 } ] - }, - "id": 123 + } } diff --git a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_1090_6.snap b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_1090_6.snap index ad9e68b1620..0274c501497 100644 --- a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_1090_6.snap +++ b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_1090_6.snap @@ -4,6 +4,7 @@ expression: parsed --- { "jsonrpc": "2.0", + "id": 123, "result": { "pool": "sapling", "start_index": 1090, @@ -33,6 +34,5 @@ expression: parsed "end_height": 2056616 } ] - }, - "id": 123 + } } diff --git a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_17_1.snap b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_17_1.snap index 8e0ddc1fb67..d2c983d78a8 100644 --- a/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_17_1.snap +++ b/zebrad/tests/snapshots/z_getsubtreesbyindex_mainnet_sapling_17_1.snap @@ -4,6 +4,7 @@ expression: parsed --- { "jsonrpc": "2.0", + "id": 123, "result": { "pool": "sapling", "start_index": 17, @@ -13,6 +14,5 @@ expression: parsed "end_height": 1703171 } ] - }, - "id": 123 + } }