use ploidy_core::codegen::IntoCode;
use ploidy_core::ir::{ContainerView, SchemaTypeView};
use proc_macro2::TokenStream;
use quote::{ToTokens, TokenStreamExt, quote};
use super::{
doc_attrs, enum_::CodegenEnum, inlines::CodegenInlines, naming::CodegenTypeName,
primitive::CodegenPrimitive, ref_::CodegenRef, struct_::CodegenStruct, tagged::CodegenTagged,
untagged::CodegenUntagged,
};
#[derive(Debug)]
pub struct CodegenSchemaType<'a> {
ty: &'a SchemaTypeView<'a>,
}
impl<'a> CodegenSchemaType<'a> {
pub fn new(ty: &'a SchemaTypeView<'a>) -> Self {
Self { ty }
}
}
impl ToTokens for CodegenSchemaType<'_> {
fn to_tokens(&self, tokens: &mut TokenStream) {
let name = CodegenTypeName::Schema(self.ty);
let ty = match self.ty {
SchemaTypeView::Struct(_, view) => CodegenStruct::new(name, view).into_token_stream(),
SchemaTypeView::Enum(_, view) => CodegenEnum::new(name, view).into_token_stream(),
SchemaTypeView::Tagged(_, view) => CodegenTagged::new(name, view).into_token_stream(),
SchemaTypeView::Untagged(_, view) => {
CodegenUntagged::new(name, view).into_token_stream()
}
SchemaTypeView::Container(_, ContainerView::Array(inner)) => {
let doc_attrs = inner.description().map(doc_attrs);
let inner_ty = inner.ty();
let inner_ref = CodegenRef::new(&inner_ty);
quote! {
#doc_attrs
pub type #name = ::std::vec::Vec<#inner_ref>;
}
}
SchemaTypeView::Container(_, ContainerView::Map(inner)) => {
let doc_attrs = inner.description().map(doc_attrs);
let inner_ty = inner.ty();
let inner_ref = CodegenRef::new(&inner_ty);
quote! {
#doc_attrs
pub type #name = ::std::collections::BTreeMap<::std::string::String, #inner_ref>;
}
}
SchemaTypeView::Container(_, ContainerView::Optional(inner)) => {
let doc_attrs = inner.description().map(doc_attrs);
let inner_ty = inner.ty();
let inner_ref = CodegenRef::new(&inner_ty);
quote! {
#doc_attrs
pub type #name = ::std::option::Option<#inner_ref>;
}
}
SchemaTypeView::Primitive(_, view) => {
let primitive = CodegenPrimitive::new(view);
quote! {
pub type #name = #primitive;
}
}
SchemaTypeView::Any(_, _) => {
quote! {
pub type #name = ::ploidy_util::serde_json::Value;
}
}
};
let inlines = CodegenInlines::Schema(self.ty);
tokens.append_all(quote! {
#ty
#inlines
});
}
}
impl IntoCode for CodegenSchemaType<'_> {
type Code = (String, TokenStream);
fn into_code(self) -> Self::Code {
let name = CodegenTypeName::Schema(self.ty);
(
format!("src/types/{}.rs", name.into_module_name().display()),
self.into_token_stream(),
)
}
}
#[cfg(test)]
mod tests {
use super::*;
use ploidy_core::{
arena::Arena,
ir::{RawGraph, SchemaTypeView, Spec},
parse::Document,
};
use pretty_assertions::assert_eq;
use syn::parse_quote;
use crate::CodegenGraph;
#[test]
fn test_schema_inline_types_order() {
let doc = Document::from_yaml(indoc::indoc! {"
openapi: 3.0.0
info:
title: Test API
version: 1.0.0
paths: {}
components:
schemas:
Container:
type: object
properties:
zebra:
type: object
properties:
name:
type: string
mango:
type: object
properties:
name:
type: string
apple:
type: object
properties:
name:
type: string
"})
.unwrap();
let arena = Arena::new();
let spec = Spec::from_doc(&arena, &doc).unwrap();
let graph = CodegenGraph::new(RawGraph::new(&arena, &spec).cook());
let schema = graph.schemas().find(|s| s.name() == "Container");
let Some(schema @ SchemaTypeView::Struct(_, _)) = &schema else {
panic!("expected struct `Container`; got `{schema:?}`");
};
let codegen = CodegenSchemaType::new(schema);
let actual: syn::File = parse_quote!(#codegen);
let expected: syn::File = parse_quote! {
#[derive(Debug, Clone, PartialEq, Eq, Hash, Default, ::ploidy_util::serde::Serialize, ::ploidy_util::serde::Deserialize, ::ploidy_util::pointer::JsonPointee, ::ploidy_util::pointer::JsonPointerTarget)]
#[serde(crate = "::ploidy_util::serde")]
#[ploidy(pointer(crate = "::ploidy_util::pointer"))]
pub struct Container {
#[serde(default, skip_serializing_if = "::ploidy_util::absent::AbsentOr::is_absent")]
pub zebra: ::ploidy_util::absent::AbsentOr<crate::types::container::types::Zebra>,
#[serde(default, skip_serializing_if = "::ploidy_util::absent::AbsentOr::is_absent")]
pub mango: ::ploidy_util::absent::AbsentOr<crate::types::container::types::Mango>,
#[serde(default, skip_serializing_if = "::ploidy_util::absent::AbsentOr::is_absent")]
pub apple: ::ploidy_util::absent::AbsentOr<crate::types::container::types::Apple>,
}
pub mod types {
#[derive(Debug, Clone, PartialEq, Eq, Hash, Default, ::ploidy_util::serde::Serialize, ::ploidy_util::serde::Deserialize, ::ploidy_util::pointer::JsonPointee, ::ploidy_util::pointer::JsonPointerTarget)]
#[serde(crate = "::ploidy_util::serde")]
#[ploidy(pointer(crate = "::ploidy_util::pointer"))]
pub struct Apple {
#[serde(default, skip_serializing_if = "::ploidy_util::absent::AbsentOr::is_absent")]
pub name: ::ploidy_util::absent::AbsentOr<::std::string::String>,
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, Default, ::ploidy_util::serde::Serialize, ::ploidy_util::serde::Deserialize, ::ploidy_util::pointer::JsonPointee, ::ploidy_util::pointer::JsonPointerTarget)]
#[serde(crate = "::ploidy_util::serde")]
#[ploidy(pointer(crate = "::ploidy_util::pointer"))]
pub struct Mango {
#[serde(default, skip_serializing_if = "::ploidy_util::absent::AbsentOr::is_absent")]
pub name: ::ploidy_util::absent::AbsentOr<::std::string::String>,
}
#[derive(Debug, Clone, PartialEq, Eq, Hash, Default, ::ploidy_util::serde::Serialize, ::ploidy_util::serde::Deserialize, ::ploidy_util::pointer::JsonPointee, ::ploidy_util::pointer::JsonPointerTarget)]
#[serde(crate = "::ploidy_util::serde")]
#[ploidy(pointer(crate = "::ploidy_util::pointer"))]
pub struct Zebra {
#[serde(default, skip_serializing_if = "::ploidy_util::absent::AbsentOr::is_absent")]
pub name: ::ploidy_util::absent::AbsentOr<::std::string::String>,
}
}
};
assert_eq!(actual, expected);
}
#[test]
fn test_container_schema_emits_type_alias_with_inline_types() {
let doc = Document::from_yaml(indoc::indoc! {"
openapi: 3.0.0
info:
title: Test API
version: 1.0.0
paths: {}
components:
schemas:
InvalidParameters:
type: array
items:
type: object
required:
- name
- reason
properties:
name:
type: string
reason:
type: string
"})
.unwrap();
let arena = Arena::new();
let spec = Spec::from_doc(&arena, &doc).unwrap();
let graph = CodegenGraph::new(RawGraph::new(&arena, &spec).cook());
let schema = graph.schemas().find(|s| s.name() == "InvalidParameters");
let Some(schema @ SchemaTypeView::Container(_, _)) = &schema else {
panic!("expected container `InvalidParameters`; got `{schema:?}`");
};
let codegen = CodegenSchemaType::new(schema);
let actual: syn::File = parse_quote!(#codegen);
let expected: syn::File = parse_quote! {
pub type InvalidParameters = ::std::vec::Vec<crate::types::invalid_parameters::types::Item>;
pub mod types {
#[derive(Debug, Clone, PartialEq, Eq, Hash, Default, ::ploidy_util::serde::Serialize, ::ploidy_util::serde::Deserialize, ::ploidy_util::pointer::JsonPointee, ::ploidy_util::pointer::JsonPointerTarget)]
#[serde(crate = "::ploidy_util::serde")]
#[ploidy(pointer(crate = "::ploidy_util::pointer"))]
pub struct Item {
pub name: ::std::string::String,
pub reason: ::std::string::String,
}
}
};
assert_eq!(actual, expected);
}
#[test]
fn test_container_schema_emits_type_alias_without_inline_types() {
let doc = Document::from_yaml(indoc::indoc! {"
openapi: 3.0.0
info:
title: Test API
version: 1.0.0
paths: {}
components:
schemas:
Tags:
type: array
items:
type: string
"})
.unwrap();
let arena = Arena::new();
let spec = Spec::from_doc(&arena, &doc).unwrap();
let graph = CodegenGraph::new(RawGraph::new(&arena, &spec).cook());
let schema = graph.schemas().find(|s| s.name() == "Tags");
let Some(schema @ SchemaTypeView::Container(_, _)) = &schema else {
panic!("expected container `Tags`; got `{schema:?}`");
};
let codegen = CodegenSchemaType::new(schema);
let actual: syn::File = parse_quote!(#codegen);
let expected: syn::File = parse_quote! {
pub type Tags = ::std::vec::Vec<::std::string::String>;
};
assert_eq!(actual, expected);
}
#[test]
fn test_container_schema_map_emits_type_alias() {
let doc = Document::from_yaml(indoc::indoc! {"
openapi: 3.0.0
info:
title: Test API
version: 1.0.0
paths: {}
components:
schemas:
Metadata:
type: object
additionalProperties:
type: string
"})
.unwrap();
let arena = Arena::new();
let spec = Spec::from_doc(&arena, &doc).unwrap();
let graph = CodegenGraph::new(RawGraph::new(&arena, &spec).cook());
let schema = graph.schemas().find(|s| s.name() == "Metadata");
let Some(schema @ SchemaTypeView::Container(_, _)) = &schema else {
panic!("expected container `Metadata`; got `{schema:?}`");
};
let codegen = CodegenSchemaType::new(schema);
let actual: syn::File = parse_quote!(#codegen);
let expected: syn::File = parse_quote! {
pub type Metadata = ::std::collections::BTreeMap<::std::string::String, ::std::string::String>;
};
assert_eq!(actual, expected);
}
#[test]
fn test_container_nullable_schema() {
let doc = Document::from_yaml(indoc::indoc! {"
openapi: 3.1.0
info:
title: Test API
version: 1.0.0
paths: {}
components:
schemas:
NullableString:
type: [string, 'null']
NullableArray:
type: [array, 'null']
items:
type: string
NullableMap:
type: [object, 'null']
additionalProperties:
type: string
NullableOneOf:
oneOf:
- type: object
properties:
value:
type: string
- type: 'null'
"})
.unwrap();
let arena = Arena::new();
let spec = Spec::from_doc(&arena, &doc).unwrap();
let graph = CodegenGraph::new(RawGraph::new(&arena, &spec).cook());
let schema = graph.schemas().find(|s| s.name() == "NullableString");
let Some(schema @ SchemaTypeView::Container(_, _)) = &schema else {
panic!("expected container `NullableString`; got `{schema:?}`");
};
let codegen = CodegenSchemaType::new(schema);
let actual: syn::File = parse_quote!(#codegen);
let expected: syn::File = parse_quote! {
pub type NullableString = ::std::option::Option<::std::string::String>;
};
assert_eq!(actual, expected);
let schema = graph.schemas().find(|s| s.name() == "NullableArray");
let Some(schema @ SchemaTypeView::Container(_, _)) = &schema else {
panic!("expected container `NullableArray`; got `{schema:?}`");
};
let codegen = CodegenSchemaType::new(schema);
let actual: syn::File = parse_quote!(#codegen);
let expected: syn::File = parse_quote! {
pub type NullableArray = ::std::option::Option<::std::vec::Vec<::std::string::String>>;
};
assert_eq!(actual, expected);
let schema = graph.schemas().find(|s| s.name() == "NullableMap");
let Some(schema @ SchemaTypeView::Container(_, _)) = &schema else {
panic!("expected container `NullableMap`; got `{schema:?}`");
};
let codegen = CodegenSchemaType::new(schema);
let actual: syn::File = parse_quote!(#codegen);
let expected: syn::File = parse_quote! {
pub type NullableMap = ::std::option::Option<::std::collections::BTreeMap<::std::string::String, ::std::string::String>>;
};
assert_eq!(actual, expected);
let schema = graph.schemas().find(|s| s.name() == "NullableOneOf");
let Some(schema @ SchemaTypeView::Container(_, _)) = &schema else {
panic!("expected container `NullableOneOf`; got `{schema:?}`");
};
let codegen = CodegenSchemaType::new(schema);
let actual: syn::File = parse_quote!(#codegen);
let expected: syn::File = parse_quote! {
pub type NullableOneOf = ::std::option::Option<crate::types::nullable_one_of::types::V1>;
pub mod types {
#[derive(Debug, Clone, PartialEq, Eq, Hash, Default, ::ploidy_util::serde::Serialize, ::ploidy_util::serde::Deserialize, ::ploidy_util::pointer::JsonPointee, ::ploidy_util::pointer::JsonPointerTarget)]
#[serde(crate = "::ploidy_util::serde")]
#[ploidy(pointer(crate = "::ploidy_util::pointer"))]
pub struct V1 {
#[serde(default, skip_serializing_if = "::ploidy_util::absent::AbsentOr::is_absent")]
pub value: ::ploidy_util::absent::AbsentOr<::std::string::String>,
}
}
};
assert_eq!(actual, expected);
}
#[test]
fn test_container_schema_preserves_description() {
let doc = Document::from_yaml(indoc::indoc! {"
openapi: 3.0.0
info:
title: Test API
version: 1.0.0
paths: {}
components:
schemas:
Tags:
description: A list of tags.
type: array
items:
type: string
"})
.unwrap();
let arena = Arena::new();
let spec = Spec::from_doc(&arena, &doc).unwrap();
let graph = CodegenGraph::new(RawGraph::new(&arena, &spec).cook());
let schema = graph.schemas().find(|s| s.name() == "Tags");
let Some(schema @ SchemaTypeView::Container(_, _)) = &schema else {
panic!("expected container `Tags`; got `{schema:?}`");
};
let codegen = CodegenSchemaType::new(schema);
let actual: syn::File = parse_quote!(#codegen);
let expected: syn::File = parse_quote! {
#[doc = "A list of tags."]
pub type Tags = ::std::vec::Vec<::std::string::String>;
};
assert_eq!(actual, expected);
}
}