Add Projects

This commit is contained in:
Joshua Coles 2024-07-15 16:38:05 +01:00
parent 06de91a6fd
commit f447cabed1
4 changed files with 80 additions and 2 deletions

12
Cargo.lock generated
View File

@ -1268,6 +1268,17 @@ dependencies = [
"serde", "serde",
] ]
[[package]]
name = "serde_json_path_to_error"
version = "0.1.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d7f2c2432eb04d880635fbf5d45cd5e619aeca7d4aff62cc1699671512db7d53"
dependencies = [
"serde",
"serde_json",
"serde_path_to_error",
]
[[package]] [[package]]
name = "serde_path_to_error" name = "serde_path_to_error"
version = "0.1.16" version = "0.1.16"
@ -1450,6 +1461,7 @@ dependencies = [
"reqwest-retry", "reqwest-retry",
"serde", "serde",
"serde_json", "serde_json",
"serde_json_path_to_error",
"thiserror", "thiserror",
"tokio", "tokio",
] ]

View File

@ -16,3 +16,4 @@ tokio = { version = "1.38.0", features = ["full"] }
base64 = "0.22.1" base64 = "0.22.1"
serde = { version = "1.0.204", features = ["derive"] } serde = { version = "1.0.204", features = ["derive"] }
serde_json = "1.0.120" serde_json = "1.0.120"
serde_json_path_to_error = "0.1.4"

View File

@ -10,5 +10,5 @@ async fn main() {
sensitive::WORKSPACE_ID, sensitive::WORKSPACE_ID,
); );
dbg!(api.get_current_time_entry().await); dbg!(api.get_projects().await);
} }

View File

@ -9,6 +9,8 @@ use governor::clock::DefaultClock;
use reqwest::header::{HeaderMap, HeaderValue}; use reqwest::header::{HeaderMap, HeaderValue};
use base64::engine::general_purpose::STANDARD; use base64::engine::general_purpose::STANDARD;
use base64::Engine; use base64::Engine;
use reqwest::Response;
use serde::de::DeserializeOwned;
struct ReqwestRateLimiter { struct ReqwestRateLimiter {
rate_limiter: governor::RateLimiter<NotKeyed, InMemoryState, DefaultClock>, rate_limiter: governor::RateLimiter<NotKeyed, InMemoryState, DefaultClock>,
@ -66,6 +68,7 @@ impl TogglApi {
let mut value = HeaderValue::from_str(&format!("Basic {}", toggl_auth)).unwrap(); let mut value = HeaderValue::from_str(&format!("Basic {}", toggl_auth)).unwrap();
value.set_sensitive(true); value.set_sensitive(true);
headers.insert("Authorization", value); headers.insert("Authorization", value);
headers.insert("Content-Type", HeaderValue::from_static("application/json"));
headers headers
} }
@ -81,10 +84,35 @@ impl TogglApi {
.await? .await?
.json().await?) .json().await?)
} }
pub async fn get_projects(&self) -> Result<Vec<types::Project>, TogglError> {
let url = format!(
"{base_url}/workspaces/{workspace_id}/projects",
base_url = BASE_URL,
workspace_id = self.workspace_id
);
Self::parse(self.client.get(&url)
.headers(self.headers.clone())
.send().await?).await
}
async fn parse<T: DeserializeOwned>(response: Response) -> Result<T, TogglError> {
let data = response.text().await?;
let result = serde_json_path_to_error::from_str(&data);
let result = result
.map_err(|error| TogglError::JsonWithDataError {
data,
error,
})?;
Ok(result)
}
} }
mod types { mod types {
use chrono::{DateTime, Utc}; use chrono::{DateTime, NaiveDate, Utc};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
#[derive(Serialize, Deserialize, Debug, Clone)] #[derive(Serialize, Deserialize, Debug, Clone)]
@ -141,6 +169,37 @@ mod types {
} }
} }
} }
#[derive(Serialize, Deserialize, Debug, Clone)]
pub struct Project {
id: i64,
workspace_id: i64,
client_id: Option<i64>,
name: String,
color: String,
status: Status,
active: bool,
at: DateTime<Utc>,
start_date: NaiveDate,
created_at: DateTime<Utc>,
server_deleted_at: Option<DateTime<Utc>>,
actual_hours: Option<i64>,
actual_seconds: Option<i64>,
can_track_time: bool,
}
#[derive(Serialize, Deserialize, Debug, Clone)]
#[serde(rename_all = "lowercase")]
pub enum Status {
Upcoming,
Active,
Archived,
Ended,
Deleted,
}
} }
#[derive(Debug, thiserror::Error)] #[derive(Debug, thiserror::Error)]
@ -150,6 +209,12 @@ pub enum TogglError {
#[error("Json error: {0}")] #[error("Json error: {0}")]
JsonError(#[from] serde_json::Error), JsonError(#[from] serde_json::Error),
#[error("Failed to parse JSON data: {data}, error: {error}")]
JsonWithDataError {
data: String,
error: serde_json_path_to_error::Error,
},
} }
impl From<reqwest::Error> for TogglError { impl From<reqwest::Error> for TogglError {