defmodule JsonSchema.Parser.AllOfParser do
@behaviour JsonSchema.Parser.ParserBehaviour
@moduledoc ~S"""
Parses a JSON schema allOf type:
{
"allOf": [
{
"type": "object",
"properties": {
"color": {
"$ref": "#/color"
},
"title": {
"type": "string"
},
"radius": {
"type": "number"
}
},
"required": [ "color", "radius" ]
},
{
"type": "string"
}
]
}
Into an `JsonSchema.Types.AllOfType`.
"""
require Logger
alias JsonSchema.{Parser, Types}
alias Parser.{ParserResult, Util}
alias Types.AllOfType
@doc ~S"""
Returns true if the JSON subschema represents an allOf type.
## Examples
iex> type?(%{})
false
iex> type?(%{"allOf" => []})
false
iex> type?(%{"allOf" => [%{"$ref" => "#foo"}]})
true
"""
@impl JsonSchema.Parser.ParserBehaviour
@spec type?(Types.schemaNode()) :: boolean
def type?(%{"allOf" => all_of}) when is_list(all_of) and length(all_of) > 0, do: true
def type?(_schema_node), do: false
@doc """
Parses a JSON schema allOf type into an `JsonSchema.Types.AllOfType`.
"""
@impl JsonSchema.Parser.ParserBehaviour
@spec parse(
Types.schemaNode(),
URI.t(),
URI.t() | nil,
URI.t(),
String.t()
) :: ParserResult.t()
def parse(%{"allOf" => all_of} = schema_node, parent_id, id, path, name)
when is_list(all_of) do
description = Map.get(schema_node, "description")
default = Map.get(schema_node, "default")
child_path = Util.add_fragment_child(path, "allOf")
child_types_result =
all_of
|> Util.parse_child_types(parent_id, child_path)
all_of_types =
child_types_result.type_dict
|> Util.create_types_list(child_path)
all_of_type = %AllOfType{
name: name,
description: description,
default: default,
path: path,
types: all_of_types
}
all_of_type
|> Util.create_type_dict(path, id)
|> ParserResult.new()
|> ParserResult.merge(child_types_result)
end
end