use derive_builder::Builder;
use reqwest::Method;
use std::borrow::Cow;
use crate::api::projects::ProjectEssentials;
use crate::api::{Endpoint, ReturnsJsonResponse};
use serde::Serialize;
#[derive(Debug, Clone, PartialEq, Eq, serde::Serialize, serde::Deserialize)]
pub struct VersionEssentials {
pub id: u64,
pub name: String,
}
impl From<Version> for VersionEssentials {
fn from(v: Version) -> Self {
VersionEssentials {
id: v.id,
name: v.name,
}
}
}
impl From<&Version> for VersionEssentials {
fn from(v: &Version) -> Self {
VersionEssentials {
id: v.id,
name: v.name.to_owned(),
}
}
}
#[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
pub struct Version {
pub id: u64,
pub name: String,
pub project: ProjectEssentials,
pub description: String,
pub status: VersionStatus,
pub due_date: Option<time::Date>,
pub sharing: VersionSharing,
#[serde(
serialize_with = "crate::api::serialize_rfc3339",
deserialize_with = "crate::api::deserialize_rfc3339"
)]
pub created_on: time::OffsetDateTime,
#[serde(
serialize_with = "crate::api::serialize_rfc3339",
deserialize_with = "crate::api::deserialize_rfc3339"
)]
pub updated_on: time::OffsetDateTime,
#[serde(default)]
wiki_page_title: Option<String>,
}
#[derive(Debug, Clone, Builder)]
#[builder(setter(strip_option))]
pub struct ListVersions<'a> {
#[builder(setter(into))]
project_id_or_name: Cow<'a, str>,
}
impl ReturnsJsonResponse for ListVersions<'_> {}
impl<'a> ListVersions<'a> {
#[must_use]
pub fn builder() -> ListVersionsBuilder<'a> {
ListVersionsBuilder::default()
}
}
impl Endpoint for ListVersions<'_> {
fn method(&self) -> Method {
Method::GET
}
fn endpoint(&self) -> Cow<'static, str> {
format!("projects/{}/versions.json", self.project_id_or_name).into()
}
}
#[derive(Debug, Clone, Builder)]
#[builder(setter(strip_option))]
pub struct GetVersion {
id: u64,
}
impl ReturnsJsonResponse for GetVersion {}
impl GetVersion {
#[must_use]
pub fn builder() -> GetVersionBuilder {
GetVersionBuilder::default()
}
}
impl Endpoint for GetVersion {
fn method(&self) -> Method {
Method::GET
}
fn endpoint(&self) -> Cow<'static, str> {
format!("versions/{}.json", &self.id).into()
}
}
#[derive(Debug, Clone, serde::Deserialize, Serialize)]
#[serde(rename_all = "snake_case")]
pub enum VersionStatus {
Open,
Locked,
Closed,
}
#[derive(Debug, Clone, serde::Deserialize, Serialize)]
#[serde(rename_all = "snake_case")]
pub enum VersionSharing {
None,
Descendants,
Hierarchy,
Tree,
System,
}
#[serde_with::skip_serializing_none]
#[derive(Debug, Clone, Builder, Serialize)]
#[builder(setter(strip_option))]
pub struct CreateVersion<'a> {
#[builder(setter(into))]
#[serde(skip_serializing)]
project_id_or_name: Cow<'a, str>,
#[builder(setter(into))]
name: Cow<'a, str>,
#[builder(default)]
status: Option<VersionStatus>,
#[builder(default)]
sharing: Option<VersionSharing>,
#[builder(default)]
due_date: Option<time::Date>,
#[builder(default)]
description: Option<Cow<'a, str>>,
#[builder(default)]
wiki_page_title: Option<Cow<'a, str>>,
}
impl ReturnsJsonResponse for CreateVersion<'_> {}
impl<'a> CreateVersion<'a> {
#[must_use]
pub fn builder() -> CreateVersionBuilder<'a> {
CreateVersionBuilder::default()
}
}
impl Endpoint for CreateVersion<'_> {
fn method(&self) -> Method {
Method::POST
}
fn endpoint(&self) -> Cow<'static, str> {
format!("projects/{}/versions.json", self.project_id_or_name).into()
}
fn body(&self) -> Result<Option<(&'static str, Vec<u8>)>, crate::Error> {
Ok(Some((
"application/json",
serde_json::to_vec(&VersionWrapper::<CreateVersion> {
version: (*self).to_owned(),
})?,
)))
}
}
#[serde_with::skip_serializing_none]
#[derive(Debug, Clone, Builder, Serialize)]
#[builder(setter(strip_option))]
pub struct UpdateVersion<'a> {
#[serde(skip_serializing)]
id: u64,
#[builder(default, setter(into))]
name: Option<Cow<'a, str>>,
#[builder(default)]
status: Option<VersionStatus>,
#[builder(default)]
sharing: Option<VersionSharing>,
#[builder(default)]
due_date: Option<time::Date>,
#[builder(default)]
description: Option<Cow<'a, str>>,
#[builder(default)]
wiki_page_title: Option<Cow<'a, str>>,
}
impl<'a> UpdateVersion<'a> {
#[must_use]
pub fn builder() -> UpdateVersionBuilder<'a> {
UpdateVersionBuilder::default()
}
}
impl Endpoint for UpdateVersion<'_> {
fn method(&self) -> Method {
Method::PUT
}
fn endpoint(&self) -> Cow<'static, str> {
format!("versions/{}.json", self.id).into()
}
fn body(&self) -> Result<Option<(&'static str, Vec<u8>)>, crate::Error> {
Ok(Some((
"application/json",
serde_json::to_vec(&VersionWrapper::<UpdateVersion> {
version: (*self).to_owned(),
})?,
)))
}
}
#[derive(Debug, Clone, Builder)]
#[builder(setter(strip_option))]
pub struct DeleteVersion {
id: u64,
}
impl DeleteVersion {
#[must_use]
pub fn builder() -> DeleteVersionBuilder {
DeleteVersionBuilder::default()
}
}
impl Endpoint for DeleteVersion {
fn method(&self) -> Method {
Method::DELETE
}
fn endpoint(&self) -> Cow<'static, str> {
format!("versions/{}.json", &self.id).into()
}
}
#[derive(Debug, Clone, PartialEq, Eq, Serialize, serde::Deserialize)]
pub struct VersionsWrapper<T> {
pub versions: Vec<T>,
}
#[derive(Debug, Clone, PartialEq, Eq, Serialize, serde::Deserialize)]
pub struct VersionWrapper<T> {
pub version: T,
}
#[cfg(test)]
mod test {
use super::*;
use crate::api::test_helpers::with_project;
use pretty_assertions::assert_eq;
use std::error::Error;
use tokio::sync::RwLock;
use tracing_test::traced_test;
static VERSION_LOCK: RwLock<()> = RwLock::const_new(());
#[traced_test]
#[test]
fn test_list_versions_no_pagination() -> Result<(), Box<dyn Error>> {
let _r_versions = VERSION_LOCK.read();
dotenvy::dotenv()?;
let redmine = crate::api::Redmine::from_env()?;
let endpoint = ListVersions::builder().project_id_or_name("92").build()?;
redmine.json_response_body::<_, VersionsWrapper<Version>>(&endpoint)?;
Ok(())
}
#[traced_test]
#[test]
fn test_get_version() -> Result<(), Box<dyn Error>> {
let _r_versions = VERSION_LOCK.read();
dotenvy::dotenv()?;
let redmine = crate::api::Redmine::from_env()?;
let endpoint = GetVersion::builder().id(1182).build()?;
redmine.json_response_body::<_, VersionWrapper<Version>>(&endpoint)?;
Ok(())
}
#[function_name::named]
#[traced_test]
#[test]
fn test_create_version() -> Result<(), Box<dyn Error>> {
let _w_versions = VERSION_LOCK.write();
let name = format!("unittest_{}", function_name!());
with_project(&name, |redmine, _, name| {
let create_endpoint = CreateVersion::builder()
.project_id_or_name(name)
.name("Test Version")
.build()?;
redmine.json_response_body::<_, VersionWrapper<Version>>(&create_endpoint)?;
Ok(())
})?;
Ok(())
}
#[function_name::named]
#[traced_test]
#[test]
fn test_update_version() -> Result<(), Box<dyn Error>> {
let _w_versions = VERSION_LOCK.write();
let name = format!("unittest_{}", function_name!());
with_project(&name, |redmine, _, name| {
let create_endpoint = CreateVersion::builder()
.project_id_or_name(name)
.name("Test Version")
.build()?;
let VersionWrapper { version } =
redmine.json_response_body::<_, VersionWrapper<Version>>(&create_endpoint)?;
let update_endpoint = super::UpdateVersion::builder()
.id(version.id)
.name("Neue Test-Version")
.build()?;
redmine.ignore_response_body::<_>(&update_endpoint)?;
Ok(())
})?;
Ok(())
}
#[traced_test]
#[test]
fn test_completeness_version_type() -> Result<(), Box<dyn Error>> {
let _r_versions = VERSION_LOCK.read();
dotenvy::dotenv()?;
let redmine = crate::api::Redmine::from_env()?;
let endpoint = ListVersions::builder().project_id_or_name("92").build()?;
let VersionsWrapper { versions: values } =
redmine.json_response_body::<_, VersionsWrapper<serde_json::Value>>(&endpoint)?;
for value in values {
let o: Version = serde_json::from_value(value.clone())?;
let reserialized = serde_json::to_value(o)?;
assert_eq!(value, reserialized);
}
Ok(())
}
}