use derive_builder::Builder;
use reqwest::Method;
use std::borrow::Cow;
use crate::api::custom_fields::CustomField;
use crate::api::custom_fields::CustomFieldEssentialsWithValue;
use crate::api::projects::ProjectEssentials;
use crate::api::{Endpoint, NoPagination, 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>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub custom_fields: Option<Vec<CustomFieldEssentialsWithValue>>,
}
#[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 NoPagination 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 NoPagination 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, PartialEq, Eq, 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,
}
#[derive(Debug, Clone, PartialEq, Eq, serde::Serialize, serde::Deserialize)]
#[serde(untagged)]
pub enum VersionStatusFilter {
#[serde(serialize_with = "serialize_any_operator")]
Any,
#[serde(serialize_with = "serialize_none_operator")]
None,
TheseStatuses(Vec<VersionStatus>),
NotTheseStatuses(Vec<VersionStatus>),
}
fn serialize_any_operator<S>(serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
serializer.serialize_str("*")
}
fn serialize_none_operator<S>(serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
serializer.serialize_str("!*")
}
#[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>>,
#[builder(default)]
custom_fields: Option<Vec<CustomField<'a>>>,
#[builder(default)]
default_project_version: Option<bool>,
}
impl ReturnsJsonResponse for CreateVersion<'_> {}
impl NoPagination 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>>,
#[builder(default)]
custom_fields: Option<Vec<CustomField<'a>>>,
#[builder(default)]
default_project_version: Option<bool>,
}
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, Builder)]
#[builder(setter(strip_option))]
pub struct CloseCompletedVersion {
id: u64,
}
impl CloseCompletedVersion {
#[must_use]
pub fn builder() -> CloseCompletedVersionBuilder {
CloseCompletedVersionBuilder::default()
}
}
impl Endpoint for CloseCompletedVersion {
fn method(&self) -> Method {
Method::POST
}
fn endpoint(&self) -> Cow<'static, str> {
format!("versions/{}/close_completed.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::custom_fields::{
CustomFieldDefinition, CustomFieldsWrapper, CustomizedType, ListCustomFields,
};
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_version = VERSION_LOCK.blocking_read();
dotenvy::dotenv()?;
let redmine = crate::api::Redmine::from_env(
reqwest::blocking::Client::builder()
.tls_backend_rustls()
.build()?,
)?;
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_version = VERSION_LOCK.blocking_read();
dotenvy::dotenv()?;
let redmine = crate::api::Redmine::from_env(
reqwest::blocking::Client::builder()
.tls_backend_rustls()
.build()?,
)?;
let endpoint = GetVersion::builder().id(1182).build()?;
redmine.json_response_body::<_, VersionWrapper<Version>>(&endpoint)?;
Ok(())
}
#[function_name::named]
#[traced_test]
#[test]
fn test_create_update_version_with_custom_fields() -> Result<(), Box<dyn Error>> {
let _w_version = VERSION_LOCK.blocking_write();
let name = format!("unittest_{}", function_name!());
with_project(&name, |redmine, project_id, _name| {
let list_custom_fields_endpoint = ListCustomFields::builder().build()?;
let CustomFieldsWrapper { custom_fields } = redmine
.json_response_body::<_, CustomFieldsWrapper<CustomFieldDefinition>>(
&list_custom_fields_endpoint,
)?;
let version_custom_field = custom_fields
.into_iter()
.find(|cf| cf.customized_type == CustomizedType::Version);
let custom_field_id = if let Some(cf) = version_custom_field {
cf.id
} else {
eprintln!("No custom field of type Version found. Skipping test.");
return Ok(());
};
let create_endpoint = CreateVersion::builder()
.project_id_or_name(project_id.to_string())
.name("Test Version with Custom Fields")
.custom_fields(vec![CustomField {
id: custom_field_id,
name: Some(Cow::Borrowed("VersionCustomField")),
value: Cow::Borrowed("Custom Value 1"),
}])
.build()?;
let VersionWrapper { version } =
redmine.json_response_body::<_, VersionWrapper<Version>>(&create_endpoint)?;
assert_eq!(version.name, "Test Version with Custom Fields");
assert_eq!(
version.custom_fields.unwrap()[0].value.as_ref().unwrap()[0],
"Custom Value 1"
);
let update_endpoint = UpdateVersion::builder()
.id(version.id)
.name("Updated Test Version with Custom Fields")
.custom_fields(vec![CustomField {
id: custom_field_id,
name: Some(Cow::Borrowed("VersionCustomField")),
value: Cow::Borrowed("Updated Custom Value 1"),
}])
.build()?;
redmine.ignore_response_body::<_>(&update_endpoint)?;
let get_endpoint = GetVersion::builder().id(version.id).build()?;
let VersionWrapper {
version: updated_version,
} = redmine.json_response_body::<_, VersionWrapper<Version>>(&get_endpoint)?;
assert_eq!(
updated_version.name,
"Updated Test Version with Custom Fields"
);
assert_eq!(
updated_version.custom_fields.unwrap()[0]
.value
.as_ref()
.unwrap()[0],
"Updated Custom Value 1"
);
Ok(())
})?;
Ok(())
}
#[function_name::named]
#[traced_test]
#[test]
fn test_create_version_with_default_project_version() -> Result<(), Box<dyn Error>> {
let _w_version = VERSION_LOCK.blocking_write();
let name = format!("unittest_{}", function_name!());
with_project(&name, |redmine, project_id, name| {
let create_endpoint = CreateVersion::builder()
.project_id_or_name(name)
.name("Default Version")
.default_project_version(true)
.build()?;
redmine.json_response_body::<_, VersionWrapper<Version>>(&create_endpoint)?;
let project_endpoint = crate::api::projects::GetProject::builder()
.project_id_or_name(project_id.to_string())
.build()?;
let project_wrapper: crate::api::projects::ProjectWrapper<
crate::api::projects::Project,
> = redmine.json_response_body(&project_endpoint)?;
assert_eq!(
project_wrapper.project.default_version.unwrap().name,
"Default Version"
);
Ok(())
})?;
Ok(())
}
#[function_name::named]
#[traced_test]
#[test]
fn test_update_version_with_default_project_version() -> Result<(), Box<dyn Error>> {
let _w_version = VERSION_LOCK.blocking_write();
let name = format!("unittest_{}", function_name!());
with_project(&name, |redmine, project_id, name| {
let create_endpoint = CreateVersion::builder()
.project_id_or_name(name)
.name("Non-Default Version")
.build()?;
let VersionWrapper { version } =
redmine.json_response_body::<_, VersionWrapper<Version>>(&create_endpoint)?;
let update_endpoint = super::UpdateVersion::builder()
.id(version.id)
.default_project_version(true)
.build()?;
redmine.ignore_response_body::<_>(&update_endpoint)?;
let project_endpoint = crate::api::projects::GetProject::builder()
.project_id_or_name(project_id.to_string())
.build()?;
let project_wrapper: crate::api::projects::ProjectWrapper<
crate::api::projects::Project,
> = redmine.json_response_body(&project_endpoint)?;
assert_eq!(
project_wrapper.project.default_version.unwrap().name,
"Non-Default Version"
);
Ok(())
})?;
Ok(())
}
}