Add Projects
This commit is contained in:
parent
06de91a6fd
commit
f447cabed1
12
Cargo.lock
generated
12
Cargo.lock
generated
@ -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",
|
||||||
]
|
]
|
||||||
|
|||||||
@ -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"
|
||||||
|
|||||||
@ -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);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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 {
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user