use indexmap::IndexMap;
use super::extensions::Extensions;
#[non_exhaustive]
#[derive(serde_derive::Serialize, serde_derive::Deserialize, Default, Clone, PartialEq, Eq, bon::Builder)]
#[cfg_attr(feature = "debug", derive(Debug))]
#[serde(rename_all = "camelCase")]
#[builder(on(_, into))]
pub struct Server {
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(field)]
pub variables: Option<IndexMap<String, ServerVariable>>,
pub url: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub description: Option<String>,
#[serde(skip_serializing_if = "Option::is_none", flatten)]
pub extensions: Option<Extensions>,
}
impl Server {
pub fn new<S: Into<String>>(url: S) -> Self {
Self {
url: url.into(),
..Default::default()
}
}
}
impl<S: server_builder::State> ServerBuilder<S> {
pub fn parameter(mut self, name: impl Into<String>, variable: impl Into<ServerVariable>) -> Self {
self.variables.get_or_insert_default().insert(name.into(), variable.into());
self
}
}
#[non_exhaustive]
#[derive(serde_derive::Serialize, serde_derive::Deserialize, Default, Clone, PartialEq, Eq, bon::Builder)]
#[cfg_attr(feature = "debug", derive(Debug))]
#[builder(on(_, into))]
pub struct ServerVariable {
#[serde(rename = "default")]
pub default_value: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub description: Option<String>,
#[serde(rename = "enum", skip_serializing_if = "Option::is_none")]
pub enum_values: Option<Vec<String>>,
#[serde(skip_serializing_if = "Option::is_none", flatten)]
pub extensions: Option<Extensions>,
}
impl<S: server_variable_builder::IsComplete> From<ServerVariableBuilder<S>> for ServerVariable {
fn from(value: ServerVariableBuilder<S>) -> Self {
value.build()
}
}
#[cfg(test)]
#[cfg_attr(coverage_nightly, coverage(off))]
mod tests {
use super::*;
macro_rules! test_fn {
($name:ident : $schema:expr; $expected:literal) => {
#[test]
fn $name() {
let value = serde_json::to_value($schema).unwrap();
let expected_value: serde_json::Value = serde_json::from_str($expected).unwrap();
assert_eq!(
value,
expected_value,
"testing serializing \"{}\": \nactual:\n{}\nexpected:\n{}",
stringify!($name),
value,
expected_value
);
println!("{}", &serde_json::to_string_pretty(&$schema).unwrap());
}
};
}
test_fn! {
create_server_with_builder_and_variable_substitution:
Server::builder().url("/api/{version}/{username}")
.parameter("version", ServerVariable::builder()
.enum_values(["v1".into(), "v2".into()])
.description("api version")
.default_value("v1"))
.parameter("username", ServerVariable::builder()
.default_value("the_user")).build();
r###"{
"url": "/api/{version}/{username}",
"variables": {
"version": {
"enum": ["v1", "v2"],
"default": "v1",
"description": "api version"
},
"username": {
"default": "the_user"
}
}
}"###
}
}