Skip to content

Commit

Permalink
refactor(turborepo): API Client Cleanup (#5084)
Browse files Browse the repository at this point in the history
### Description

I had this inside the HTTP cache PR, but it makes sense as a separate
PR.

Refactors the API client in the following ways:

- Changes the retry logic to use try_clone from reqwest instead of the
awkward (and unnecessarily general) `retry_future` function that relied
on a closure that generates a Rust future.
- Use a crate Error type with thiserror instead of anyhow. This is
pretty standard.
- Factor out some common logic into helper functions

### Testing Instructions

Existing tests should suffice.

Co-authored-by: --global <Nicholas Yang>
  • Loading branch information
NicholasLYang authored May 31, 2023
1 parent 945756e commit 17de157
Show file tree
Hide file tree
Showing 6 changed files with 143 additions and 165 deletions.
2 changes: 2 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 2 additions & 0 deletions crates/turborepo-api-client/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -15,4 +15,6 @@ chrono = { workspace = true, features = ["serde"] }
reqwest = { workspace = true, features = ["json"] }
rustc_version_runtime = "0.2.1"
serde = { workspace = true }
thiserror = { workspace = true }
tokio = { workspace = true }
url = { workspace = true }
34 changes: 34 additions & 0 deletions crates/turborepo-api-client/src/error.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
use std::backtrace::Backtrace;

use reqwest::header::ToStrError;
use thiserror::Error;

use crate::CachingStatus;

#[derive(Debug, Error)]
pub enum Error {
#[error("Error making HTTP request: {0}")]
ReqwestError(#[from] reqwest::Error),
#[error("skipping HTTP Request, too many failures have occurred.\nLast error: {0}")]
TooManyFailures(#[from] Box<reqwest::Error>),
#[error("Error parsing header: {0}")]
InvalidHeader(#[from] ToStrError),
#[error("Error parsing URL: {0}")]
InvalidUrl(#[from] url::ParseError),
#[error("unknown caching status: {0}")]
UnknownCachingStatus(String, #[backtrace] Backtrace),
#[error("unknown status {code}: {message}")]
UnknownStatus {
code: String,
message: String,
#[backtrace]
backtrace: Backtrace,
},
#[error("{message}")]
CacheDisabled {
status: CachingStatus,
message: String,
},
}

pub type Result<T> = std::result::Result<T, Error>;
212 changes: 72 additions & 140 deletions crates/turborepo-api-client/src/lib.rs
Original file line number Diff line number Diff line change
@@ -1,11 +1,15 @@
use std::{env, future::Future};
#![feature(async_closure)]
#![feature(provide_any)]
#![feature(error_generic_member_access)]

use anyhow::{anyhow, Result};
use reqwest::StatusCode;
use std::env;

use reqwest::RequestBuilder;
use serde::{Deserialize, Serialize};

use crate::retry::retry_future;
pub use crate::error::{Error, Result};

mod error;
mod retry;

#[derive(Debug, Clone, Deserialize)]
Expand Down Expand Up @@ -115,54 +119,33 @@ pub struct APIClient {

impl APIClient {
pub async fn get_user(&self, token: &str) -> Result<UserResponse> {
let response = self
.make_retryable_request(|| {
let url = self.make_url("/v2/user");
let request_builder = self
.client
.get(url)
.header("User-Agent", self.user_agent.clone())
.header("Authorization", format!("Bearer {}", token))
.header("Content-Type", "application/json");

request_builder.send()
})
let url = self.make_url("/v2/user");
let request_builder = self
.client
.get(url)
.header("User-Agent", self.user_agent.clone())
.header("Authorization", format!("Bearer {}", token))
.header("Content-Type", "application/json");
let response = retry::make_retryable_request(request_builder)
.await?
.error_for_status()?;

response.json().await.map_err(|err| {
anyhow!(
"Error getting user: {}",
err.status()
.and_then(|status| status.canonical_reason())
.unwrap_or(&err.to_string())
)
})
Ok(response.json().await?)
}

pub async fn get_teams(&self, token: &str) -> Result<TeamsResponse> {
let response = self
.make_retryable_request(|| {
let request_builder = self
.client
.get(self.make_url("/v2/teams?limit=100"))
.header("User-Agent", self.user_agent.clone())
.header("Content-Type", "application/json")
.header("Authorization", format!("Bearer {}", token));

request_builder.send()
})
let request_builder = self
.client
.get(self.make_url("/v2/teams?limit=100"))
.header("User-Agent", self.user_agent.clone())
.header("Content-Type", "application/json")
.header("Authorization", format!("Bearer {}", token));

let response = retry::make_retryable_request(request_builder)
.await?
.error_for_status()?;

response.json().await.map_err(|err| {
anyhow!(
"Error getting teams: {}",
err.status()
.and_then(|status| status.canonical_reason())
.unwrap_or(&err.to_string())
)
})
Ok(response.json().await?)
}

pub async fn get_team(&self, token: &str, team_id: &str) -> Result<Option<Team>> {
Expand All @@ -177,14 +160,22 @@ impl APIClient {
.await?
.error_for_status()?;

response.json().await.map_err(|err| {
anyhow!(
"Error getting team: {}",
err.status()
.and_then(|status| status.canonical_reason())
.unwrap_or(&err.to_string())
)
})
Ok(response.json().await?)
}

fn add_team_params(
mut request_builder: RequestBuilder,
team_id: &str,
team_slug: Option<&str>,
) -> RequestBuilder {
if let Some(slug) = team_slug {
request_builder = request_builder.query(&[("teamSlug", slug)]);
}
if team_id.starts_with("team_") {
request_builder = request_builder.query(&[("teamId", team_id)]);
}

request_builder
}

pub async fn get_caching_status(
Expand All @@ -193,35 +184,20 @@ impl APIClient {
team_id: &str,
team_slug: Option<&str>,
) -> Result<CachingStatusResponse> {
let response = self
.make_retryable_request(|| {
let mut request_builder = self
.client
.get(self.make_url("/v8/artifacts/status"))
.header("User-Agent", self.user_agent.clone())
.header("Content-Type", "application/json")
.header("Authorization", format!("Bearer {}", token));

if let Some(slug) = team_slug {
request_builder = request_builder.query(&[("teamSlug", slug)]);
}
if team_id.starts_with("team_") {
request_builder = request_builder.query(&[("teamId", team_id)]);
}

request_builder.send()
})
let request_builder = self
.client
.get(self.make_url("/v8/artifacts/status"))
.header("User-Agent", self.user_agent.clone())
.header("Content-Type", "application/json")
.header("Authorization", format!("Bearer {}", token));

let request_builder = Self::add_team_params(request_builder, team_id, team_slug);

let response = retry::make_retryable_request(request_builder)
.await?
.error_for_status()?;

response.json().await.map_err(|err| {
anyhow!(
"Error getting caching status: {}",
err.status()
.and_then(|status| status.canonical_reason())
.unwrap_or(&err.to_string())
)
})
Ok(response.json().await?)
}

pub async fn get_spaces(&self, token: &str, team_id: Option<&str>) -> Result<SpacesResponse> {
Expand All @@ -231,84 +207,40 @@ impl APIClient {
None => "/v0/spaces?limit=100".to_string(),
};

let response = self
.make_retryable_request(|| {
let request_builder = self
.client
.get(self.make_url(endpoint.as_str()))
.header("User-Agent", self.user_agent.clone())
.header("Content-Type", "application/json")
.header("Authorization", format!("Bearer {}", token));

request_builder.send()
})
let request_builder = self
.client
.get(self.make_url(endpoint.as_str()))
.header("User-Agent", self.user_agent.clone())
.header("Content-Type", "application/json")
.header("Authorization", format!("Bearer {}", token));

let response = retry::make_retryable_request(request_builder)
.await?
.error_for_status()?;

response.json().await.map_err(|err| {
anyhow!(
"Error getting spaces: {}",
err.status()
.and_then(|status| status.canonical_reason())
.unwrap_or(&err.to_string())
)
})
Ok(response.json().await?)
}

pub async fn verify_sso_token(&self, token: &str, token_name: &str) -> Result<VerifiedSsoUser> {
let response = self
.make_retryable_request(|| {
let request_builder = self
.client
.get(self.make_url("/registration/verify"))
.query(&[("token", token), ("tokenName", token_name)])
.header("User-Agent", self.user_agent.clone());

request_builder.send()
})
let request_builder = self
.client
.get(self.make_url("/registration/verify"))
.query(&[("token", token), ("tokenName", token_name)])
.header("User-Agent", self.user_agent.clone());

let response = retry::make_retryable_request(request_builder)
.await?
.error_for_status()?;

let verification_response: VerificationResponse = response.json().await.map_err(|err| {
anyhow!(
"Error verifying token: {}",
err.status()
.and_then(|status| status.canonical_reason())
.unwrap_or(&err.to_string())
)
})?;
let verification_response: VerificationResponse = response.json().await?;

Ok(VerifiedSsoUser {
token: verification_response.token,
team_id: verification_response.team_id,
})
}

const RETRY_MAX: u32 = 2;

async fn make_retryable_request<
F: Future<Output = Result<reqwest::Response, reqwest::Error>>,
>(
&self,
request_builder: impl Fn() -> F,
) -> Result<reqwest::Response> {
retry_future(Self::RETRY_MAX, request_builder, Self::should_retry_request).await
}

fn should_retry_request(error: &reqwest::Error) -> bool {
if let Some(status) = error.status() {
if status == StatusCode::TOO_MANY_REQUESTS {
return true;
}

if status.as_u16() >= 500 && status.as_u16() != 501 {
return true;
}
}

false
}

pub fn new(base_url: impl AsRef<str>, timeout: u64, version: &'static str) -> Result<Self> {
pub fn new(base_url: impl AsRef<str>, timeout: u64, version: &str) -> Result<Self> {
let client = if timeout != 0 {
reqwest::Client::builder()
.timeout(std::time::Duration::from_secs(timeout))
Expand Down
Loading

0 comments on commit 17de157

Please sign in to comment.