trello-to-deck/src/nextcloud/client.rs
Paul Campbell 4f69fc0a4b
Some checks failed
Release Please / Release-plz (push) Failing after 42s
Test / build (map[name:stable]) (push) Has been cancelled
Test / build (map[name:nightly]) (push) Has been cancelled
feat: make best-effort to maintain order of stacks and cards
2024-12-23 09:42:52 +00:00

243 lines
7.2 KiB
Rust

//
use bytes::Bytes;
use kxio::{
fs::FileHandle,
net::{Net, ReqBuilder},
};
use reqwest::multipart;
use serde_json::json;
use crate::nextcloud::model::NextcloudOrder;
use crate::{
api_result::APIResult,
f,
nextcloud::model::{
Attachment, Board, Card, Label, NextcloudBoardId, NextcloudCardDescription,
NextcloudCardId, NextcloudCardTitle, NextcloudHostname, NextcloudLabelColour,
NextcloudLabelId, NextcloudLabelTitle, NextcloudPassword, NextcloudStackId,
NextcloudStackTitle, NextcloudUsername, Stack,
},
with_exponential_backoff, FullCtx,
};
pub(crate) struct DeckClient<'ctx> {
ctx: &'ctx FullCtx,
hostname: &'ctx NextcloudHostname,
username: &'ctx NextcloudUsername,
password: &'ctx NextcloudPassword,
}
// Uses the API described here: https://deck.readthedocs.io/en/stable/API/#cards
impl<'ctx> DeckClient<'ctx> {
pub const fn new(ctx: &'ctx FullCtx) -> Self {
Self {
ctx,
hostname: &ctx.cfg.nextcloud.hostname,
username: &ctx.cfg.nextcloud.username,
password: &ctx.cfg.nextcloud.password,
}
}
fn url(&self, path: impl Into<String>) -> String {
f!(
"{}/index.php/apps/deck/api/v1.0/{}",
self.hostname,
path.into()
)
}
async fn request<T: for<'a> serde::Deserialize<'a>>(
&self,
url: impl Into<String>,
custom: fn(&Net, String) -> ReqBuilder,
) -> APIResult<T> {
let url = url.into();
APIResult::new(
with_exponential_backoff!(
&self.ctx,
custom(&self.ctx.net, self.url(url.clone()))
.basic_auth(self.username.as_str(), Some(self.password.as_str()))
.header("accept", "application/json")
.header("content-type", "application/json")
.send()
.await
),
&self.ctx.prt,
)
.await
}
async fn request_with_body<T: for<'a> serde::Deserialize<'a>>(
&self,
url: impl Into<String>,
body: impl Into<Bytes>,
custom: fn(&Net, String) -> ReqBuilder,
) -> APIResult<T> {
let url = self.url(url.into());
let body = body.into();
APIResult::new(
with_exponential_backoff!(
&self.ctx,
custom(&self.ctx.net, url.clone())
.basic_auth(self.username.as_str(), Some(self.password.as_str()))
.header("accept", "application/json")
.header("content-type", "application/json")
.body(body.clone())
.send()
.await
),
&self.ctx.prt,
)
.await
}
pub(crate) async fn get_boards(&self) -> APIResult<Vec<Board>> {
self.request("boards", |net, url| net.get(url)).await
}
pub(crate) async fn get_board(&self, board_id: NextcloudBoardId) -> APIResult<Board> {
self.request(f!("boards/{board_id}"), |net, url| net.get(url))
.await
}
pub(crate) async fn get_stacks(&self, board_id: NextcloudBoardId) -> APIResult<Vec<Stack>> {
self.request(f!("boards/{board_id}/stacks"), |net, url| net.get(url))
.await
}
pub(crate) async fn get_stack(
&self,
board_id: NextcloudBoardId,
stack_id: NextcloudStackId,
) -> APIResult<Stack> {
self.request(f!("boards/{board_id}/stacks/{stack_id}"), |net, url| {
net.get(url)
})
.await
}
pub(crate) async fn create_stack(
&self,
board_id: NextcloudBoardId,
stack_title: NextcloudStackTitle,
stack_order: NextcloudOrder,
) -> APIResult<Stack> {
self.request_with_body(
f!("boards/{board_id}/stacks"),
json!({
"title": stack_title,
"order": stack_order,
})
.to_string(),
|net, url| net.post(url),
)
.await
}
pub(crate) async fn create_card(
&self,
board_id: NextcloudBoardId,
stack_id: NextcloudStackId,
title: &NextcloudCardTitle,
description: Option<&NextcloudCardDescription>,
) -> APIResult<Card> {
let mut body = json!({
"title": title,
});
if let Some(desc) = &description {
body["description"] = serde_json::Value::String(desc.to_string());
}
self.request_with_body(
f!("boards/{board_id}/stacks/{stack_id}/cards"),
body.to_string(),
|net, url| net.post(url),
)
.await
}
pub(crate) async fn create_label(
&self,
board_id: NextcloudBoardId,
name: &NextcloudLabelTitle,
colour: &NextcloudLabelColour,
) -> APIResult<Label> {
self.request_with_body(
f!("boards/{board_id}/labels"),
json!({
"title": name,
"color": colour,
})
.to_string(),
|net, url| net.post(url),
)
.await
}
pub(crate) async fn get_card(
&self,
board_id: NextcloudBoardId,
stack_id: NextcloudStackId,
card_id: NextcloudCardId,
) -> APIResult<Card> {
self.request(
f!("boards/{board_id}/stacks/{stack_id}/cards/{card_id}"),
|net, url| net.get(url),
)
.await
}
pub(crate) async fn add_label_to_card(
&self,
board_id: NextcloudBoardId,
stack_id: NextcloudStackId,
card_id: NextcloudCardId,
label_id: NextcloudLabelId,
) -> APIResult<()> {
self.request_with_body(
f!("boards/{board_id}/stacks/{stack_id}/cards/{card_id}/assignLabel"),
json!({
"labelId": label_id
})
.to_string(),
|net, url| net.put(url),
)
.await
}
pub(crate) async fn add_attachment_to_card(
&self,
board_id: NextcloudBoardId,
stack_id: NextcloudStackId,
card_id: NextcloudCardId,
file: &FileHandle,
) -> APIResult<Attachment> {
APIResult::new(
with_exponential_backoff!(&self.ctx, {
let form: multipart::Form = multipart::Form::new();
let form = form.text("type", "file");
let form = form
.file("file", file.as_pathbuf())
.await
.expect("read file");
self.ctx
.net
.send(
self.ctx
.net
.client()
.post(self.url(f!(
"boards/{board_id}/stacks/{stack_id}/cards/{card_id}/attachments"
)))
.basic_auth(self.username.as_str(), Some(self.password.as_str()))
.header("accept", "application/json")
.multipart(form),
)
.await
}),
&self.ctx.prt,
)
.await
}
}