use std::{collections::BTreeMap, iter};
use serde::{Deserialize, Serialize};
use super::extensions::Extensions;
use super::{builder, set_value};
builder! {
ServerBuilder;
#[non_exhaustive]
#[derive(Serialize, Deserialize, Default, Clone, PartialEq, Eq)]
#[cfg_attr(feature = "debug", derive(Debug))]
#[serde(rename_all = "camelCase")]
pub struct Server {
pub url: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub description: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub variables: Option<BTreeMap<String, ServerVariable>>,
#[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 ServerBuilder {
pub fn url<U: Into<String>>(mut self, url: U) -> Self {
set_value!(self url url.into())
}
pub fn description<S: Into<String>>(mut self, description: Option<S>) -> Self {
set_value!(self description description.map(|description| description.into()))
}
pub fn parameter<N: Into<String>, V: Into<ServerVariable>>(
mut self,
name: N,
variable: V,
) -> Self {
match self.variables {
Some(ref mut variables) => {
variables.insert(name.into(), variable.into());
}
None => {
self.variables = Some(BTreeMap::from_iter(iter::once((
name.into(),
variable.into(),
))))
}
}
self
}
pub fn extensions(mut self, extensions: Option<Extensions>) -> Self {
set_value!(self extensions extensions)
}
}
builder! {
ServerVariableBuilder;
#[non_exhaustive]
#[derive(Serialize, Deserialize, Default, Clone, PartialEq, Eq)]
#[cfg_attr(feature = "debug", derive(Debug))]
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 ServerVariableBuilder {
pub fn default_value<S: Into<String>>(mut self, default_value: S) -> Self {
set_value!(self default_value default_value.into())
}
pub fn description<S: Into<String>>(mut self, description: Option<S>) -> Self {
set_value!(self description description.map(|description| description.into()))
}
pub fn enum_values<I: IntoIterator<Item = V>, V: Into<String>>(
mut self,
enum_values: Option<I>,
) -> Self {
set_value!(self enum_values enum_values
.map(|enum_values| enum_values.into_iter().map(|value| value.into()).collect()))
}
pub fn extensions(mut self, extensions: Option<Extensions>) -> Self {
set_value!(self extensions extensions)
}
}
#[cfg(test)]
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:
ServerBuilder::new().url("/api/{version}/{username}")
.parameter("version", ServerVariableBuilder::new()
.enum_values(Some(["v1", "v2"]))
.description(Some("api version"))
.default_value("v1"))
.parameter("username", ServerVariableBuilder::new()
.default_value("the_user")).build();
r###"{
"url": "/api/{version}/{username}",
"variables": {
"version": {
"enum": ["v1", "v2"],
"default": "v1",
"description": "api version"
},
"username": {
"default": "the_user"
}
}
}"###
}
}