From 73fe9643689affcb487df77365b73a93b1f448b2 Mon Sep 17 00:00:00 2001 From: Zeljko Mihaljcic <7150613+zehiko@users.noreply.github.com> Date: Thu, 10 Oct 2024 12:16:51 +0200 Subject: [PATCH] Add definitions for remote store gRPC API as well as the logic for building rust types from the proto definition We introduce re_remote_store_types with proto definition files for V0 remote store gRPC API. We also introduce re_remote_store_tpyes_builder that heaveily relies on tonic_build to build the actual rust code from protobuf files. --- ARCHITECTURE.md | 30 +- Cargo.lock | 533 ++++++++++-- Cargo.toml | 11 +- crates/build/README.md | 13 +- .../re_remote_store_types_builder/Cargo.toml | 26 + .../re_remote_store_types_builder/README.md | 12 + .../src/bin/build_re_remote_store_types.rs | 38 + .../re_remote_store_types_builder/src/lib.rs | 26 + crates/store/re_remote_store_types/Cargo.toml | 23 + crates/store/re_remote_store_types/README.md | 14 + .../proto/rerun/v0/common.proto | 117 +++ .../proto/rerun/v0/remote_store.proto | 91 +++ crates/store/re_remote_store_types/src/lib.rs | 217 +++++ .../src/v0/rerun.remote_store.v0.rs | 760 ++++++++++++++++++ crates/store/re_ws_comms/src/client.rs | 3 +- deny.toml | 2 +- pixi.toml | 3 + scripts/lint.py | 1 + 18 files changed, 1846 insertions(+), 74 deletions(-) create mode 100644 crates/build/re_remote_store_types_builder/Cargo.toml create mode 100644 crates/build/re_remote_store_types_builder/README.md create mode 100644 crates/build/re_remote_store_types_builder/src/bin/build_re_remote_store_types.rs create mode 100644 crates/build/re_remote_store_types_builder/src/lib.rs create mode 100644 crates/store/re_remote_store_types/Cargo.toml create mode 100644 crates/store/re_remote_store_types/README.md create mode 100644 crates/store/re_remote_store_types/proto/rerun/v0/common.proto create mode 100644 crates/store/re_remote_store_types/proto/rerun/v0/remote_store.proto create mode 100644 crates/store/re_remote_store_types/src/lib.rs create mode 100644 crates/store/re_remote_store_types/src/v0/rerun.remote_store.v0.rs diff --git a/ARCHITECTURE.md b/ARCHITECTURE.md index 376d4fb5167b..cdd9347e7980 100644 --- a/ARCHITECTURE.md +++ b/ARCHITECTURE.md @@ -159,14 +159,15 @@ Update instructions: ### Application-level store -| Crate | Description | -|----------------------|--------------------------------------------------------------------------| -| re_entity_db | In-memory storage of Rerun entities | -| re_query | Querying data in the re_chunk_store | -| re_dataframe | The Rerun public data APIs. | -| re_types | The built-in Rerun data types, component types, and archetypes. | -| re_types_blueprint | The core traits and types that power Rerun's Blueprint sub-system. | -| re_log_encoding | Helpers for encoding and transporting Rerun log messages | +| Crate | Description | +|-----------------------|--------------------------------------------------------------------------| +| re_entity_db | In-memory storage of Rerun entities | +| re_query | Querying data in the re_chunk_store | +| re_dataframe | The Rerun public data APIs. | +| re_remote_store_types | Rerun remote store gRPC API types | +| re_types | The built-in Rerun data types, component types, and archetypes. | +| re_types_blueprint | The core traits and types that power Rerun's Blueprint sub-system. | +| re_log_encoding | Helpers for encoding and transporting Rerun log messages | ### Low-level store @@ -193,12 +194,13 @@ Update instructions: ### Build support -| Crate | Description | -|----------------------------|------------------------------------------------------------------| -| re_build_info | Information about the build. Use together with re_build_tools | -| re_build_tools | build.rs helpers for generating build info | -| re_types_builder | Generates code for Rerun's SDKs from flatbuffers definitions. | -| re_dev_tools | Various tools for Rerun development. Each tool has a subcommand. | +| Crate | Description | +|-------------------------------|------------------------------------------------------------------| +| re_build_info | Information about the build. Use together with re_build_tools | +| re_build_tools | build.rs helpers for generating build info | +| re_remote_store_types_builder | Generates code for Rerun remote store gRPC API | +| re_types_builder | Generates code for Rerun's SDKs from flatbuffers definitions. | +| re_dev_tools | Various tools for Rerun development. Each tool has a subcommand. | ### Utilities diff --git a/Cargo.lock b/Cargo.lock index 6aff8cbde065..82aeef2e1bbf 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -630,7 +630,7 @@ dependencies = [ "polling 2.8.0", "rustix 0.37.27", "slab", - "socket2", + "socket2 0.4.9", "waker-fn", ] @@ -731,6 +731,28 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "async-stream" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b5a71a6f37880a80d1d7f19efd781e4b5de42c88f0722cc13bcb6cc2cfe8476" +dependencies = [ + "async-stream-impl", + "futures-core", + "pin-project-lite", +] + +[[package]] +name = "async-stream-impl" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c7c24de15d275a1ecfd47a380fb4d5ec9bfe0933f309ed5e705b775596a3574d" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.48", +] + [[package]] name = "async-task" version = "4.4.0" @@ -851,6 +873,53 @@ dependencies = [ "thiserror", ] +[[package]] +name = "axum" +version = "0.7.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "504e3947307ac8326a5437504c517c4b56716c9d98fac0028c2acc7ca47d70ae" +dependencies = [ + "async-trait", + "axum-core", + "bytes", + "futures-util", + "http", + "http-body", + "http-body-util", + "itoa", + "matchit", + "memchr", + "mime", + "percent-encoding", + "pin-project-lite", + "rustversion", + "serde", + "sync_wrapper 1.0.1", + "tower 0.5.1", + "tower-layer", + "tower-service", +] + +[[package]] +name = "axum-core" +version = "0.4.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09f2bd6146b97ae3359fa0cc6d6b376d9539582c7b4220f041a33ec24c226199" +dependencies = [ + "async-trait", + "bytes", + "futures-util", + "http", + "http-body", + "http-body-util", + "mime", + "pin-project-lite", + "rustversion", + "sync_wrapper 1.0.1", + "tower-layer", + "tower-service", +] + [[package]] name = "az" version = "1.2.1" @@ -878,12 +947,6 @@ version = "0.13.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9e1b586273c5702936fe7b7d6896644d8be71e6314cfe09d3167c95f712589e8" -[[package]] -name = "base64" -version = "0.21.7" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9d297deb1925b89f2ccc13d7635fa0714f12c87adce1c75356b39ca9b7178567" - [[package]] name = "base64" version = "0.22.1" @@ -1025,9 +1088,9 @@ checksum = "14c189c53d098945499cdfa7ecc63567cf3886b3332b312a5b4585d8d3a6a610" [[package]] name = "bytes" -version = "1.4.0" +version = "1.7.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "89b2fd2a0dcf38d7971e2194b6b6eebab45ae01067456a7fd93d5547a61b70be" +checksum = "428d9aa8fbc0670b7b8d6030a7fadd0f86151cae55e4dbbece15f3780a3dfaf3" [[package]] name = "calloop" @@ -2275,8 +2338,7 @@ dependencies = [ [[package]] name = "ewebsock" version = "0.6.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1bbed098b2bf9abcfe50eeaa01ae77a2a1da931bdcd83d23fcd7b8f941cd52c9" +source = "git+https://github.com/rerun-io/ewebsock?rev=1a9e78aa18390511476f9e2c0453502f11a7aa78#1a9e78aa18390511476f9e2c0453502f11a7aa78" dependencies = [ "document-features", "js-sys", @@ -2725,6 +2787,25 @@ dependencies = [ "bitflags 2.6.0", ] +[[package]] +name = "h2" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "524e8ac6999421f49a846c2d4411f337e53497d8ec55d67753beffa43c5d9205" +dependencies = [ + "atomic-waker", + "bytes", + "fnv", + "futures-core", + "futures-sink", + "http", + "indexmap 2.1.0", + "slab", + "tokio", + "tokio-util", + "tracing", +] + [[package]] name = "half" version = "1.8.2" @@ -2844,15 +2925,38 @@ dependencies = [ [[package]] name = "http" -version = "0.2.9" +version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bd6effc99afb63425aff9b05836f029929e345a6148a14b7ecd5ab67af944482" +checksum = "21b9ddb458710bc376481b842f5da65cdf31522de232c1ca8146abce2a358258" dependencies = [ "bytes", "fnv", "itoa", ] +[[package]] +name = "http-body" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184" +dependencies = [ + "bytes", + "http", +] + +[[package]] +name = "http-body-util" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "793429d76616a256bcb62c2a2ec2bed781c8307e797e2598c50010f2bee2544f" +dependencies = [ + "bytes", + "futures-util", + "http", + "http-body", + "pin-project-lite", +] + [[package]] name = "httparse" version = "1.8.0" @@ -2871,6 +2975,59 @@ version = "2.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4" +[[package]] +name = "hyper" +version = "1.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "50dfd22e0e76d0f662d429a5f80fcaf3855009297eab6a0a9f8543834744ba05" +dependencies = [ + "bytes", + "futures-channel", + "futures-util", + "h2", + "http", + "http-body", + "httparse", + "httpdate", + "itoa", + "pin-project-lite", + "smallvec", + "tokio", + "want", +] + +[[package]] +name = "hyper-timeout" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3203a961e5c83b6f5498933e78b6b263e208c197b63e9c6c53cc82ffd3f63793" +dependencies = [ + "hyper", + "hyper-util", + "pin-project-lite", + "tokio", + "tower-service", +] + +[[package]] +name = "hyper-util" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "41296eb09f183ac68eec06e03cdbea2e759633d4067b2f6552fc2e009bcad08b" +dependencies = [ + "bytes", + "futures-channel", + "futures-util", + "http", + "http-body", + "hyper", + "pin-project-lite", + "socket2 0.5.7", + "tokio", + "tower-service", + "tracing", +] + [[package]] name = "iana-time-zone" version = "0.1.60" @@ -3392,6 +3549,12 @@ dependencies = [ "libc", ] +[[package]] +name = "matchit" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e7465ac9959cc2b1404e8e2367b43684a6d13790fe23056cc8c6c5a6b7bcb94" + [[package]] name = "matrixmultiply" version = "0.3.7" @@ -4094,8 +4257,8 @@ dependencies = [ "anyhow", "clap", "glam", - "prost", - "prost-build", + "prost 0.12.6", + "prost-build 0.12.6", "protoc-prebuilt", "re_build_tools", "rerun", @@ -4498,7 +4661,17 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "deb1435c188b76130da55f17a466d252ff7b1418b2ad3e037d127b94e3411f29" dependencies = [ "bytes", - "prost-derive", + "prost-derive 0.12.6", +] + +[[package]] +name = "prost" +version = "0.13.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b0487d90e047de87f984913713b85c601c05609aad5b0df4b4573fbf69aa13f" +dependencies = [ + "bytes", + "prost-derive 0.13.3", ] [[package]] @@ -4515,8 +4688,29 @@ dependencies = [ "once_cell", "petgraph", "prettyplease", - "prost", - "prost-types", + "prost 0.12.6", + "prost-types 0.12.6", + "regex", + "syn 2.0.48", + "tempfile", +] + +[[package]] +name = "prost-build" +version = "0.13.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c1318b19085f08681016926435853bbf7858f9c082d0999b80550ff5d9abe15" +dependencies = [ + "bytes", + "heck 0.4.1", + "itertools 0.13.0", + "log", + "multimap", + "once_cell", + "petgraph", + "prettyplease", + "prost 0.13.3", + "prost-types 0.13.3", "regex", "syn 2.0.48", "tempfile", @@ -4535,13 +4729,35 @@ dependencies = [ "syn 2.0.48", ] +[[package]] +name = "prost-derive" +version = "0.13.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e9552f850d5f0964a4e4d0bf306459ac29323ddfbae05e35a7c0d35cb0803cc5" +dependencies = [ + "anyhow", + "itertools 0.13.0", + "proc-macro2", + "quote", + "syn 2.0.48", +] + [[package]] name = "prost-types" version = "0.12.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9091c90b0a32608e984ff2fa4091273cbdd755d54935c51d520887f4a1dbd5b0" dependencies = [ - "prost", + "prost 0.12.6", +] + +[[package]] +name = "prost-types" +version = "0.13.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4759aa0d3a6232fb8dbdb97b61de2c20047c68aca932c7ed76da9d788508d670" +dependencies = [ + "prost 0.13.3", ] [[package]] @@ -5371,6 +5587,31 @@ dependencies = [ "thiserror", ] +[[package]] +name = "re_remote_store_types" +version = "0.19.0-alpha.1+dev" +dependencies = [ + "prost 0.13.3", + "re_dataframe", + "re_log_types", + "thiserror", + "tonic", + "tonic-build", +] + +[[package]] +name = "re_remote_store_types_builder" +version = "0.19.0-alpha.1+dev" +dependencies = [ + "camino", + "re_dataframe", + "re_log", + "re_log_types", + "re_tracing", + "tonic", + "tonic-build", +] + [[package]] name = "re_renderer" version = "0.19.0-alpha.1+dev" @@ -6682,23 +6923,33 @@ dependencies = [ [[package]] name = "rustls" -version = "0.21.11" +version = "0.23.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7fecbfb7b1444f477b345853b1fce097a2c6fb637b2bfb87e6bc5db0f043fae4" +checksum = "415d9944693cb90382053259f89fbb077ea730ad7273047ec63b19bc9b160ba8" dependencies = [ "log", + "once_cell", "ring", + "rustls-pki-types", "rustls-webpki", - "sct", + "subtle", + "zeroize", ] +[[package]] +name = "rustls-pki-types" +version = "1.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e696e35370c65c9c541198af4543ccd580cf17fc25d8e05c5a242b202488c55" + [[package]] name = "rustls-webpki" -version = "0.101.7" +version = "0.102.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8b6275d1ee7a1cd780b64aca7726599a1dbc893b1e64144529e55c3c2f745765" +checksum = "64ca1bc8749bd4cf37b5ce386cc146580777b4e8572c7b97baf22c83f444bee9" dependencies = [ "ring", + "rustls-pki-types", "untrusted", ] @@ -6735,16 +6986,6 @@ version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d29ab0c6d3fc0ee92fe66e2d99f700eab17a8d57d1c1d3b748380fb20baa78cd" -[[package]] -name = "sct" -version = "0.7.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "da046153aa2352493d6cb7da4b6e5c0c057d8a1d0a9aa8560baffdd945acd414" -dependencies = [ - "ring", - "untrusted", -] - [[package]] name = "semver" version = "1.0.17" @@ -6959,9 +7200,9 @@ dependencies = [ [[package]] name = "smallvec" -version = "1.10.0" +version = "1.13.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a507befe795404456341dfab10cef66ead4c041f62b8b11bbb92bffe5d0953e0" +checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" dependencies = [ "serde", ] @@ -7034,6 +7275,16 @@ dependencies = [ "winapi", ] +[[package]] +name = "socket2" +version = "0.5.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce305eb0b4296696835b71df73eb912e0f1ffd2556a501fcede6e0c50349191c" +dependencies = [ + "libc", + "windows-sys 0.52.0", +] + [[package]] name = "spawn_viewer" version = "0.19.0-alpha.1+dev" @@ -7109,6 +7360,12 @@ version = "0.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "fa7986063f7c0ab374407e586d7048a3d5aac94f103f751088bf398e07cd5400" +[[package]] +name = "subtle" +version = "2.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292" + [[package]] name = "syn" version = "1.0.109" @@ -7131,6 +7388,18 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "sync_wrapper" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2047c6ded9c721764247e62cd3b03c09ffc529b2ba5b10ec482ae507a4a70160" + +[[package]] +name = "sync_wrapper" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a7065abeca94b6a8a577f9bd45aa0867a2238b74e8eb67cf10d492bc39351394" + [[package]] name = "sysinfo" version = "0.30.2" @@ -7389,6 +7658,57 @@ dependencies = [ "ahash", ] +[[package]] +name = "tokio" +version = "1.38.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eb2caba9f80616f438e09748d5acda951967e1ea58508ef53d9c6402485a46df" +dependencies = [ + "backtrace", + "bytes", + "libc", + "mio", + "pin-project-lite", + "socket2 0.5.7", + "tokio-macros", + "windows-sys 0.48.0", +] + +[[package]] +name = "tokio-macros" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f5ae998a069d4b5aba8ee9dad856af7d520c3699e6159b185c2acd48155d39a" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.48", +] + +[[package]] +name = "tokio-stream" +version = "0.1.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4f4e6ce100d0eb49a2734f8c0812bcd324cf357d21810932c5df6b96ef2b86f1" +dependencies = [ + "futures-core", + "pin-project-lite", + "tokio", +] + +[[package]] +name = "tokio-util" +version = "0.7.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61e7c3654c13bcd040d4a03abee2c75b1d14a37b423cf5a813ceae1cc903ec6a" +dependencies = [ + "bytes", + "futures-core", + "futures-sink", + "pin-project-lite", + "tokio", +] + [[package]] name = "toml" version = "0.8.19" @@ -7435,6 +7755,96 @@ dependencies = [ "winnow 0.6.18", ] +[[package]] +name = "tonic" +version = "0.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "877c5b330756d856ffcc4553ab34a5684481ade925ecc54bcd1bf02b1d0d4d52" +dependencies = [ + "async-stream", + "async-trait", + "axum", + "base64 0.22.1", + "bytes", + "h2", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-timeout", + "hyper-util", + "percent-encoding", + "pin-project", + "prost 0.13.3", + "socket2 0.5.7", + "tokio", + "tokio-stream", + "tower 0.4.13", + "tower-layer", + "tower-service", + "tracing", +] + +[[package]] +name = "tonic-build" +version = "0.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9557ce109ea773b399c9b9e5dca39294110b74f1f342cb347a80d1fce8c26a11" +dependencies = [ + "prettyplease", + "proc-macro2", + "prost-build 0.13.3", + "prost-types 0.13.3", + "quote", + "syn 2.0.48", +] + +[[package]] +name = "tower" +version = "0.4.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8fa9be0de6cf49e536ce1851f987bd21a43b771b09473c3549a6c853db37c1c" +dependencies = [ + "futures-core", + "futures-util", + "indexmap 1.9.3", + "pin-project", + "pin-project-lite", + "rand", + "slab", + "tokio", + "tokio-util", + "tower-layer", + "tower-service", + "tracing", +] + +[[package]] +name = "tower" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2873938d487c3cfb9aed7546dc9f2711d867c9f90c46b889989a2cb84eba6b4f" +dependencies = [ + "futures-core", + "futures-util", + "pin-project-lite", + "sync_wrapper 0.1.2", + "tower-layer", + "tower-service", +] + +[[package]] +name = "tower-layer" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "121c2a6cda46980bb0fcd1647ffaf6cd3fc79a013de288782836f6df9c48780e" + +[[package]] +name = "tower-service" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8df9b6e13f2d32c91b9bd719c00d1958837bc7dec474d94952798cc8e69eeec3" + [[package]] name = "tracing" version = "0.1.40" @@ -7467,6 +7877,12 @@ dependencies = [ "once_cell", ] +[[package]] +name = "try-lock" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b" + [[package]] name = "ttf-parser" version = "0.19.0" @@ -7475,9 +7891,9 @@ checksum = "44dcf002ae3b32cd25400d6df128c5babec3927cd1eb7ce813cfff20eb6c3746" [[package]] name = "tungstenite" -version = "0.20.1" +version = "0.23.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9e3dac10fd62eaf6617d3a904ae222845979aec67c615d1c842b4002c7666fb9" +checksum = "6e2e2ce1e47ed2994fd43b04c8f618008d4cabdd5ee34027cf14f9d918edd9c8" dependencies = [ "byteorder", "bytes", @@ -7487,11 +7903,11 @@ dependencies = [ "log", "rand", "rustls", + "rustls-pki-types", "sha1", "thiserror", - "url", "utf-8", - "webpki-roots 0.24.0", + "webpki-roots", ] [[package]] @@ -7597,20 +8013,20 @@ checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1" [[package]] name = "ureq" -version = "2.9.1" +version = "2.10.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f8cdd25c339e200129fe4de81451814e5228c9b771d57378817d6117cc2b3f97" +checksum = "b74fc6b57825be3373f7054754755f03ac3a8f5d70015ccad699ba2029956f4a" dependencies = [ - "base64 0.21.7", + "base64 0.22.1", "flate2", "log", "once_cell", "rustls", - "rustls-webpki", + "rustls-pki-types", "serde", "serde_json", "url", - "webpki-roots 0.25.4", + "webpki-roots", ] [[package]] @@ -7713,6 +8129,15 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "want" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bfa7760aed19e106de2c7c0b581b509f2f25d3dacaf737cb82ac61bc6d760b0e" +dependencies = [ + "try-lock", +] + [[package]] name = "wasi" version = "0.11.0+wasi-snapshot-preview1" @@ -8051,19 +8476,13 @@ dependencies = [ [[package]] name = "webpki-roots" -version = "0.24.0" +version = "0.26.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b291546d5d9d1eab74f069c77749f2cb8504a12caa20f0f2de93ddbf6f411888" +checksum = "841c67bff177718f1d4dfefde8d8f0e78f9b6589319ba88312f567fc5841a958" dependencies = [ - "rustls-webpki", + "rustls-pki-types", ] -[[package]] -name = "webpki-roots" -version = "0.25.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5f20c57d8d7db6d3b86154206ae5d8fba62dd39573114de97c2cb0578251f8e1" - [[package]] name = "weezl" version = "0.1.7" @@ -8731,6 +9150,12 @@ dependencies = [ "syn 2.0.48", ] +[[package]] +name = "zeroize" +version = "1.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ced3678a2879b30306d323f4542626697a464a97c0a07c9aebf7ebca65cd4dde" + [[package]] name = "zip" version = "0.6.6" diff --git a/Cargo.toml b/Cargo.toml index 59262497c974..449dd1b55e41 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -37,6 +37,7 @@ version = "0.19.0-alpha.1+dev" re_build_info = { path = "crates/build/re_build_info", version = "=0.19.0-alpha.1", default-features = false } re_build_tools = { path = "crates/build/re_build_tools", version = "=0.19.0-alpha.1", default-features = false } re_dev_tools = { path = "crates/build/re_dev_tools", version = "=0.19.0-alpha.1", default-features = false } +re_remote_store_types_builder = { path = "crates/build/re_remote_store_types_builder", version = "=0.19.0-alpha.1", default-features = false } re_types_builder = { path = "crates/build/re_types_builder", version = "=0.19.0-alpha.1", default-features = false } # crates/store: @@ -50,6 +51,7 @@ re_format_arrow = { path = "crates/store/re_format_arrow", version = "=0.19.0-al re_log_encoding = { path = "crates/store/re_log_encoding", version = "=0.19.0-alpha.1", default-features = false } re_log_types = { path = "crates/store/re_log_types", version = "=0.19.0-alpha.1", default-features = false } re_query = { path = "crates/store/re_query", version = "=0.19.0-alpha.1", default-features = false } +re_remote_store_types = { path = "crates/store/re_remote_store_types", version = "=0.19.0-alpha.1", default-features = false } re_sdk_comms = { path = "crates/store/re_sdk_comms", version = "=0.19.0-alpha.1", default-features = false } re_types = { path = "crates/store/re_types", version = "=0.19.0-alpha.1", default-features = false } re_types_blueprint = { path = "crates/store/re_types_blueprint", version = "=0.19.0-alpha.1", default-features = false } @@ -103,7 +105,7 @@ re_viewport_blueprint = { path = "crates/viewer/re_viewport_blueprint", version re_web_viewer_server = { path = "crates/viewer/re_web_viewer_server", version = "=0.19.0-alpha.1", default-features = false } # Rerun crates in other repos: -ewebsock = "0.6.0" +ewebsock = { git = "https://github.com/rerun-io/ewebsock", rev = "1a9e78aa18390511476f9e2c0453502f11a7aa78" } re_math = "0.20.0" # egui-crates: @@ -219,6 +221,7 @@ pollster = "0.3" prettyplease = "0.2" proc-macro2 = { version = "1.0", default-features = false } profiling = { version = "1.0.12", default-features = false } +prost = "0.13.3" puffin = "0.19.1" puffin_http = "0.16" pyo3 = "0.21.2" @@ -258,12 +261,14 @@ tinystl = { version = "0.0.3", default-features = false } tinyvec = { version = "1.6", features = ["alloc", "rustc_1_55"] } tobj = "4.0" toml = { version = "0.8.10", default-features = false } +tonic = "0.12.3" +tonic-build = "0.12.3" tracing = { version = "0.1", default-features = false } -tungstenite = { version = "0.20", default-features = false } +tungstenite = { version = "0.23", default-features = false } type-map = "0.5" typenum = "1.15" unindent = "0.2" -ureq = "2.6" +ureq = "2.9.2" url = "2.3" uuid = "1.1" vec1 = "1.8" diff --git a/crates/build/README.md b/crates/build/README.md index 8e09a075d66f..615f8490ad09 100644 --- a/crates/build/README.md +++ b/crates/build/README.md @@ -1 +1,12 @@ -Crates run at build-time +# re_remote_store_types_builder + +Part of the [`rerun`](https://github.com/rerun-io/rerun) family of crates. + +[![Latest version](https://img.shields.io/crates/v/re_remote_store_types_builder.svg)](https://crates.io/crates/re_remote_store_types_builder) +[![Documentation](https://docs.rs/re_remote_store_types_builder/badge.svg)](https://docs.rs/re_remote_store_types_builder) +![MIT](https://img.shields.io/badge/license-MIT-blue.svg) +![Apache](https://img.shields.io/badge/license-Apache-blue.svg) + +This crate implements Rerun's code generation for the remote store node gRPC API definition. + +You can generate the code with `pixi run codegen-rstore`. diff --git a/crates/build/re_remote_store_types_builder/Cargo.toml b/crates/build/re_remote_store_types_builder/Cargo.toml new file mode 100644 index 000000000000..348173742ed2 --- /dev/null +++ b/crates/build/re_remote_store_types_builder/Cargo.toml @@ -0,0 +1,26 @@ +[package] +name = "re_remote_store_types_builder" +authors.workspace = true +edition.workspace = true +homepage.workspace = true +include.workspace = true +license.workspace = true +repository.workspace = true +rust-version.workspace = true +version.workspace = true + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[dependencies] +re_dataframe.workspace = true +re_log = { workspace = true, features = ["setup"] } +re_log_types.workspace = true +re_tracing = { workspace = true, features = ["server"] } + +# External +camino.workspace = true +tonic.workspace = true +tonic-build.workspace = true + +[lints] +workspace = true diff --git a/crates/build/re_remote_store_types_builder/README.md b/crates/build/re_remote_store_types_builder/README.md new file mode 100644 index 000000000000..615f8490ad09 --- /dev/null +++ b/crates/build/re_remote_store_types_builder/README.md @@ -0,0 +1,12 @@ +# re_remote_store_types_builder + +Part of the [`rerun`](https://github.com/rerun-io/rerun) family of crates. + +[![Latest version](https://img.shields.io/crates/v/re_remote_store_types_builder.svg)](https://crates.io/crates/re_remote_store_types_builder) +[![Documentation](https://docs.rs/re_remote_store_types_builder/badge.svg)](https://docs.rs/re_remote_store_types_builder) +![MIT](https://img.shields.io/badge/license-MIT-blue.svg) +![Apache](https://img.shields.io/badge/license-Apache-blue.svg) + +This crate implements Rerun's code generation for the remote store node gRPC API definition. + +You can generate the code with `pixi run codegen-rstore`. diff --git a/crates/build/re_remote_store_types_builder/src/bin/build_re_remote_store_types.rs b/crates/build/re_remote_store_types_builder/src/bin/build_re_remote_store_types.rs new file mode 100644 index 000000000000..f70dc0cca261 --- /dev/null +++ b/crates/build/re_remote_store_types_builder/src/bin/build_re_remote_store_types.rs @@ -0,0 +1,38 @@ +//! This binary runs the remote store gRPC service codegen manually. +//! +//! It is easiest to call this using `pixi run codegen-rstore`, +//! which will set up the necessary tools. + +#![allow(clippy::unwrap_used)] + +use camino::Utf8Path; + +const PROTOBUF_DEFINITIONS_DIR_PATH: &str = "crates/store/re_remote_store_types/proto"; +const PROTOBUF_REMOTE_STORE_V0_RELATIVE_PATH: &str = "rerun/v0/remote_store.proto"; +const RUST_V0_OUTPUT_DIR_PATH: &str = "crates/store/re_remote_store_types/src/v0"; + +fn main() { + re_log::setup_logging(); + + let workspace_dir = Utf8Path::new(env!("CARGO_MANIFEST_DIR")) + .parent() + .and_then(|p| p.parent()) + .and_then(|p| p.parent()) + .unwrap(); + + assert!( + workspace_dir.join("CODE_OF_CONDUCT.md").exists(), + "failed to find workspace root" + ); + + let definitions_dir_path = workspace_dir.join(PROTOBUF_DEFINITIONS_DIR_PATH); + let rust_generated_output_dir_path = workspace_dir.join(RUST_V0_OUTPUT_DIR_PATH); + + re_log::info!("Running codegen for storage node types"); + + re_remote_store_types_builder::generate_rust_code( + definitions_dir_path, + &[PROTOBUF_REMOTE_STORE_V0_RELATIVE_PATH], + rust_generated_output_dir_path, + ); +} diff --git a/crates/build/re_remote_store_types_builder/src/lib.rs b/crates/build/re_remote_store_types_builder/src/lib.rs new file mode 100644 index 000000000000..f9ca751381a2 --- /dev/null +++ b/crates/build/re_remote_store_types_builder/src/lib.rs @@ -0,0 +1,26 @@ +//! This crate contains logic for generating remote store gRPC API types as defined in +//! `re_remote_store_types` proto files. We are currently generating both client and server +//! definitions in the same file. +//! + +#![allow(clippy::unwrap_used)] + +use std::path::Path; + +/// Generate rust from from protobuf definitions. We rely on `tonic_build` to do the heavy lifting. +/// `tonic_build` relies on `prost` which itself relies on `protoc`. +/// +/// Note: make sure to invoke this via `pixi run codegen-rstore` in order to use the right `protoc` version. +pub fn generate_rust_code( + definitions_dir: impl AsRef, + proto_paths: &[impl AsRef], + output_dir: impl AsRef, +) { + tonic_build::configure() + .out_dir(output_dir.as_ref()) + .build_client(true) + .build_server(true) + .build_transport(true) + .compile_protos(proto_paths, &[definitions_dir]) + .unwrap(); +} diff --git a/crates/store/re_remote_store_types/Cargo.toml b/crates/store/re_remote_store_types/Cargo.toml new file mode 100644 index 000000000000..5db04a5a089a --- /dev/null +++ b/crates/store/re_remote_store_types/Cargo.toml @@ -0,0 +1,23 @@ +[package] +name = "re_remote_store_types" +authors.workspace = true +edition.workspace = true +homepage.workspace = true +include.workspace = true +license.workspace = true +repository.workspace = true +rust-version.workspace = true +version.workspace = true + +[dependencies] +re_log_types.workspace = true +re_dataframe.workspace = true + +# External +prost.workspace = true +thiserror.workspace = true +tonic.workspace = true +tonic-build.workspace = true + +[lints] +workspace = true diff --git a/crates/store/re_remote_store_types/README.md b/crates/store/re_remote_store_types/README.md new file mode 100644 index 000000000000..418df5c16895 --- /dev/null +++ b/crates/store/re_remote_store_types/README.md @@ -0,0 +1,14 @@ +# re_remote_store_types + +Part of the [`rerun`](https://github.com/rerun-io/rerun) family of crates. + +[![Latest version](https://img.shields.io/crates/v/re_remote_store_types.svg)](https://crates.io/crates/re_remote_store_types) +[![Documentation](https://docs.rs/re_remote_store_types/badge.svg)](https://docs.rs/re_remote_store_types) +![MIT](https://img.shields.io/badge/license-MIT-blue.svg) +![Apache](https://img.shields.io/badge/license-Apache-blue.svg) + +Rerun remote store node gRPC API service types (client and server). + +This crate includes both the language-agnostic definitions (protobuf) as well as the generated code. + +The code is generated with `pixi run codegen-rstore`. diff --git a/crates/store/re_remote_store_types/proto/rerun/v0/common.proto b/crates/store/re_remote_store_types/proto/rerun/v0/common.proto new file mode 100644 index 000000000000..072cff95f2a5 --- /dev/null +++ b/crates/store/re_remote_store_types/proto/rerun/v0/common.proto @@ -0,0 +1,117 @@ +syntax = "proto3"; + +package rerun.remote_store.v0; + +// unique recording identifier. At this point in time it is the same id as the ChunkStore's StoreId +message RecordingId { + string id = 1; +} + +// A recording can have multiple timelines, each is identified by a name, for example `log_tick`, `log_time`, etc. +message Timeline { + string name = 1; +} + +// A time range between start and end time points. Each 64 bit number can represent different time point data +// depending on the timeline it is associated with. +message TimeRange { + int64 start = 1; + int64 end = 2; +} + +// arrow IPC serialized schema +message Schema { + bytes arrow_schema = 1; +} + +message Query { + // database view defined by entity paths and components + ViewContents view_contents = 1; + // filtering index (just a string i.e. a name of the timeline for starters) + IndexColumnSelector filtered_index = 2; + // Optional specific range for the index selector + IndexRange filtered_index_range = 3; + // Optional specific values for the index selector + IndexValues filtered_index_values = 4; + // Optional index selector sampling + IndexValues using_index_values = 5; + // PoV (filtering) component + ComponentColumnSelector filtered_pov = 6; + // which columns to include in the response + // Note - we have one more layer of indiraction to ensure the field is optional, + // same as in the query expression. We can't have both 'repeated' and 'optional' field labels. + ColumnSelection column_selection = 7; + // how are null values filled in the response + SparseFillStrategy sparse_fill_strategy = 8; +} + +message ColumnSelection { + repeated ColumnSelector columns = 1; +} + +message ColumnSelector { + oneof selector_type { + ComponentColumnSelector component_column = 2; + TimeColumnSelector time_column = 3; + } +} + +message IndexColumnSelector { + // TODO (zehiko) we need to add support for other types of index selectors + Timeline timeline = 1; +} + +message IndexRange { + // TODO (zehiko) support for other ranges for other index selectors + TimeRange time_range = 1; +} + +message IndexValues { + // TODO (zehiko) we need to add support for other types of index selectors + repeated TimeInt time_points = 1; +} + +message SampledIndexValues { + repeated TimeInt sample_points = 1; +} + +message TimeInt { + int64 time = 1; +} + +message ViewContents { + repeated ViewContentsPart contents = 1; +} + +message ViewContentsPart { + EntityPath path = 1; + ComponentsSet components = 2; +} + +message ComponentsSet { + repeated Component components = 1; +} + +message EntityPath { + string path = 1; +} + +message Component { + // component name needs to be a string as user can define their own component + string name = 1; +} + +message TimeColumnSelector { + Timeline timeline = 1; +} + +message ComponentColumnSelector { + EntityPath entity_path = 1; + Component component = 2; + // TODO do we need join encoding? +} + +enum SparseFillStrategy { + NONE = 0; + LATEST_AL_GLOBAL = 1; +} diff --git a/crates/store/re_remote_store_types/proto/rerun/v0/remote_store.proto b/crates/store/re_remote_store_types/proto/rerun/v0/remote_store.proto new file mode 100644 index 000000000000..8be53c1a0ffe --- /dev/null +++ b/crates/store/re_remote_store_types/proto/rerun/v0/remote_store.proto @@ -0,0 +1,91 @@ +syntax = "proto3"; + +package rerun.remote_store.v0; + +import "rerun/v0/common.proto"; + +service StorageNode { + rpc ListRecordings(ListRecordingsRequest) returns (ListRecordingsResponse) {} + rpc Query(QueryRequest) returns (stream QueryResponse) {} + rpc GetRecordingMetadata(GetRecordingMetadataRequest) returns (GetRecordingMetadataResponse) {} + // TODO (zehiko) - should this be singular recording registration? Currently we can have 1 rrd => many recordings + rpc RegisterRecordings(RegisterRecordingsRequest) returns (RegisterRecordingsResponse) {} +} + +// ---------------- RegisterRecording ------------------ + +message RegisterRecordingsRequest { + string description = 1; + ObjectStorage obj_storage = 2; + // TODO (zehiko) should this be auto-discoverable? + RecordingType typ = 3; +} + +message ObjectStorage { + string bucket_name = 1; + string url = 2; +} + +message RegisterRecordingsResponse { + // Note / TODO (zehiko): this implies we read the record (for example go through entire .rrd file + // chunk by chunk) and extract the metadata. So we might want to 1/ not do this i.e. + // only do it as part of explicit GetMetadata request or 2/ do it if Request has "include_metadata=true" + // or 3/ do it always + repeated RecordingMetadata metadata = 2; +} + +// ---------------- GetRecordingMetadata ----------------- + +message GetRecordingMetadataRequest { + RecordingId recording_id = 1; +} + +message GetRecordingMetadataResponse { + RecordingMetadata metadata = 1; +} + +message RecordingMetadata { + RecordingId id = 1; + Schema schema = 2; + repeated TimeMetadata time_metadata = 3; +} + +message TimeMetadata { + Timeline timeline = 1; + TimeRange time_range = 2; +} + +// ---------------- Query ----------------- + +message QueryRequest { + // unique identifier of the recording + RecordingId recording_id = 1; + // query to execute + Query query = 2; +} + +message QueryResponse { + // single record batch (encoding TBD - TODO). + bytes record_batch = 1; +} + + +// ----------------- ListRecordings ----------------- + +message ListRecordingsRequest {} + +message ListRecordingsResponse { + repeated RecordingInfo recordings = 1; +} + +message RecordingInfo { + RecordingId id = 1; + string description = 2; + string storage_url = 3; + uint64 size_bytes = 4; + RecordingType typ = 5; +} + +enum RecordingType { + RRD = 0; +} diff --git a/crates/store/re_remote_store_types/src/lib.rs b/crates/store/re_remote_store_types/src/lib.rs new file mode 100644 index 000000000000..f1a327ae94fd --- /dev/null +++ b/crates/store/re_remote_store_types/src/lib.rs @@ -0,0 +1,217 @@ +//! This crate contains generated types for the remote store gRPC service API. +//! Generation is done using the `re_remote_store_types_builder` crate. +//! +//! We want clear separation between 'internal' types and gRPC types and don't want +//! to use gRPC types in the rerun viewer codebase. That's why we implement all the +//! necessary conversion code (in the form of `From` and `TryFrom` traits) in this crate. +//! + +// Ignoring all warnings for the auto-generated code. +#![allow(clippy::doc_markdown)] +#![allow(clippy::derive_partial_eq_without_eq)] +#![allow(clippy::enum_variant_names)] +#![allow(clippy::unwrap_used)] +#![allow(clippy::wildcard_imports)] +#![allow(clippy::manual_is_variant_and)] +/// Generated types for the remote store gRPC service API v0. +pub mod v0 { + #[path = "../v0/rerun.remote_store.v0.rs"] + mod _v0; + + pub use self::_v0::*; + + // ==== below are all necessary transforms from internal rerun types to protobuf types ===== + + use std::collections::BTreeSet; + + #[derive(Debug, thiserror::Error)] + pub enum TypeConversionError { + #[error("missing required field: {0}")] + MissingField(&'static str), + } + + impl From for TimeRange { + fn from(time_range: re_log_types::ResolvedTimeRange) -> Self { + Self { + start: time_range.min().as_i64(), + end: time_range.max().as_i64(), + } + } + } + + impl TryFrom for re_dataframe::external::re_chunk_store::QueryExpression { + type Error = TypeConversionError; + + fn try_from(value: Query) -> Result { + let filtered_index = value + .filtered_index + .ok_or(TypeConversionError::MissingField("filtered_index"))? + .try_into()?; + + let selection = value + .column_selection + .map(|cs| { + cs.columns + .into_iter() + .map(|c| { + re_dataframe::external::re_chunk_store::ColumnSelector::try_from(c) + }) + .collect::, _>>() + }) + .transpose()?; + + let filtered_point_of_view = value + .filtered_pov + .map(|fp| { + re_dataframe::external::re_chunk_store::ComponentColumnSelector::try_from(fp) + }) + .transpose()?; + + Ok(Self { + view_contents: value.view_contents.map(|vc| vc.into()), + filtered_index, + filtered_index_range: value + .filtered_index_range + .map(|ir| ir.try_into()) + .transpose()?, + filtered_index_values: value + .filtered_index_values + .map(|iv| iv.time_points.into_iter().map(|v| v.into()).collect()), + using_index_values: value + .using_index_values + .map(|uiv| uiv.time_points.into_iter().map(|v| v.into()).collect()), + filtered_point_of_view, + sparse_fill_strategy: + re_dataframe::external::re_chunk_store::SparseFillStrategy::default(), // TODO(zehiko) implement support for sparse fill strategy + selection, + }) + } + } + + impl From for re_dataframe::external::re_chunk_store::ViewContentsSelector { + fn from(value: ViewContents) -> Self { + value + .contents + .into_iter() + .map(|part| { + // TODO(zehiko) option unwrap + let entity_path = Into::::into(part.path.unwrap()); + let column_selector = part.components.map(|cs| { + cs.components + .into_iter() + .map(|c| re_dataframe::external::re_chunk::ComponentName::new(&c.name)) + .collect::>() + }); + (entity_path, column_selector) + }) + .collect::() + } + } + + impl From for re_log_types::EntityPath { + fn from(value: EntityPath) -> Self { + Self::from(value.path) + } + } + + impl TryFrom for re_log_types::Timeline { + type Error = TypeConversionError; + + fn try_from(value: IndexColumnSelector) -> Result { + let timeline_name = value + .timeline + .ok_or(TypeConversionError::MissingField("timeline"))? + .name; + + #[allow(clippy::match_same_arms)] + let timeline = match timeline_name.as_str() { + "log_time" => Self::new_temporal(timeline_name), + "log_tick" => Self::new_sequence(timeline_name), + "frame" => Self::new_sequence(timeline_name), + "frame_nr" => Self::new_sequence(timeline_name), + _ => Self::new_temporal(timeline_name), + }; + + Ok(timeline) + } + } + + impl TryFrom for re_dataframe::external::re_chunk_store::IndexRange { + type Error = TypeConversionError; + + fn try_from(value: IndexRange) -> Result { + let time_range = value + .time_range + .ok_or(TypeConversionError::MissingField("time_range"))?; + + Ok(Self::new(time_range.start, time_range.end)) + } + } + + impl From for re_log_types::TimeInt { + fn from(value: TimeInt) -> Self { + Self::new_temporal(value.time) + } + } + + impl TryFrom + for re_dataframe::external::re_chunk_store::ComponentColumnSelector + { + type Error = TypeConversionError; + + fn try_from(value: ComponentColumnSelector) -> Result { + let entity_path = value + .entity_path + .ok_or(TypeConversionError::MissingField("entity_path"))? + .into(); + + let component = value + .component + .ok_or(TypeConversionError::MissingField("component"))? + .name; + + Ok(Self { + entity_path, + component: re_dataframe::external::re_chunk::ComponentName::new(&component), + join_encoding: re_dataframe::external::re_chunk_store::JoinEncoding::default(), // TODO(zehiko) implement + }) + } + } + + impl TryFrom for re_dataframe::external::re_chunk_store::TimeColumnSelector { + type Error = TypeConversionError; + + fn try_from(value: TimeColumnSelector) -> Result { + let timeline = value + .timeline + .ok_or(TypeConversionError::MissingField("timeline"))?; + + Ok(Self { + timeline: timeline.name.into(), + }) + } + } + + impl TryFrom for re_dataframe::external::re_chunk_store::ColumnSelector { + type Error = TypeConversionError; + + fn try_from(value: ColumnSelector) -> Result { + match value + .selector_type + .ok_or(TypeConversionError::MissingField("selector_type"))? + { + column_selector::SelectorType::ComponentColumn(component_column_selector) => { + let selector: re_dataframe::external::re_chunk_store::ComponentColumnSelector = + component_column_selector.try_into()?; + Ok(selector.into()) + } + column_selector::SelectorType::TimeColumn(time_column_selector) => { + let selector: re_dataframe::external::re_chunk_store::TimeColumnSelector = + time_column_selector.try_into()?; + + Ok(selector.into()) + } + } + } + } +} diff --git a/crates/store/re_remote_store_types/src/v0/rerun.remote_store.v0.rs b/crates/store/re_remote_store_types/src/v0/rerun.remote_store.v0.rs new file mode 100644 index 000000000000..8ba9fadd7e2a --- /dev/null +++ b/crates/store/re_remote_store_types/src/v0/rerun.remote_store.v0.rs @@ -0,0 +1,760 @@ +// This file is @generated by prost-build. +/// unique recording identifier. At this point in time it is the same id as the ChunkStore's StoreId +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct RecordingId { + #[prost(string, tag = "1")] + pub id: ::prost::alloc::string::String, +} +/// A recording can have multiple timelines, each is identified by a name, for example `log_tick`, `log_time`, etc. +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct Timeline { + #[prost(string, tag = "1")] + pub name: ::prost::alloc::string::String, +} +/// A time range between start and end time points. Each 64 bit number can represent different time point data +/// depending on the timeline it is associated with. +#[derive(Clone, Copy, PartialEq, ::prost::Message)] +pub struct TimeRange { + #[prost(int64, tag = "1")] + pub start: i64, + #[prost(int64, tag = "2")] + pub end: i64, +} +/// arrow IPC serialized schema +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct Schema { + #[prost(bytes = "vec", tag = "1")] + pub arrow_schema: ::prost::alloc::vec::Vec, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct Query { + /// database view defined by entity paths and components + #[prost(message, optional, tag = "1")] + pub view_contents: ::core::option::Option, + /// filtering index (just a string i.e. a name of the timeline for starters) + #[prost(message, optional, tag = "2")] + pub filtered_index: ::core::option::Option, + /// Optional specific range for the index selector + #[prost(message, optional, tag = "3")] + pub filtered_index_range: ::core::option::Option, + /// Optional specific values for the index selector + #[prost(message, optional, tag = "4")] + pub filtered_index_values: ::core::option::Option, + /// Optional index selector sampling + #[prost(message, optional, tag = "5")] + pub using_index_values: ::core::option::Option, + /// PoV (filtering) component + #[prost(message, optional, tag = "6")] + pub filtered_pov: ::core::option::Option, + /// which columns to include in the response + /// Note - we have one more layer of indiraction to ensure the field is optional, + /// same as in the query expression. We can't have both 'repeated' and 'optional' field labels. + #[prost(message, optional, tag = "7")] + pub column_selection: ::core::option::Option, + /// how are null values filled in the response + #[prost(enumeration = "SparseFillStrategy", tag = "8")] + pub sparse_fill_strategy: i32, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ColumnSelection { + #[prost(message, repeated, tag = "1")] + pub columns: ::prost::alloc::vec::Vec, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ColumnSelector { + #[prost(oneof = "column_selector::SelectorType", tags = "2, 3")] + pub selector_type: ::core::option::Option, +} +/// Nested message and enum types in `ColumnSelector`. +pub mod column_selector { + #[derive(Clone, PartialEq, ::prost::Oneof)] + pub enum SelectorType { + #[prost(message, tag = "2")] + ComponentColumn(super::ComponentColumnSelector), + #[prost(message, tag = "3")] + TimeColumn(super::TimeColumnSelector), + } +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct IndexColumnSelector { + /// TODO (zehiko) we need to add support for other types of index selectors + #[prost(message, optional, tag = "1")] + pub timeline: ::core::option::Option, +} +#[derive(Clone, Copy, PartialEq, ::prost::Message)] +pub struct IndexRange { + /// TODO (zehiko) support for other ranges for other index selectors + #[prost(message, optional, tag = "1")] + pub time_range: ::core::option::Option, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct IndexValues { + /// TODO (zehiko) we need to add support for other types of index selectors + #[prost(message, repeated, tag = "1")] + pub time_points: ::prost::alloc::vec::Vec, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct SampledIndexValues { + #[prost(message, repeated, tag = "1")] + pub sample_points: ::prost::alloc::vec::Vec, +} +#[derive(Clone, Copy, PartialEq, ::prost::Message)] +pub struct TimeInt { + #[prost(int64, tag = "1")] + pub time: i64, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ViewContents { + #[prost(message, repeated, tag = "1")] + pub contents: ::prost::alloc::vec::Vec, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ViewContentsPart { + #[prost(message, optional, tag = "1")] + pub path: ::core::option::Option, + #[prost(message, optional, tag = "2")] + pub components: ::core::option::Option, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ComponentsSet { + #[prost(message, repeated, tag = "1")] + pub components: ::prost::alloc::vec::Vec, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct EntityPath { + #[prost(string, tag = "1")] + pub path: ::prost::alloc::string::String, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct Component { + /// component name needs to be a string as user can define their own component + #[prost(string, tag = "1")] + pub name: ::prost::alloc::string::String, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct TimeColumnSelector { + #[prost(message, optional, tag = "1")] + pub timeline: ::core::option::Option, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ComponentColumnSelector { + #[prost(message, optional, tag = "1")] + pub entity_path: ::core::option::Option, + /// TODO do we need join encoding? + #[prost(message, optional, tag = "2")] + pub component: ::core::option::Option, +} +#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash, PartialOrd, Ord, ::prost::Enumeration)] +#[repr(i32)] +pub enum SparseFillStrategy { + None = 0, + LatestAlGlobal = 1, +} +impl SparseFillStrategy { + /// String value of the enum field names used in the ProtoBuf definition. + /// + /// The values are not transformed in any way and thus are considered stable + /// (if the ProtoBuf definition does not change) and safe for programmatic use. + pub fn as_str_name(&self) -> &'static str { + match self { + Self::None => "NONE", + Self::LatestAlGlobal => "LATEST_AL_GLOBAL", + } + } + /// Creates an enum from field names used in the ProtoBuf definition. + pub fn from_str_name(value: &str) -> ::core::option::Option { + match value { + "NONE" => Some(Self::None), + "LATEST_AL_GLOBAL" => Some(Self::LatestAlGlobal), + _ => None, + } + } +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct RegisterRecordingsRequest { + #[prost(string, tag = "1")] + pub description: ::prost::alloc::string::String, + #[prost(message, optional, tag = "2")] + pub obj_storage: ::core::option::Option, + /// TODO (zehiko) should this be auto-discoverable? + #[prost(enumeration = "RecordingType", tag = "3")] + pub typ: i32, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ObjectStorage { + #[prost(string, tag = "1")] + pub bucket_name: ::prost::alloc::string::String, + #[prost(string, tag = "2")] + pub url: ::prost::alloc::string::String, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct RegisterRecordingsResponse { + /// Note / TODO (zehiko): this implies we read the record (for example go through entire .rrd file + /// chunk by chunk) and extract the metadata. So we might want to 1/ not do this i.e. + /// only do it as part of explicit GetMetadata request or 2/ do it if Request has "include_metadata=true" + /// or 3/ do it always + #[prost(message, repeated, tag = "2")] + pub metadata: ::prost::alloc::vec::Vec, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct GetRecordingMetadataRequest { + #[prost(message, optional, tag = "1")] + pub recording_id: ::core::option::Option, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct GetRecordingMetadataResponse { + #[prost(message, optional, tag = "1")] + pub metadata: ::core::option::Option, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct RecordingMetadata { + #[prost(message, optional, tag = "1")] + pub id: ::core::option::Option, + #[prost(message, optional, tag = "2")] + pub schema: ::core::option::Option, + #[prost(message, repeated, tag = "3")] + pub time_metadata: ::prost::alloc::vec::Vec, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct TimeMetadata { + #[prost(message, optional, tag = "1")] + pub timeline: ::core::option::Option, + #[prost(message, optional, tag = "2")] + pub time_range: ::core::option::Option, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct QueryRequest { + /// unique identifier of the recording + #[prost(message, optional, tag = "1")] + pub recording_id: ::core::option::Option, + /// query to execute + #[prost(message, optional, tag = "2")] + pub query: ::core::option::Option, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct QueryResponse { + /// single record batch (encoding TBD - TODO). + #[prost(bytes = "vec", tag = "1")] + pub record_batch: ::prost::alloc::vec::Vec, +} +#[derive(Clone, Copy, PartialEq, ::prost::Message)] +pub struct ListRecordingsRequest {} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ListRecordingsResponse { + #[prost(message, repeated, tag = "1")] + pub recordings: ::prost::alloc::vec::Vec, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct RecordingInfo { + #[prost(message, optional, tag = "1")] + pub id: ::core::option::Option, + #[prost(string, tag = "2")] + pub description: ::prost::alloc::string::String, + #[prost(string, tag = "3")] + pub storage_url: ::prost::alloc::string::String, + #[prost(uint64, tag = "4")] + pub size_bytes: u64, + #[prost(enumeration = "RecordingType", tag = "5")] + pub typ: i32, +} +#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash, PartialOrd, Ord, ::prost::Enumeration)] +#[repr(i32)] +pub enum RecordingType { + Rrd = 0, +} +impl RecordingType { + /// String value of the enum field names used in the ProtoBuf definition. + /// + /// The values are not transformed in any way and thus are considered stable + /// (if the ProtoBuf definition does not change) and safe for programmatic use. + pub fn as_str_name(&self) -> &'static str { + match self { + Self::Rrd => "RRD", + } + } + /// Creates an enum from field names used in the ProtoBuf definition. + pub fn from_str_name(value: &str) -> ::core::option::Option { + match value { + "RRD" => Some(Self::Rrd), + _ => None, + } + } +} +/// Generated client implementations. +pub mod storage_node_client { + #![allow( + unused_variables, + dead_code, + missing_docs, + clippy::wildcard_imports, + clippy::let_unit_value + )] + use tonic::codegen::http::Uri; + use tonic::codegen::*; + #[derive(Debug, Clone)] + pub struct StorageNodeClient { + inner: tonic::client::Grpc, + } + impl StorageNodeClient { + /// Attempt to create a new client by connecting to a given endpoint. + pub async fn connect(dst: D) -> Result + where + D: TryInto, + D::Error: Into, + { + let conn = tonic::transport::Endpoint::new(dst)?.connect().await?; + Ok(Self::new(conn)) + } + } + impl StorageNodeClient + where + T: tonic::client::GrpcService, + T::Error: Into, + T::ResponseBody: Body + std::marker::Send + 'static, + ::Error: Into + std::marker::Send, + { + pub fn new(inner: T) -> Self { + let inner = tonic::client::Grpc::new(inner); + Self { inner } + } + pub fn with_origin(inner: T, origin: Uri) -> Self { + let inner = tonic::client::Grpc::with_origin(inner, origin); + Self { inner } + } + pub fn with_interceptor( + inner: T, + interceptor: F, + ) -> StorageNodeClient> + where + F: tonic::service::Interceptor, + T::ResponseBody: Default, + T: tonic::codegen::Service< + http::Request, + Response = http::Response< + >::ResponseBody, + >, + >, + >>::Error: + Into + std::marker::Send + std::marker::Sync, + { + StorageNodeClient::new(InterceptedService::new(inner, interceptor)) + } + /// Compress requests with the given encoding. + /// + /// This requires the server to support it otherwise it might respond with an + /// error. + #[must_use] + pub fn send_compressed(mut self, encoding: CompressionEncoding) -> Self { + self.inner = self.inner.send_compressed(encoding); + self + } + /// Enable decompressing responses. + #[must_use] + pub fn accept_compressed(mut self, encoding: CompressionEncoding) -> Self { + self.inner = self.inner.accept_compressed(encoding); + self + } + /// Limits the maximum size of a decoded message. + /// + /// Default: `4MB` + #[must_use] + pub fn max_decoding_message_size(mut self, limit: usize) -> Self { + self.inner = self.inner.max_decoding_message_size(limit); + self + } + /// Limits the maximum size of an encoded message. + /// + /// Default: `usize::MAX` + #[must_use] + pub fn max_encoding_message_size(mut self, limit: usize) -> Self { + self.inner = self.inner.max_encoding_message_size(limit); + self + } + pub async fn list_recordings( + &mut self, + request: impl tonic::IntoRequest, + ) -> std::result::Result, tonic::Status> + { + self.inner.ready().await.map_err(|e| { + tonic::Status::unknown(format!("Service was not ready: {}", e.into())) + })?; + let codec = tonic::codec::ProstCodec::default(); + let path = http::uri::PathAndQuery::from_static( + "/rerun.remote_store.v0.StorageNode/ListRecordings", + ); + let mut req = request.into_request(); + req.extensions_mut().insert(GrpcMethod::new( + "rerun.remote_store.v0.StorageNode", + "ListRecordings", + )); + self.inner.unary(req, path, codec).await + } + pub async fn query( + &mut self, + request: impl tonic::IntoRequest, + ) -> std::result::Result< + tonic::Response>, + tonic::Status, + > { + self.inner.ready().await.map_err(|e| { + tonic::Status::unknown(format!("Service was not ready: {}", e.into())) + })?; + let codec = tonic::codec::ProstCodec::default(); + let path = + http::uri::PathAndQuery::from_static("/rerun.remote_store.v0.StorageNode/Query"); + let mut req = request.into_request(); + req.extensions_mut().insert(GrpcMethod::new( + "rerun.remote_store.v0.StorageNode", + "Query", + )); + self.inner.server_streaming(req, path, codec).await + } + pub async fn get_recording_metadata( + &mut self, + request: impl tonic::IntoRequest, + ) -> std::result::Result, tonic::Status> + { + self.inner.ready().await.map_err(|e| { + tonic::Status::unknown(format!("Service was not ready: {}", e.into())) + })?; + let codec = tonic::codec::ProstCodec::default(); + let path = http::uri::PathAndQuery::from_static( + "/rerun.remote_store.v0.StorageNode/GetRecordingMetadata", + ); + let mut req = request.into_request(); + req.extensions_mut().insert(GrpcMethod::new( + "rerun.remote_store.v0.StorageNode", + "GetRecordingMetadata", + )); + self.inner.unary(req, path, codec).await + } + /// TODO (zehiko) - should this be singular recording registration? Currently we can have 1 rrd => many recordings + pub async fn register_recordings( + &mut self, + request: impl tonic::IntoRequest, + ) -> std::result::Result, tonic::Status> + { + self.inner.ready().await.map_err(|e| { + tonic::Status::unknown(format!("Service was not ready: {}", e.into())) + })?; + let codec = tonic::codec::ProstCodec::default(); + let path = http::uri::PathAndQuery::from_static( + "/rerun.remote_store.v0.StorageNode/RegisterRecordings", + ); + let mut req = request.into_request(); + req.extensions_mut().insert(GrpcMethod::new( + "rerun.remote_store.v0.StorageNode", + "RegisterRecordings", + )); + self.inner.unary(req, path, codec).await + } + } +} +/// Generated server implementations. +pub mod storage_node_server { + #![allow( + unused_variables, + dead_code, + missing_docs, + clippy::wildcard_imports, + clippy::let_unit_value + )] + use tonic::codegen::*; + /// Generated trait containing gRPC methods that should be implemented for use with StorageNodeServer. + #[async_trait] + pub trait StorageNode: std::marker::Send + std::marker::Sync + 'static { + async fn list_recordings( + &self, + request: tonic::Request, + ) -> std::result::Result, tonic::Status>; + /// Server streaming response type for the Query method. + type QueryStream: tonic::codegen::tokio_stream::Stream< + Item = std::result::Result, + > + std::marker::Send + + 'static; + async fn query( + &self, + request: tonic::Request, + ) -> std::result::Result, tonic::Status>; + async fn get_recording_metadata( + &self, + request: tonic::Request, + ) -> std::result::Result, tonic::Status>; + /// TODO (zehiko) - should this be singular recording registration? Currently we can have 1 rrd => many recordings + async fn register_recordings( + &self, + request: tonic::Request, + ) -> std::result::Result, tonic::Status>; + } + #[derive(Debug)] + pub struct StorageNodeServer { + inner: Arc, + accept_compression_encodings: EnabledCompressionEncodings, + send_compression_encodings: EnabledCompressionEncodings, + max_decoding_message_size: Option, + max_encoding_message_size: Option, + } + impl StorageNodeServer { + pub fn new(inner: T) -> Self { + Self::from_arc(Arc::new(inner)) + } + pub fn from_arc(inner: Arc) -> Self { + Self { + inner, + accept_compression_encodings: Default::default(), + send_compression_encodings: Default::default(), + max_decoding_message_size: None, + max_encoding_message_size: None, + } + } + pub fn with_interceptor(inner: T, interceptor: F) -> InterceptedService + where + F: tonic::service::Interceptor, + { + InterceptedService::new(Self::new(inner), interceptor) + } + /// Enable decompressing requests with the given encoding. + #[must_use] + pub fn accept_compressed(mut self, encoding: CompressionEncoding) -> Self { + self.accept_compression_encodings.enable(encoding); + self + } + /// Compress responses with the given encoding, if the client supports it. + #[must_use] + pub fn send_compressed(mut self, encoding: CompressionEncoding) -> Self { + self.send_compression_encodings.enable(encoding); + self + } + /// Limits the maximum size of a decoded message. + /// + /// Default: `4MB` + #[must_use] + pub fn max_decoding_message_size(mut self, limit: usize) -> Self { + self.max_decoding_message_size = Some(limit); + self + } + /// Limits the maximum size of an encoded message. + /// + /// Default: `usize::MAX` + #[must_use] + pub fn max_encoding_message_size(mut self, limit: usize) -> Self { + self.max_encoding_message_size = Some(limit); + self + } + } + impl tonic::codegen::Service> for StorageNodeServer + where + T: StorageNode, + B: Body + std::marker::Send + 'static, + B::Error: Into + std::marker::Send + 'static, + { + type Response = http::Response; + type Error = std::convert::Infallible; + type Future = BoxFuture; + fn poll_ready( + &mut self, + _cx: &mut Context<'_>, + ) -> Poll> { + Poll::Ready(Ok(())) + } + fn call(&mut self, req: http::Request) -> Self::Future { + match req.uri().path() { + "/rerun.remote_store.v0.StorageNode/ListRecordings" => { + #[allow(non_camel_case_types)] + struct ListRecordingsSvc(pub Arc); + impl tonic::server::UnaryService + for ListRecordingsSvc + { + type Response = super::ListRecordingsResponse; + type Future = BoxFuture, tonic::Status>; + fn call( + &mut self, + request: tonic::Request, + ) -> Self::Future { + let inner = Arc::clone(&self.0); + let fut = async move { + ::list_recordings(&inner, request).await + }; + Box::pin(fut) + } + } + let accept_compression_encodings = self.accept_compression_encodings; + let send_compression_encodings = self.send_compression_encodings; + let max_decoding_message_size = self.max_decoding_message_size; + let max_encoding_message_size = self.max_encoding_message_size; + let inner = self.inner.clone(); + let fut = async move { + let method = ListRecordingsSvc(inner); + let codec = tonic::codec::ProstCodec::default(); + let mut grpc = tonic::server::Grpc::new(codec) + .apply_compression_config( + accept_compression_encodings, + send_compression_encodings, + ) + .apply_max_message_size_config( + max_decoding_message_size, + max_encoding_message_size, + ); + let res = grpc.unary(method, req).await; + Ok(res) + }; + Box::pin(fut) + } + "/rerun.remote_store.v0.StorageNode/Query" => { + #[allow(non_camel_case_types)] + struct QuerySvc(pub Arc); + impl tonic::server::ServerStreamingService for QuerySvc { + type Response = super::QueryResponse; + type ResponseStream = T::QueryStream; + type Future = + BoxFuture, tonic::Status>; + fn call( + &mut self, + request: tonic::Request, + ) -> Self::Future { + let inner = Arc::clone(&self.0); + let fut = + async move { ::query(&inner, request).await }; + Box::pin(fut) + } + } + let accept_compression_encodings = self.accept_compression_encodings; + let send_compression_encodings = self.send_compression_encodings; + let max_decoding_message_size = self.max_decoding_message_size; + let max_encoding_message_size = self.max_encoding_message_size; + let inner = self.inner.clone(); + let fut = async move { + let method = QuerySvc(inner); + let codec = tonic::codec::ProstCodec::default(); + let mut grpc = tonic::server::Grpc::new(codec) + .apply_compression_config( + accept_compression_encodings, + send_compression_encodings, + ) + .apply_max_message_size_config( + max_decoding_message_size, + max_encoding_message_size, + ); + let res = grpc.server_streaming(method, req).await; + Ok(res) + }; + Box::pin(fut) + } + "/rerun.remote_store.v0.StorageNode/GetRecordingMetadata" => { + #[allow(non_camel_case_types)] + struct GetRecordingMetadataSvc(pub Arc); + impl + tonic::server::UnaryService + for GetRecordingMetadataSvc + { + type Response = super::GetRecordingMetadataResponse; + type Future = BoxFuture, tonic::Status>; + fn call( + &mut self, + request: tonic::Request, + ) -> Self::Future { + let inner = Arc::clone(&self.0); + let fut = async move { + ::get_recording_metadata(&inner, request).await + }; + Box::pin(fut) + } + } + let accept_compression_encodings = self.accept_compression_encodings; + let send_compression_encodings = self.send_compression_encodings; + let max_decoding_message_size = self.max_decoding_message_size; + let max_encoding_message_size = self.max_encoding_message_size; + let inner = self.inner.clone(); + let fut = async move { + let method = GetRecordingMetadataSvc(inner); + let codec = tonic::codec::ProstCodec::default(); + let mut grpc = tonic::server::Grpc::new(codec) + .apply_compression_config( + accept_compression_encodings, + send_compression_encodings, + ) + .apply_max_message_size_config( + max_decoding_message_size, + max_encoding_message_size, + ); + let res = grpc.unary(method, req).await; + Ok(res) + }; + Box::pin(fut) + } + "/rerun.remote_store.v0.StorageNode/RegisterRecordings" => { + #[allow(non_camel_case_types)] + struct RegisterRecordingsSvc(pub Arc); + impl + tonic::server::UnaryService + for RegisterRecordingsSvc + { + type Response = super::RegisterRecordingsResponse; + type Future = BoxFuture, tonic::Status>; + fn call( + &mut self, + request: tonic::Request, + ) -> Self::Future { + let inner = Arc::clone(&self.0); + let fut = async move { + ::register_recordings(&inner, request).await + }; + Box::pin(fut) + } + } + let accept_compression_encodings = self.accept_compression_encodings; + let send_compression_encodings = self.send_compression_encodings; + let max_decoding_message_size = self.max_decoding_message_size; + let max_encoding_message_size = self.max_encoding_message_size; + let inner = self.inner.clone(); + let fut = async move { + let method = RegisterRecordingsSvc(inner); + let codec = tonic::codec::ProstCodec::default(); + let mut grpc = tonic::server::Grpc::new(codec) + .apply_compression_config( + accept_compression_encodings, + send_compression_encodings, + ) + .apply_max_message_size_config( + max_decoding_message_size, + max_encoding_message_size, + ); + let res = grpc.unary(method, req).await; + Ok(res) + }; + Box::pin(fut) + } + _ => Box::pin(async move { + let mut response = http::Response::new(empty_body()); + let headers = response.headers_mut(); + headers.insert( + tonic::Status::GRPC_STATUS, + (tonic::Code::Unimplemented as i32).into(), + ); + headers.insert( + http::header::CONTENT_TYPE, + tonic::metadata::GRPC_CONTENT_TYPE, + ); + Ok(response) + }), + } + } + } + impl Clone for StorageNodeServer { + fn clone(&self) -> Self { + let inner = self.inner.clone(); + Self { + inner, + accept_compression_encodings: self.accept_compression_encodings, + send_compression_encodings: self.send_compression_encodings, + max_decoding_message_size: self.max_decoding_message_size, + max_encoding_message_size: self.max_encoding_message_size, + } + } + } + /// Generated gRPC service name + pub const SERVICE_NAME: &str = "rerun.remote_store.v0.StorageNode"; + impl tonic::server::NamedService for StorageNodeServer { + const NAME: &'static str = SERVICE_NAME; + } +} diff --git a/crates/store/re_ws_comms/src/client.rs b/crates/store/re_ws_comms/src/client.rs index 7d793fc9648c..e3bb9568dde7 100644 --- a/crates/store/re_ws_comms/src/client.rs +++ b/crates/store/re_ws_comms/src/client.rs @@ -1,4 +1,4 @@ -use std::ops::ControlFlow; +use std::{ops::ControlFlow, time::Duration}; use ewebsock::{WsEvent, WsMessage}; @@ -16,6 +16,7 @@ pub fn viewer_to_server( // We set a very high limit, because we should be able to trust the server. // See https://github.com/rerun-io/rerun/issues/5268 for more max_incoming_frame_size: 2 * gigs, + delay_blocking: Duration::from_millis(10), }; ewebsock::ws_receive( diff --git a/deny.toml b/deny.toml index d99c623c3e5d..467180881539 100644 --- a/deny.toml +++ b/deny.toml @@ -60,7 +60,6 @@ skip = [ { name = "raw-window-handle" }, # Pretty small crate; some crates still on old version { name = "redox_syscall" }, # Plenty of versions in the wild { name = "spin" }, # Old version used by rusttls - { name = "webpki-roots" }, # ureq and tungstenite are on different version 😭 ] skip-tree = [ { name = "async-io" }, # Old version via rfd @@ -71,6 +70,7 @@ skip-tree = [ { name = "prost-derive" }, # only used in objectron example { name = "toml_edit" }, # Old version via egui-winit, newer used by rustdoc-json { name = "windows" }, # Old version used by accesskit_windows, newer version used by wgpu + { name = "tower" }, # tonic depends on 0.4.3, but also transitively (axum) on 0.5.1 ] diff --git a/pixi.toml b/pixi.toml index 56ab2208126c..019453bab53c 100644 --- a/pixi.toml +++ b/pixi.toml @@ -114,6 +114,9 @@ examples-pypi = ["examples-common", "python-pypi"] # Run the codegen. Optionally pass `--profile` argument if you want. codegen = "cargo --quiet run --package re_types_builder -- " +# Run the codegen for remote store types. +codegen-rstore = "cargo --quiet run --package re_remote_store_types_builder -- " + # Generate the Rerun CLI manual. # NOTE:must be --all-features, otherwise we might miss some optional commands. diff --git a/scripts/lint.py b/scripts/lint.py index ce9a83ec8392..5926bc3c0dc1 100755 --- a/scripts/lint.py +++ b/scripts/lint.py @@ -1181,6 +1181,7 @@ def main() -> None: "./.pytest_cache", "./CODE_STYLE.md", "./crates/build/re_types_builder/src/reflection.rs", # auto-generated + "./crates/store/re_remote_store_types/src/v0/rerun.remote_store.v0.rs", # auto-generated "./docs/content/reference/cli.md", # auto-generated "./examples/assets", "./examples/python/detect_and_track_objects/cache/version.txt",