From 2f160f6a15e7d7827ee58e5b53279d837abfbb11 Mon Sep 17 00:00:00 2001 From: peasee <98815791+peasee@users.noreply.github.com> Date: Wed, 18 Sep 2024 21:04:05 +1000 Subject: [PATCH 01/10] feat: Add spice user agent, update for clippy rules --- .github/workflows/build.yml | 86 +++++++++++++++++++++++++++++++------ Cargo.toml | 3 ++ README.md | 2 +- src/client.rs | 22 ++++++++++ src/config.rs | 64 +++++++++++++++++++++++++++ src/flight.rs | 9 +++- tests/client_test.rs | 12 +++--- tests/readme_test.rs | 35 +++++++++------ 8 files changed, 199 insertions(+), 34 deletions(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index bdef0ff..006bbfa 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -10,15 +10,15 @@ env: CARGO_TERM_COLOR: always jobs: - build_and_test: + build_toolchains: runs-on: ubuntu-latest - strategy: - matrix: - toolchain: - - stable - - beta - - nightly - + strategy: + fail-fast: false + matrix: + toolchain: + - stable + - beta + - nightly steps: - uses: actions/checkout@v4 @@ -45,24 +45,84 @@ jobs: sarif_file: rust-clippy-results.sarif wait-for-processing: true - - name: Build - run: cargo build --verbose + build_and_test: + runs-on: ubuntu-latest + strategy: + matrix: + os: [ubuntu-latest, macos-latest, windows-latest] + toolchain: + - stable + - beta + + steps: + - uses: actions/checkout@v4 + + - name: Install Rust toolchain + uses: actions-rs/toolchain@v1 + with: + profile: minimal + toolchain: ${{ matrix.toolchain }} + components: clippy + override: true + + - name: Cargo clippy + run: cargo clippy --all-features - - name: install Spice (https://install.spiceai.org) + - name: Install Spice (https://install.spiceai.org) (Linux) + if: matrix.os == 'ubuntu-latest' + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} run: | curl https://install.spiceai.org | /bin/bash echo "$HOME/.spice/bin" >> $GITHUB_PATH + $HOME/.spice/bin/spice install + + - name: Install Spice (https://install.spiceai.org) (MacOS) + if: matrix.os == 'macos-latest' + run: | + brew install spiceai/spiceai/spice + brew install spiceai/spiceai/spiced + + - name: install Spice (Windows) + if: matrix.os == 'windows-latest' + run: | + curl -L "https://install.spiceai.org/Install.ps1" -o Install.ps1 && PowerShell -ExecutionPolicy Bypass -File ./Install.ps1 + + - name: add Spice bin to PATH (Windows) + if: matrix.os == 'windows-latest' + run: | + Add-Content $env:GITHUB_PATH (Join-Path $HOME ".spice\bin") + shell: pwsh - name: Init and start spice app + if: matrix.os != 'windows-latest' + run: | + spice init spice_qs + cd spice_qs + spice add spiceai/quickstart + spiced &> spice.log & + # time to initialize added dataset + sleep 5 + + - name: Init and start spice app (Windows) + if: matrix.os == 'windows-latest' run: | spice init spice_qs cd spice_qs spice add spiceai/quickstart - spice run &> spice.log & + Start-Process -FilePath spice run # time to initialize added dataset - sleep 10 + Start-Sleep -Seconds 5 + shell: pwsh - name: Run tests run: cargo test --verbose env: API_KEY: ${{ secrets.TEST_API_KEY }} + + - name: Stop spice and check logs + working-directory: spice_qs + if: matrix.os != 'windows-latest' && always() + run: | + killall spice || true + cat spice.log diff --git a/Cargo.toml b/Cargo.toml index 86cc3c6..48c4283 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -27,3 +27,6 @@ dotenv = "0.15.0" arrow = "51.0.0" futures = "0.3.30" base64 = "0.22.0" + +[dev-dependencies] +regex = "1.10.6" \ No newline at end of file diff --git a/README.md b/README.md index 0448660..2e56390 100644 --- a/README.md +++ b/README.md @@ -33,7 +33,7 @@ async fn main() { } ``` -### New client with https://spice.ai cloud +### New client with cloud ```rust use spiceai::ClientBuilder; diff --git a/src/client.rs b/src/client.rs index 034a9f1..0aeea71 100644 --- a/src/client.rs +++ b/src/client.rs @@ -49,6 +49,10 @@ impl SpiceClient { /// let mut client = Client::new("API_KEY").await.unwrap(); /// } /// ``` + /// + /// ## Errors + /// + /// - `Box` for any query error pub async fn new(api_key: &str) -> Result> { let config = SpiceClientConfig::load_from_default().await?; @@ -58,6 +62,7 @@ impl SpiceClient { }) } + #[must_use] pub fn builder() -> SpiceClientBuilder { SpiceClientBuilder::new() } @@ -72,6 +77,10 @@ impl SpiceClient { /// let data = client.query("SELECT * FROM eth.recent_blocks LIMIT 10;").await; /// # } /// ```` + /// + /// ## Errors + /// + /// - `Box` for any query error pub async fn query(&mut self, query: &str) -> Result> { self.flight.query(query).await } @@ -86,6 +95,10 @@ impl SpiceClient { /// let data = client.fire_query("SELECT * FROM eth.recent_blocks LIMIT 10;").await; /// # } /// ```` + /// + /// ## Errors + /// + /// - `Box` for any query error pub async fn fire_query( &mut self, query: &str, @@ -138,6 +151,7 @@ impl Default for SpiceClientBuilder { } impl SpiceClientBuilder { + #[must_use] pub fn new() -> Self { Self { api_key: None, @@ -147,18 +161,21 @@ impl SpiceClientBuilder { } /// Configures the `SpiceClient` to use the given API key. + #[must_use] pub fn api_key(mut self, api_key: &str) -> Self { self.api_key = Some(api_key.to_string()); self } /// Configures the `SpiceClient` to use the given Spice Firecache endpoint. + #[must_use] pub fn firecache_url(mut self, firecache_url: &str) -> Self { self.firecache_url = Some(firecache_url.to_string()); self } /// Configures the `SpiceClient` to use the given Spice Flight endpoint. + #[must_use] pub fn flight_url(mut self, flight_url: &str) -> Self { self.flight_url = Some(flight_url.to_string()); self @@ -166,6 +183,7 @@ impl SpiceClientBuilder { /// Configures the `SpiceClient` to use default Spice.ai Cloud endpoints. /// Equivalent to calling `.firecache_url("https://firecache.spiceai.io")` and `.flight_url("https://flight.spiceai.io")`. + #[must_use] pub fn use_spiceai_cloud(mut self) -> Self { self.flight_url = Some(SPICE_CLOUD_FLIGHT_ADDR.to_string()); self.firecache_url = Some(SPICE_CLOUD_FIRECACHE_ADDR.to_string()); @@ -173,6 +191,10 @@ impl SpiceClientBuilder { } /// Builds the `SpiceClient` with the specified configuration. + /// + /// ## Errors + /// + /// - `Box` if flight or firecache channel creation fails pub async fn build(self) -> Result> { let flight_channel = match self.flight_url { Some(url) => new_tls_flight_channel(&url).await?, diff --git a/src/config.rs b/src/config.rs index 3580e3d..81e14f7 100644 --- a/src/config.rs +++ b/src/config.rs @@ -3,3 +3,67 @@ pub const SPICE_CLOUD_FIRECACHE_ADDR: &str = "https://firecache.spiceai.io"; // default address for local spice runtime pub const SPICE_LOCAL_FLIGHT_ADDR: &str = "http://localhost:50051"; + +#[cfg(target_family = "unix")] +fn get_os_release() -> Result> { + // call uname -r to get release text + use std::process::Command; + let output = Command::new("uname").arg("-r").output()?; + let release = String::from_utf8(output.stdout)?; + + Ok(release) +} + +#[cfg(target_family = "windows")] +fn get_os_release() -> Result> { + todo!("get_os_release not implemented for Windows") +} + +pub(crate) fn get_user_agent() -> String { + let os_type = std::env::consts::OS; + let os_type = if os_type.is_empty() { + "unknown".to_string() + } else { + // capitalize first letter + let mut os_type = os_type.to_string(); + os_type[..1].make_ascii_uppercase(); + os_type + }; + + let os_arch = std::env::consts::ARCH; + let os_arch = match os_arch { + "" => "unknown".to_string(), + "x86" => "i386".to_string(), + _ => os_arch.to_string(), + }; + + let os_release = get_os_release() + .unwrap_or_else(|_| "unknown".to_string()) + .trim() + .to_string(); + + format!( + "spice-rs {} ({os_type}/{os_release} {os_arch})", + env!("CARGO_PKG_VERSION") + ) +} + +#[cfg(test)] +mod test { + use super::*; + + #[test] + fn test_get_user_agent() { + let matching_regex = regex::Regex::new( + r"spice-rs \d+\.\d+\.\d+ \((Linux|Windows|macOS)/[\d\w\.\-\_]+ (x86_64|aarch64|i386)\)", + ) + .expect("regex should be constructed"); + + let user_agent = get_user_agent(); + let agent_matches = matching_regex.is_match(&user_agent); + assert!( + agent_matches, + "expected user agent to match regex, but got {user_agent}" + ); + } +} diff --git a/src/flight.rs b/src/flight.rs index 8414bd4..caf5544 100644 --- a/src/flight.rs +++ b/src/flight.rs @@ -1,3 +1,4 @@ +use crate::config::get_user_agent; use arrow::error::ArrowError; use arrow_flight::decode::FlightRecordBatchStream; use arrow_flight::error::FlightError; @@ -24,16 +25,20 @@ pub struct SqlFlightClient { api_key: Option, } +#[allow(clippy::needless_pass_by_value)] fn status_to_arrow_error(status: tonic::Status) -> ArrowError { ArrowError::IpcError(format!("{status:?}")) } impl SqlFlightClient { pub fn new(chan: Channel, api_key: Option) -> Self { + let mut headers = HashMap::new(); + headers.insert("x-spice-user-agent".to_string(), get_user_agent()); + SqlFlightClient { api_key, + headers, client: FlightServiceClient::new(chan), - headers: HashMap::default(), token: None, } } @@ -41,7 +46,7 @@ impl SqlFlightClient { async fn handshake(&mut self, username: &str, password: &str) -> Result { let cmd = HandshakeRequest { protocol_version: 0, - payload: Default::default(), + payload: Bytes::default(), }; let mut req = tonic::Request::new(stream::iter(vec![cmd])); let val = BASE64_STANDARD.encode(format!("{username}:{password}")); diff --git a/tests/client_test.rs b/tests/client_test.rs index b6c7e1e..d1e30bf 100644 --- a/tests/client_test.rs +++ b/tests/client_test.rs @@ -41,13 +41,13 @@ mod tests { assert_eq!(batch.num_rows(), 10); }, Err(e) => { - panic!("Error: {}", e) + panic!("Error: {e}") }, }; } } Err(e) => { - panic!("Error: {}", e); + panic!("Error: {e}"); } }; } @@ -67,13 +67,13 @@ mod tests { assert_eq!(batch.num_rows(), 10); }, Err(e) => { - panic!("Error: {}", e) + panic!("Error: {e}") } }; } } Err(e) => { - panic!("Error: {}", e); + panic!("Error: {e}"); } }; } @@ -95,7 +95,7 @@ mod tests { total_rows += batch.num_rows(); }, Err(e) => { - panic!("Error: {}", e) + panic!("Error: {e}") }, }; } @@ -103,7 +103,7 @@ mod tests { assert_ne!(num_batches, 1); } Err(e) => { - panic!("Error: {}", e); + panic!("Error: {e}"); } }; } diff --git a/tests/readme_test.rs b/tests/readme_test.rs index bb287c3..4340a12 100644 --- a/tests/readme_test.rs +++ b/tests/readme_test.rs @@ -11,13 +11,17 @@ mod tests { dotenv::from_path(Path::new(".env.local")).ok(); let api_key = env::var("API_KEY").expect("API_KEY not found"); - let mut client = Client::new(&api_key).await.unwrap(); + let mut client = Client::new(&api_key) + .await + .expect("SpiceClient should be created"); let data = client .query("SELECT * FROM eth.recent_blocks LIMIT 10;") .await; - if data.is_err() { - panic!("failed to query: {:#?}", data.expect_err("")) - } + assert!( + data.is_ok(), + "failed to query: {:#?}", + data.expect_err("should be an error") + ); } #[tokio::test] @@ -31,24 +35,31 @@ mod tests { .use_spiceai_cloud() .build() .await - .unwrap(); + .expect("SpiceClient should be created"); let data = client .query("SELECT * FROM eth.recent_blocks LIMIT 10;") .await; - if data.is_err() { - panic!("failed to query: {:#?}", data.expect_err("")) - } + assert!( + data.is_ok(), + "failed to query: {:#?}", + data.expect_err("should be an error") + ); } #[tokio::test] async fn test_readme_builder_local() { // NOTE: If you're changing the code below, make sure you update the README.md. - let mut client = ClientBuilder::new().build().await.unwrap(); + let mut client = ClientBuilder::new() + .build() + .await + .expect("SpiceClient should be created"); let data = client.query("select * from taxi_trips limit 3;").await; - if data.is_err() { - panic!("failed to query: {:#?}", data.expect_err("")) - } + assert!( + data.is_ok(), + "failed to query: {:#?}", + data.expect_err("should be an error") + ); } } From 390ecf82c0e159e2632803842f085d611687f97d Mon Sep 17 00:00:00 2001 From: peasee <98815791+peasee@users.noreply.github.com> Date: Wed, 18 Sep 2024 21:06:23 +1000 Subject: [PATCH 02/10] fix: Increase startup delay --- .github/workflows/build.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 006bbfa..e0ec41f 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -102,7 +102,7 @@ jobs: spice add spiceai/quickstart spiced &> spice.log & # time to initialize added dataset - sleep 5 + sleep 10 - name: Init and start spice app (Windows) if: matrix.os == 'windows-latest' @@ -112,7 +112,7 @@ jobs: spice add spiceai/quickstart Start-Process -FilePath spice run # time to initialize added dataset - Start-Sleep -Seconds 5 + Start-Sleep -Seconds 10 shell: pwsh - name: Run tests From 772cb18737b9a8da1b0de4a084243f3f0eac8a3c Mon Sep 17 00:00:00 2001 From: peasee <98815791+peasee@users.noreply.github.com> Date: Wed, 18 Sep 2024 21:09:31 +1000 Subject: [PATCH 03/10] fix: Build workflow --- .github/workflows/build.yml | 19 ++++++++++--------- 1 file changed, 10 insertions(+), 9 deletions(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index e0ec41f..929a914 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -11,14 +11,15 @@ env: jobs: build_toolchains: + name: Clippy on ${{ matrix.toolchain }} runs-on: ubuntu-latest - strategy: - fail-fast: false - matrix: - toolchain: - - stable - - beta - - nightly + strategy: + fail-fast: false + matrix: + toolchain: + - stable + - beta + - nightly steps: - uses: actions/checkout@v4 @@ -46,14 +47,14 @@ jobs: wait-for-processing: true build_and_test: - runs-on: ubuntu-latest + name: Build and test ${{matrix.os}} on ${{ matrix.toolchain }} + runs-on: ${{ matrix.os }} strategy: matrix: os: [ubuntu-latest, macos-latest, windows-latest] toolchain: - stable - beta - steps: - uses: actions/checkout@v4 From f52aa42b5fa30f457f793e801f2e460b36dfb784 Mon Sep 17 00:00:00 2001 From: peasee <98815791+peasee@users.noreply.github.com> Date: Wed, 18 Sep 2024 21:12:00 +1000 Subject: [PATCH 04/10] fix: Turn off verbose tests and builds --- .github/workflows/build.yml | 2 +- .github/workflows/publish.yml | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 929a914..406de68 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -117,7 +117,7 @@ jobs: shell: pwsh - name: Run tests - run: cargo test --verbose + run: cargo test env: API_KEY: ${{ secrets.TEST_API_KEY }} diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml index 63fde69..7eaa77e 100644 --- a/.github/workflows/publish.yml +++ b/.github/workflows/publish.yml @@ -20,8 +20,8 @@ jobs: toolchain: stable override: true - - run: cargo build --verbose + - run: cargo build - - run: cargo publish --verbose + - run: cargo publish env: CARGO_REGISTRY_TOKEN: ${{ secrets.CARGO_REGISTRY_TOKEN }} From d9722d0ec8b323a96284b01c5b4f64b0527931f4 Mon Sep 17 00:00:00 2001 From: peasee <98815791+peasee@users.noreply.github.com> Date: Wed, 18 Sep 2024 21:15:18 +1000 Subject: [PATCH 05/10] fix: Use a match arm for OS name --- .github/workflows/build.yml | 1 + src/config.rs | 20 +++++++++++++------- 2 files changed, 14 insertions(+), 7 deletions(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 406de68..47df165 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -51,6 +51,7 @@ jobs: runs-on: ${{ matrix.os }} strategy: matrix: + fail-fast: false os: [ubuntu-latest, macos-latest, windows-latest] toolchain: - stable diff --git a/src/config.rs b/src/config.rs index 81e14f7..7833350 100644 --- a/src/config.rs +++ b/src/config.rs @@ -21,13 +21,19 @@ fn get_os_release() -> Result> { pub(crate) fn get_user_agent() -> String { let os_type = std::env::consts::OS; - let os_type = if os_type.is_empty() { - "unknown".to_string() - } else { - // capitalize first letter - let mut os_type = os_type.to_string(); - os_type[..1].make_ascii_uppercase(); - os_type + let os_type = match os_type { + "" => "unknown".to_string(), + "macos" => "macOS".to_string(), + "linux" => "Linux".to_string(), + "windows" => "Windows".to_string(), + "ios" => "iOS".to_string(), + "android" => "Android".to_string(), + "freebsd" => "FreeBSD".to_string(), + "dragonfly" => "DragonFlyBSD".to_string(), + "netbsd" => "NetBSD".to_string(), + "openbsd" => "OpenBSD".to_string(), + "solaris" => "Solaris".to_string(), + _ => os_type.to_string(), }; let os_arch = std::env::consts::ARCH; From 3f814d80f075a2d1d5c5826a84b9997ba33f7e95 Mon Sep 17 00:00:00 2001 From: peasee <98815791+peasee@users.noreply.github.com> Date: Wed, 18 Sep 2024 21:16:32 +1000 Subject: [PATCH 06/10] fix: Put fail-fast in the right spot --- .github/workflows/build.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 47df165..8ffef2d 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -50,8 +50,8 @@ jobs: name: Build and test ${{matrix.os}} on ${{ matrix.toolchain }} runs-on: ${{ matrix.os }} strategy: + fail-fast: false matrix: - fail-fast: false os: [ubuntu-latest, macos-latest, windows-latest] toolchain: - stable From 2eae58d7946dd2745c2260492a276288828f7ec8 Mon Sep 17 00:00:00 2001 From: peasee <98815791+peasee@users.noreply.github.com> Date: Thu, 19 Sep 2024 08:56:16 +1000 Subject: [PATCH 07/10] test: Expect Darwin not macOS --- src/config.rs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/config.rs b/src/config.rs index 7833350..0e8fd21 100644 --- a/src/config.rs +++ b/src/config.rs @@ -23,7 +23,7 @@ pub(crate) fn get_user_agent() -> String { let os_type = std::env::consts::OS; let os_type = match os_type { "" => "unknown".to_string(), - "macos" => "macOS".to_string(), + "macos" => "Darwin".to_string(), "linux" => "Linux".to_string(), "windows" => "Windows".to_string(), "ios" => "iOS".to_string(), @@ -33,7 +33,7 @@ pub(crate) fn get_user_agent() -> String { "netbsd" => "NetBSD".to_string(), "openbsd" => "OpenBSD".to_string(), "solaris" => "Solaris".to_string(), - _ => os_type.to_string(), + _ => os_type.to_string(), }; let os_arch = std::env::consts::ARCH; @@ -61,7 +61,7 @@ mod test { #[test] fn test_get_user_agent() { let matching_regex = regex::Regex::new( - r"spice-rs \d+\.\d+\.\d+ \((Linux|Windows|macOS)/[\d\w\.\-\_]+ (x86_64|aarch64|i386)\)", + r"spice-rs \d+\.\d+\.\d+ \((Linux|Windows|Darwin)/[\d\w\.\-\_]+ (x86_64|aarch64|i386)\)", ) .expect("regex should be constructed"); From 6f04031e05c1096dd19f27ccb31b7095b29cb258 Mon Sep 17 00:00:00 2001 From: peasee <98815791+peasee@users.noreply.github.com> Date: Thu, 19 Sep 2024 17:27:07 +1000 Subject: [PATCH 08/10] feat: Use winver for Windows release version --- Cargo.toml | 3 +++ src/config.rs | 4 +++- 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/Cargo.toml b/Cargo.toml index 48c4283..2ba9993 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -28,5 +28,8 @@ arrow = "51.0.0" futures = "0.3.30" base64 = "0.22.0" +[target.'cfg(windows)'.dependencies] +winver = "1.0.0" + [dev-dependencies] regex = "1.10.6" \ No newline at end of file diff --git a/src/config.rs b/src/config.rs index 0e8fd21..f70f549 100644 --- a/src/config.rs +++ b/src/config.rs @@ -16,7 +16,9 @@ fn get_os_release() -> Result> { #[cfg(target_family = "windows")] fn get_os_release() -> Result> { - todo!("get_os_release not implemented for Windows") + use winver::WindowsVersion; + let version = WindowsVersion::get()?; + Ok(version.to_string()) } pub(crate) fn get_user_agent() -> String { From aab717f4530c65c386e731adb72a86b94fbf19e5 Mon Sep 17 00:00:00 2001 From: peasee <98815791+peasee@users.noreply.github.com> Date: Thu, 19 Sep 2024 17:34:26 +1000 Subject: [PATCH 09/10] fix: Expect option for WindowsVersion --- src/config.rs | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/src/config.rs b/src/config.rs index f70f549..bbb2cef 100644 --- a/src/config.rs +++ b/src/config.rs @@ -17,8 +17,11 @@ fn get_os_release() -> Result> { #[cfg(target_family = "windows")] fn get_os_release() -> Result> { use winver::WindowsVersion; - let version = WindowsVersion::get()?; - Ok(version.to_string()) + if let Some(version) = WindowsVersion::detect() { + Ok(version.to_string()) + } else { + Ok("unknown".to_string()) + } } pub(crate) fn get_user_agent() -> String { From 2819dd912db7934dc7993e357758c6b22b6a23b2 Mon Sep 17 00:00:00 2001 From: peasee <98815791+peasee@users.noreply.github.com> Date: Thu, 19 Sep 2024 18:13:09 +1000 Subject: [PATCH 10/10] refactor: Update Box to add Send + Sync --- src/client.rs | 23 ++++++++++++----------- src/config.rs | 6 ++++-- src/flight.rs | 6 +++--- src/tls.rs | 7 ++++--- 4 files changed, 23 insertions(+), 19 deletions(-) diff --git a/src/client.rs b/src/client.rs index 0aeea71..35e3989 100644 --- a/src/client.rs +++ b/src/client.rs @@ -1,11 +1,12 @@ use crate::{ - config::{SPICE_CLOUD_FIRECACHE_ADDR, SPICE_CLOUD_FLIGHT_ADDR, SPICE_LOCAL_FLIGHT_ADDR}, + config::{ + GenericError, SPICE_CLOUD_FIRECACHE_ADDR, SPICE_CLOUD_FLIGHT_ADDR, SPICE_LOCAL_FLIGHT_ADDR, + }, flight::SqlFlightClient, tls::new_tls_flight_channel, }; use arrow_flight::decode::FlightRecordBatchStream; use futures::try_join; -use std::error::Error; use tonic::transport::Channel; struct SpiceClientConfig { @@ -21,7 +22,7 @@ impl SpiceClientConfig { } } - pub async fn load_from_default() -> Result> { + pub async fn load_from_default() -> Result { let (flight_chan, firecache_chan) = try_join!( new_tls_flight_channel(SPICE_CLOUD_FLIGHT_ADDR), new_tls_flight_channel(SPICE_CLOUD_FIRECACHE_ADDR) @@ -52,8 +53,8 @@ impl SpiceClient { /// /// ## Errors /// - /// - `Box` for any query error - pub async fn new(api_key: &str) -> Result> { + /// - `Box` for any query error + pub async fn new(api_key: &str) -> Result { let config = SpiceClientConfig::load_from_default().await?; Ok(Self { @@ -80,8 +81,8 @@ impl SpiceClient { /// /// ## Errors /// - /// - `Box` for any query error - pub async fn query(&mut self, query: &str) -> Result> { + /// - `Box` for any query error + pub async fn query(&mut self, query: &str) -> Result { self.flight.query(query).await } @@ -98,11 +99,11 @@ impl SpiceClient { /// /// ## Errors /// - /// - `Box` for any query error + /// - `Box` for any query error pub async fn fire_query( &mut self, query: &str, - ) -> Result> { + ) -> Result { self.firecache.query(query).await } } @@ -194,8 +195,8 @@ impl SpiceClientBuilder { /// /// ## Errors /// - /// - `Box` if flight or firecache channel creation fails - pub async fn build(self) -> Result> { + /// - `Box` if flight or firecache channel creation fails + pub async fn build(self) -> Result { let flight_channel = match self.flight_url { Some(url) => new_tls_flight_channel(&url).await?, None => new_tls_flight_channel(SPICE_LOCAL_FLIGHT_ADDR).await?, diff --git a/src/config.rs b/src/config.rs index bbb2cef..9cafb19 100644 --- a/src/config.rs +++ b/src/config.rs @@ -4,8 +4,10 @@ pub const SPICE_CLOUD_FIRECACHE_ADDR: &str = "https://firecache.spiceai.io"; // default address for local spice runtime pub const SPICE_LOCAL_FLIGHT_ADDR: &str = "http://localhost:50051"; +pub type GenericError = Box; + #[cfg(target_family = "unix")] -fn get_os_release() -> Result> { +fn get_os_release() -> Result { // call uname -r to get release text use std::process::Command; let output = Command::new("uname").arg("-r").output()?; @@ -15,7 +17,7 @@ fn get_os_release() -> Result> { } #[cfg(target_family = "windows")] -fn get_os_release() -> Result> { +fn get_os_release() -> Result { use winver::WindowsVersion; if let Some(version) = WindowsVersion::detect() { Ok(version.to_string()) diff --git a/src/flight.rs b/src/flight.rs index caf5544..e569a18 100644 --- a/src/flight.rs +++ b/src/flight.rs @@ -1,4 +1,5 @@ use crate::config::get_user_agent; +use crate::config::GenericError; use arrow::error::ArrowError; use arrow_flight::decode::FlightRecordBatchStream; use arrow_flight::error::FlightError; @@ -12,7 +13,6 @@ use bytes::Bytes; use futures::stream; use futures::TryStreamExt; use std::collections::HashMap; -use std::error::Error; use std::str::FromStr; use tonic::metadata::AsciiMetadataKey; use tonic::transport::Channel; @@ -86,7 +86,7 @@ impl SqlFlightClient { Ok(resp) } - async fn authenticate(&mut self, api_key: &str) -> std::result::Result<(), Box> { + async fn authenticate(&mut self, api_key: &str) -> std::result::Result<(), GenericError> { if api_key.split('|').collect::().len() < 2 { return Err("Invalid API key format".into()); } @@ -119,7 +119,7 @@ impl SqlFlightClient { pub async fn query( &mut self, query: &str, - ) -> std::result::Result> { + ) -> std::result::Result { let api_key = self.api_key.clone(); if let Some(api_key) = api_key { self.authenticate(&api_key).await?; diff --git a/src/tls.rs b/src/tls.rs index 3e043b8..c72ad5c 100644 --- a/src/tls.rs +++ b/src/tls.rs @@ -1,9 +1,10 @@ -use std::error::Error; use std::str::FromStr; use tonic::transport::channel::{ClientTlsConfig, Endpoint}; use tonic::transport::Channel; -pub fn system_tls_certificate() -> Result> { +use crate::config::GenericError; + +pub fn system_tls_certificate() -> Result { // Load root certificates found in the platform’s native certificate store. let certs = rustls_native_certs::load_native_certs()?; @@ -19,7 +20,7 @@ pub fn system_tls_certificate() -> Result Result> { +pub async fn new_tls_flight_channel(https_url: &str) -> Result { let mut endpoint = Endpoint::from_str(https_url)?; if https_url.starts_with("https://") {