use serde::{Deserialize, Deserializer, Serialize};
use super::common::FlagUpdate;
macro_rules! match_field {
($name:expr, $self:expr, $wrap:ident, $default:expr,
{ $($field:literal => $member:ident),+ $(,)? }) => {
match $name {
$($field => $wrap!($self.$member),)+
_ => $default,
}
};
}
fn deserialize_null_string<'de, D: Deserializer<'de>>(d: D) -> Result<String, D::Error> {
Option::<String>::deserialize(d).map(Option::unwrap_or_default)
}
#[derive(Debug, Serialize, Deserialize)]
#[non_exhaustive]
pub struct Bug {
pub id: u64,
#[serde(default)]
pub summary: String,
#[serde(default)]
pub status: String,
#[serde(default)]
pub resolution: Option<String>,
#[serde(default)]
pub product: Option<String>,
#[serde(default)]
pub component: Option<String>,
#[serde(default)]
pub version: Option<String>,
#[serde(default)]
pub assigned_to: Option<String>,
#[serde(default)]
pub priority: Option<String>,
#[serde(default)]
pub severity: Option<String>,
#[serde(default)]
pub creation_time: Option<String>,
#[serde(default)]
pub last_change_time: Option<String>,
#[serde(default)]
pub creator: Option<String>,
#[serde(default)]
pub url: Option<String>,
#[serde(default)]
pub whiteboard: Option<String>,
#[serde(default)]
pub keywords: Vec<String>,
#[serde(default)]
pub blocks: Vec<u64>,
#[serde(default)]
pub depends_on: Vec<u64>,
#[serde(default)]
pub cc: Vec<String>,
#[serde(default)]
pub op_sys: Option<String>,
#[serde(default)]
pub rep_platform: Option<String>,
}
#[derive(Clone, Debug, Default)]
#[non_exhaustive]
pub struct SearchParams {
pub product: Vec<String>,
pub component: Vec<String>,
pub status: Vec<String>,
pub assigned_to: Vec<String>,
pub creator: Vec<String>,
pub priority: Vec<String>,
pub severity: Vec<String>,
pub cc: Option<String>,
pub alias: Option<String>,
pub id: Vec<u64>,
pub limit: Option<u32>,
pub summary: Option<String>,
pub quicksearch: Option<String>,
pub include_fields: Option<String>,
pub exclude_fields: Option<String>,
pub raw_params: Vec<(String, String)>,
}
impl SearchParams {
pub fn apply_overrides(
&mut self,
limit: Option<u32>,
fields: Option<&str>,
exclude_fields: Option<&str>,
) {
if let Some(l) = limit {
self.limit = Some(l);
}
if let Some(f) = fields {
self.include_fields = Some(f.to_string());
}
if let Some(ef) = exclude_fields {
self.exclude_fields = Some(ef.to_string());
}
}
pub fn get_field(&self, name: &str) -> &[String] {
macro_rules! as_ref {
($e:expr) => {
&$e
};
}
match_field!(name, self, as_ref, panic!("unknown field: {name}"), {
"product" => product,
"component" => component,
"status" => status,
"assigned_to" => assigned_to,
"creator" => creator,
"priority" => priority,
"severity" => severity,
})
}
pub fn has_filters(&self) -> bool {
!self.product.is_empty()
|| !self.component.is_empty()
|| !self.status.is_empty()
|| !self.assigned_to.is_empty()
|| !self.creator.is_empty()
|| !self.priority.is_empty()
|| !self.severity.is_empty()
|| self.cc.is_some()
|| self.alias.is_some()
|| !self.id.is_empty()
|| self.summary.is_some()
|| self.quicksearch.is_some()
|| !self.raw_params.is_empty()
}
}
pub fn partition_filters(values: &[String]) -> (Vec<&str>, Vec<&str>) {
let mut positive = Vec::new();
let mut negated = Vec::new();
for v in values {
if let Some(stripped) = v.strip_prefix('!') {
negated.push(stripped);
} else {
positive.push(v.as_str());
}
}
(positive, negated)
}
pub struct FieldMapping {
pub struct_field: &'static str,
pub url_param: &'static str,
pub internal_name: &'static str,
}
pub const FIELD_MAPPINGS: &[FieldMapping] = &[
FieldMapping {
struct_field: "product",
url_param: "product",
internal_name: "product",
},
FieldMapping {
struct_field: "component",
url_param: "component",
internal_name: "component",
},
FieldMapping {
struct_field: "status",
url_param: "bug_status",
internal_name: "bug_status",
},
FieldMapping {
struct_field: "assigned_to",
url_param: "assigned_to",
internal_name: "assigned_to",
},
FieldMapping {
struct_field: "creator",
url_param: "reporter",
internal_name: "reporter",
},
FieldMapping {
struct_field: "priority",
url_param: "priority",
internal_name: "priority",
},
FieldMapping {
struct_field: "severity",
url_param: "bug_severity",
internal_name: "bug_severity",
},
];
#[derive(Debug, Serialize)]
#[non_exhaustive]
pub struct CreateBugParams {
pub product: String,
pub component: String,
pub summary: String,
pub version: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub description: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub priority: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub severity: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub assigned_to: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub op_sys: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub rep_platform: Option<String>,
#[serde(skip_serializing_if = "Vec::is_empty")]
pub blocks: Vec<u64>,
#[serde(skip_serializing_if = "Vec::is_empty")]
pub depends_on: Vec<u64>,
#[serde(skip_serializing_if = "Vec::is_empty")]
pub cc: Vec<String>,
#[serde(skip_serializing_if = "Vec::is_empty")]
pub keywords: Vec<String>,
}
#[derive(Debug, Default, Serialize)]
#[non_exhaustive]
pub struct IdListUpdate {
#[serde(skip_serializing_if = "Vec::is_empty")]
pub add: Vec<u64>,
#[serde(skip_serializing_if = "Vec::is_empty")]
pub remove: Vec<u64>,
}
impl IdListUpdate {
pub fn is_empty(&self) -> bool {
self.add.is_empty() && self.remove.is_empty()
}
}
#[derive(Debug, Default, Serialize)]
#[non_exhaustive]
pub struct UpdateBugParams {
#[serde(skip_serializing_if = "Option::is_none")]
pub status: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub resolution: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub assigned_to: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub priority: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub severity: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub summary: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub whiteboard: Option<String>,
#[serde(skip_serializing_if = "Vec::is_empty")]
pub flags: Vec<FlagUpdate>,
#[serde(skip_serializing_if = "IdListUpdate::is_empty")]
pub blocks: IdListUpdate,
#[serde(skip_serializing_if = "IdListUpdate::is_empty")]
pub depends_on: IdListUpdate,
}
#[derive(Debug, Serialize, Deserialize)]
#[non_exhaustive]
pub struct HistoryEntry {
pub who: String,
pub when: String,
pub changes: Vec<FieldChange>,
}
#[derive(Debug, Serialize, Deserialize)]
#[non_exhaustive]
pub struct FieldChange {
pub field_name: String,
#[serde(default)]
pub removed: String,
#[serde(default)]
pub added: String,
#[serde(default)]
pub attachment_id: Option<u64>,
}
#[derive(Debug, Serialize, Deserialize)]
#[non_exhaustive]
pub struct FieldValue {
#[serde(default, deserialize_with = "deserialize_null_string")]
pub name: String,
#[serde(default)]
pub sort_key: u64,
#[serde(default)]
pub is_active: bool,
#[serde(default)]
pub can_change_to: Option<Vec<StatusTransition>>,
}
#[derive(Debug, Serialize, Deserialize)]
#[non_exhaustive]
pub struct StatusTransition {
pub name: String,
}
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize, Default)]
#[serde(rename_all = "lowercase")]
pub enum QueryKind {
#[default]
List,
Search,
Url,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
pub struct SavedQuery {
#[serde(default)]
pub kind: QueryKind,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub product: Vec<String>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub component: Vec<String>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub status: Vec<String>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub assignee: Vec<String>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub creator: Vec<String>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub priority: Vec<String>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub severity: Vec<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub quicksearch: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub limit: Option<u32>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub fields: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub exclude_fields: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub source_url: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub server: Option<String>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub raw_params: Vec<(String, String)>,
}
impl SavedQuery {
pub fn to_search_params(&self) -> SearchParams {
self.clone().into_search_params()
}
pub fn into_search_params(self) -> SearchParams {
SearchParams {
product: self.product,
component: self.component,
status: self.status,
assigned_to: self.assignee,
creator: self.creator,
priority: self.priority,
severity: self.severity,
quicksearch: self.quicksearch,
limit: self.limit,
include_fields: self.fields,
exclude_fields: self.exclude_fields,
raw_params: self.raw_params,
..Default::default()
}
}
pub fn get_field_mut(&mut self, name: &str) -> Option<&mut Vec<String>> {
macro_rules! some_mut {
($e:expr) => {
Some(&mut $e)
};
}
match_field!(name, self, some_mut, None, {
"product" => product,
"component" => component,
"status" => status,
"assigned_to" => assignee,
"creator" => creator,
"priority" => priority,
"severity" => severity,
})
}
pub fn has_filters(&self) -> bool {
!self.product.is_empty()
|| !self.component.is_empty()
|| !self.status.is_empty()
|| !self.assignee.is_empty()
|| !self.creator.is_empty()
|| !self.priority.is_empty()
|| !self.severity.is_empty()
|| self.quicksearch.is_some()
|| !self.raw_params.is_empty()
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[non_exhaustive]
pub struct BugTemplate {
#[serde(default, skip_serializing_if = "Option::is_none")]
pub product: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub component: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub version: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub priority: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub severity: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub assignee: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub op_sys: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub rep_platform: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub description: Option<String>,
}
#[cfg(test)]
#[path = "bug_tests.rs"]
mod tests;