diff --git a/zio-http-gen/src/main/scala/zio/http/gen/openapi/EndpointGen.scala b/zio-http-gen/src/main/scala/zio/http/gen/openapi/EndpointGen.scala index d1cb7a85a..d913d9e29 100644 --- a/zio-http-gen/src/main/scala/zio/http/gen/openapi/EndpointGen.scala +++ b/zio-http-gen/src/main/scala/zio/http/gen/openapi/EndpointGen.scala @@ -673,85 +673,44 @@ final case class EndpointGen(config: Config) { } else None @tailrec - private def schemaToPathCodec(schema: JsonSchema, openAPI: OpenAPI, name: String): Code.PathSegmentCode = { + private def schemaToPathCodec(schema: JsonSchema, openAPI: OpenAPI, name: String): Code.PathSegmentCode = schema match { case JsonSchema.AnnotatedSchema(s, _) => schemaToPathCodec(s, openAPI, name) case JsonSchema.RefSchema(ref) => schemaToPathCodec(resolveSchemaRef(openAPI, ref), openAPI, name) - case JsonSchema.Integer(JsonSchema.IntegerFormat.Int32, _, _, _, _, _) => - Code.PathSegmentCode(name = name, segmentType = Code.CodecType.Int) - case JsonSchema.Integer(JsonSchema.IntegerFormat.Int64, _, _, _, _, _) => - Code.PathSegmentCode(name = name, segmentType = Code.CodecType.Long) - case JsonSchema.Integer(JsonSchema.IntegerFormat.Timestamp, _, _, _, _, _) => - Code.PathSegmentCode(name = name, segmentType = Code.CodecType.Long) - case JsonSchema.String(Some(JsonSchema.StringFormat.UUID), _, _, _) => - Code.PathSegmentCode(name = name, segmentType = Code.CodecType.UUID) - case JsonSchema.String(Some(JsonSchema.StringFormat.Date), _, _, _) => - Code.PathSegmentCode(name = name, segmentType = Code.CodecType.LocalDate) - case JsonSchema.String(Some(JsonSchema.StringFormat.DateTime), _, _, _) => - Code.PathSegmentCode(name = name, segmentType = Code.CodecType.Instant) - case JsonSchema.String(Some(JsonSchema.StringFormat.Time), _, _, _) => - Code.PathSegmentCode(name = name, segmentType = Code.CodecType.LocalTime) - case JsonSchema.String(Some(JsonSchema.StringFormat.Duration), _, _, _) => - Code.PathSegmentCode(name = name, segmentType = Code.CodecType.Duration) - case JsonSchema.String(_, _, _, _) => - Code.PathSegmentCode(name = name, segmentType = Code.CodecType.String) - case JsonSchema.Boolean => - Code.PathSegmentCode(name = name, segmentType = Code.CodecType.Boolean) - case JsonSchema.OneOfSchema(_) => throw new Exception("Alternative path variables are not supported") - case JsonSchema.AllOfSchema(_) => throw new Exception("Path variables must have exactly one schema") - case JsonSchema.AnyOfSchema(_) => throw new Exception("Path variables must have exactly one schema") - case JsonSchema.Number(_, _, _, _, _, _) => - throw new Exception("Floating point path variables are currently not supported") - case JsonSchema.ArrayType(_, _, _) => throw new Exception("Array path variables are not supported") - case JsonSchema.Object(_, _, _) => throw new Exception("Object path variables are not supported") - case JsonSchema.Enum(_) => throw new Exception("Enum path variables are not supported") - case JsonSchema.Null => throw new Exception("Null path variables are not supported") - case JsonSchema.AnyJson => throw new Exception("AnyJson path variables are not supported") + case s: JsonSchema.Integer => Code.PathSegmentCode(name = name, segmentType = integerCodec(s.format)) + case s: JsonSchema.String => Code.PathSegmentCode(name = name, segmentType = stringCodec(s.format)) + case JsonSchema.Boolean => Code.PathSegmentCode(name = name, segmentType = Code.CodecType.Boolean) + case JsonSchema.OneOfSchema(_) => throw new Exception("Alternative path variables are not supported") + case JsonSchema.AllOfSchema(_) => throw new Exception("Path variables must have exactly one schema") + case JsonSchema.AnyOfSchema(_) => throw new Exception("Path variables must have exactly one schema") + case JsonSchema.ArrayType(_, _, _) => throw new Exception("Array path variables are not supported") + case JsonSchema.Object(_, _, _) => throw new Exception("Object path variables are not supported") + case JsonSchema.Enum(_) => throw new Exception("Enum path variables are not supported") + case JsonSchema.Null => throw new Exception("Null path variables are not supported") + case JsonSchema.AnyJson => throw new Exception("AnyJson path variables are not supported") + case _: JsonSchema.Number => throw new Exception("Floating point path variables are not supported") } - } @tailrec private def schemaToQueryParamCodec( schema: JsonSchema, openAPI: OpenAPI, name: String, - ): Code.QueryParamCode = { - schema match { - case JsonSchema.AnnotatedSchema(s, _) => - schemaToQueryParamCodec(s, openAPI, name) - case JsonSchema.RefSchema(ref) => - schemaToQueryParamCodec(resolveSchemaRef(openAPI, ref), openAPI, name) - case JsonSchema.Integer(JsonSchema.IntegerFormat.Int32, _, _, _, _, _) => - Code.QueryParamCode(name = name, queryType = Code.CodecType.Int) - case JsonSchema.Integer(JsonSchema.IntegerFormat.Int64, _, _, _, _, _) => - Code.QueryParamCode(name = name, queryType = Code.CodecType.Long) - case JsonSchema.Integer(JsonSchema.IntegerFormat.Timestamp, _, _, _, _, _) => - Code.QueryParamCode(name = name, queryType = Code.CodecType.Long) - case JsonSchema.String(Some(JsonSchema.StringFormat.Date), _, _, _) => - Code.QueryParamCode(name = name, queryType = Code.CodecType.LocalDate) - case JsonSchema.String(Some(JsonSchema.StringFormat.DateTime), _, _, _) => - Code.QueryParamCode(name = name, queryType = Code.CodecType.Instant) - case JsonSchema.String(Some(JsonSchema.StringFormat.Duration), _, _, _) => - Code.QueryParamCode(name = name, queryType = Code.CodecType.Duration) - case JsonSchema.String(Some(JsonSchema.StringFormat.Time), _, _, _) => - Code.QueryParamCode(name = name, queryType = Code.CodecType.LocalTime) - case JsonSchema.String(Some(JsonSchema.StringFormat.UUID), _, _, _) => - Code.QueryParamCode(name = name, queryType = Code.CodecType.UUID) - case JsonSchema.String(_, _, _, _) => - Code.QueryParamCode(name = name, queryType = Code.CodecType.String) - case JsonSchema.Boolean => - Code.QueryParamCode(name = name, queryType = Code.CodecType.Boolean) - case JsonSchema.OneOfSchema(_) => throw new Exception("Alternative query parameters are not supported") - case JsonSchema.AllOfSchema(_) => throw new Exception("Query parameters must have exactly one schema") - case JsonSchema.AnyOfSchema(_) => throw new Exception("Query parameters must have exactly one schema") - case JsonSchema.Number(_, _, _, _, _, _) => - throw new Exception("Floating point query parameters are currently not supported") - case JsonSchema.ArrayType(_, _, _) => throw new Exception("Array query parameters are not supported") - case JsonSchema.Object(_, _, _) => throw new Exception("Object query parameters are not supported") - case JsonSchema.Enum(_) => throw new Exception("Enum query parameters are not supported") - case JsonSchema.Null => throw new Exception("Null query parameters are not supported") - case JsonSchema.AnyJson => throw new Exception("AnyJson query parameters are not supported") - } + ): Code.QueryParamCode = schema match { + case JsonSchema.AnnotatedSchema(s, _) => schemaToQueryParamCodec(s, openAPI, name) + case JsonSchema.RefSchema(ref) => schemaToQueryParamCodec(resolveSchemaRef(openAPI, ref), openAPI, name) + case JsonSchema.Boolean => Code.QueryParamCode(name = name, queryType = Code.CodecType.Boolean) + case s: JsonSchema.Integer => Code.QueryParamCode(name = name, queryType = integerCodec(s.format)) + case s: JsonSchema.String => Code.QueryParamCode(name = name, queryType = stringCodec(s.format)) + case _: JsonSchema.Number => throw new Exception("Floating point query parameters are not supported") + case JsonSchema.OneOfSchema(_) => throw new Exception("Alternative query parameters are not supported") + case JsonSchema.AllOfSchema(_) => throw new Exception("Query parameters must have exactly one schema") + case JsonSchema.AnyOfSchema(_) => throw new Exception("Query parameters must have exactly one schema") + case JsonSchema.ArrayType(_, _, _) => throw new Exception("Array query parameters are not supported") + case JsonSchema.Object(_, _, _) => throw new Exception("Object query parameters are not supported") + case JsonSchema.Enum(_) => throw new Exception("Enum query parameters are not supported") + case JsonSchema.Null => throw new Exception("Null query parameters are not supported") + case JsonSchema.AnyJson => throw new Exception("AnyJson query parameters are not supported") } private def fieldsOfObject(openAPI: OpenAPI, annotations: Chunk[JsonSchema.MetaData])( @@ -1042,6 +1001,7 @@ final case class EndpointGen(config: Config) { properties.map { case (name, schema) => name -> schema.withoutAnnotations }.collect { case (name, schema) if !schema.isInstanceOf[JsonSchema.RefSchema] + && !(schema == JsonSchema.AnyJson) && !schema.isPrimitive && !schema.isCollection => schemaToCode(schema, openAPI, name.capitalize, Chunk.empty) @@ -1093,7 +1053,7 @@ final case class EndpointGen(config: Config) { ), ) case JsonSchema.Null => throw new Exception("Null query parameters are not supported") - case JsonSchema.AnyJson => throw new Exception("AnyJson query parameters are not supported") + case JsonSchema.AnyJson => None } } @@ -1378,4 +1338,18 @@ final case class EndpointGen(config: Config) { } } + private def integerCodec(format: JsonSchema.IntegerFormat): Code.CodecType = format match { + case JsonSchema.IntegerFormat.Int32 => Code.CodecType.Int + case JsonSchema.IntegerFormat.Int64 => Code.CodecType.Long + case JsonSchema.IntegerFormat.Timestamp => Code.CodecType.Long + } + + private def stringCodec(format: Option[JsonSchema.StringFormat]): Code.CodecType = format match { + case Some(JsonSchema.StringFormat.Date) => Code.CodecType.LocalDate + case Some(JsonSchema.StringFormat.DateTime) => Code.CodecType.Instant + case Some(JsonSchema.StringFormat.Duration) => Code.CodecType.Duration + case Some(JsonSchema.StringFormat.Time) => Code.CodecType.LocalTime + case Some(JsonSchema.StringFormat.UUID) => Code.CodecType.UUID + case _ => Code.CodecType.String + } } diff --git a/zio-http-gen/src/main/scala/zio/http/gen/scala/CodeGen.scala b/zio-http-gen/src/main/scala/zio/http/gen/scala/CodeGen.scala index 6b582ca7d..bda36f4f7 100644 --- a/zio-http-gen/src/main/scala/zio/http/gen/scala/CodeGen.scala +++ b/zio-http-gen/src/main/scala/zio/http/gen/scala/CodeGen.scala @@ -4,8 +4,6 @@ import java.nio.charset.StandardCharsets import java.nio.file.StandardOpenOption._ import java.nio.file._ -import scala.util.matching.Regex - object CodeGen { private val EndpointImports = @@ -265,6 +263,9 @@ object CodeGen { case Code.TypeRef(name) => Nil -> name + case Code.ScalaType.JsonAST => + List(Code.Import("zio.json.ast.Json")) -> "Json" + case scalaType => throw new Exception(s"Unknown ScalaType: $scalaType") } diff --git a/zio-http-gen/src/test/resources/AnimalWithAny.scala b/zio-http-gen/src/test/resources/AnimalWithAny.scala new file mode 100644 index 000000000..119f0c684 --- /dev/null +++ b/zio-http-gen/src/test/resources/AnimalWithAny.scala @@ -0,0 +1,14 @@ +package test.component + +import zio.json.ast.Json +import zio.schema._ +import zio.schema.annotation.fieldName + +case class Animal( + name: String, + eats: Json, + @fieldName("extra_attributes") extraAttributes: Map[String, Json], +) +object Animal { + implicit val codec: Schema[Animal] = DeriveSchema.gen[Animal] +} diff --git a/zio-http-gen/src/test/resources/inline_schema_any_and_any_object.yaml b/zio-http-gen/src/test/resources/inline_schema_any_and_any_object.yaml new file mode 100644 index 000000000..fbe3b9c2f --- /dev/null +++ b/zio-http-gen/src/test/resources/inline_schema_any_and_any_object.yaml @@ -0,0 +1,49 @@ +info: + title: Animals Service + version: 0.0.1 +tags: + - name: Animals_API +paths: + /api/v1/zoo/{animal}: + get: + operationId: get_animal + parameters: + - in: path + name: animal + schema: + type: string + required: true + tags: + - Animals_API + description: Get animals by species name + responses: + "200": + content: + application/json: + schema: + type: array + items: + $ref: '#/components/schemas/Animal' + description: OK + "500": + content: + application/json: + schema: + $ref: '#/components/schemas/HttpError' + description: Internal Server Error +openapi: 3.0.3 +components: + schemas: + Animal: + type: object + required: + - name + - eats + - extra_attributes + properties: + name: + type: string + eats: {} + extra_attributes: + type: object + additionalProperties: true \ No newline at end of file diff --git a/zio-http-gen/src/test/scala/zio/http/gen/scala/CodeGenSpec.scala b/zio-http-gen/src/test/scala/zio/http/gen/scala/CodeGenSpec.scala index fe0aa3f37..fda70f5ff 100644 --- a/zio-http-gen/src/test/scala/zio/http/gen/scala/CodeGenSpec.scala +++ b/zio-http-gen/src/test/scala/zio/http/gen/scala/CodeGenSpec.scala @@ -1012,6 +1012,30 @@ object CodeGenSpec extends ZIOSpecDefault { } } } @@ TestAspect.exceptScala3, + test("Schema with any and any object") { + val openAPIString = stringFromResource("/inline_schema_any_and_any_object.yaml") + + openApiFromYamlString(openAPIString) { oapi => + codeGenFromOpenAPI( + oapi, + Config.default.copy( + fieldNamesNormalization = Config.default.fieldNamesNormalization.copy(enableAutomatic = true), + ), + ) { testDir => + allFilesShouldBe( + testDir.toFile, + List( + "api/v1/zoo/Animal.scala", + "component/Animal.scala", + ), + ) && fileShouldBe( + testDir, + "component/Animal.scala", + "/AnimalWithAny.scala", + ) + } + } + } @@ TestAspect.exceptScala3, test("Generate all responses") { val oapi = OpenAPI( diff --git a/zio-http/shared/src/main/scala/zio/http/endpoint/openapi/JsonSchema.scala b/zio-http/shared/src/main/scala/zio/http/endpoint/openapi/JsonSchema.scala index 3ccac01aa..8dced7348 100644 --- a/zio-http/shared/src/main/scala/zio/http/endpoint/openapi/JsonSchema.scala +++ b/zio-http/shared/src/main/scala/zio/http/endpoint/openapi/JsonSchema.scala @@ -254,111 +254,139 @@ object JsonSchema { .get private[openapi] def fromSerializableSchema(schema: SerializableJsonSchema): JsonSchema = { - val additionalProperties = schema.additionalProperties match { - case Some(BoolOrSchema.BooleanWrapper(bool)) => Left(bool) - case Some(BoolOrSchema.SchemaWrapper(schema)) => - val valuesSchema = fromSerializableSchema(schema) - Right( - schema.optionalKeySchema.fold(valuesSchema)(keySchema => - valuesSchema.annotate( - MetaData.KeySchema( - fromSerializableSchema(keySchema), + + val definedAttributesCount = schema.productIterator.count(_.asInstanceOf[Option[_]].isDefined) + + // if type: object with additionalProperties defined, + // but nothing else, we should assume a free form object + // if type is not defined, but additionalProperties is, + // and nothing else, object is assumed again. + // if both type: object and additionalProperties are defined, + // and nothing else, object is assumed. + def anyObject: Boolean = { + val isObject = schema.schemaType.contains(TypeOrTypes.Type("object")) + val hasAttrs = schema.additionalProperties.collect { case BoolOrSchema.BooleanWrapper(b) => + b + }.exists(identity) + + // if definedAttributesCount == 0, this also yields true, + // but we check for it before calling this function, + // thus no need to check it here. + val isAnyObj = List(isObject, hasAttrs).count(identity) == definedAttributesCount + + isAnyObj + } + + if (definedAttributesCount == 0) JsonSchema.AnyJson + else if (anyObject) JsonSchema.Object(Map.empty, Right(JsonSchema.AnyJson), Chunk.empty) + else { + + val additionalProperties = schema.additionalProperties match { + case Some(BoolOrSchema.BooleanWrapper(bool)) => Left(bool) + case Some(BoolOrSchema.SchemaWrapper(schema)) => + val valuesSchema = fromSerializableSchema(schema) + Right( + schema.optionalKeySchema.fold(valuesSchema)(keySchema => + valuesSchema.annotate( + MetaData.KeySchema( + fromSerializableSchema(keySchema), + ), ), ), - ), - ) - case None => Left(true) - } + ) + case None => Left(true) + } - var jsonSchema: JsonSchema = schema match { - case schema if schema.ref.isDefined => - RefSchema(schema.ref.get) - case schema if schema.schemaType.contains(TypeOrTypes.Type("number")) => - JsonSchema.Number( - NumberFormat.fromString(schema.format.getOrElse("double")), - schema.minimum.map(_.fold(identity, _.toDouble)), - schema.exclusiveMinimum.map(_.map(_.fold(identity, _.toDouble))), - schema.maximum.map(_.fold(identity, _.toDouble)), - schema.exclusiveMaximum.map(_.map(_.fold(identity, _.toDouble))), - ) - case schema if schema.schemaType.contains(TypeOrTypes.Type("integer")) => - JsonSchema.Integer( - IntegerFormat.fromString(schema.format.getOrElse("int64")), - schema.minimum.map(_.fold(_.toLong, identity)), - schema.exclusiveMinimum.map(_.map(_.fold(_.toLong, identity))), - schema.maximum.map(_.fold(_.toLong, identity)), - schema.exclusiveMaximum.map(_.map(_.fold(_.toLong, identity))), - ) - case schema if schema.schemaType.contains(TypeOrTypes.Type("string")) && schema.enumValues.isEmpty => - JsonSchema.String( - schema.format.map(StringFormat.fromString), - schema.pattern.map(Pattern.apply), - schema.maxLength, - schema.minLength, - ) - case schema if schema.schemaType.contains(TypeOrTypes.Type("boolean")) => - JsonSchema.Boolean - case schema if schema.schemaType.contains(TypeOrTypes.Type("array")) => - JsonSchema.ArrayType( - schema.items.map(fromSerializableSchema), - schema.minItems, - schema.uniqueItems.contains(true), - ) - case schema if schema.enumValues.isDefined => - JsonSchema.Enum(schema.enumValues.get.map(EnumValue.fromJson)) - case schema if schema.oneOf.isDefined => - OneOfSchema(schema.oneOf.get.map(fromSerializableSchema)) - case schema if schema.allOf.isDefined => - AllOfSchema(schema.allOf.get.map(fromSerializableSchema)) - case schema if schema.anyOf.isDefined => - AnyOfSchema(schema.anyOf.get.map(fromSerializableSchema)) - case schema if schema.schemaType.contains(TypeOrTypes.Type("null")) => - JsonSchema.Null - case schema if schema.schemaType.contains(TypeOrTypes.Type("object")) || schema.schemaType.isEmpty => - JsonSchema.Object( - schema.properties - .map(_.map { case (name, schema) => name -> fromSerializableSchema(schema) }) - .getOrElse(Map.empty), - additionalProperties, - schema.required.getOrElse(Chunk.empty), - ) - case _ => - throw new IllegalArgumentException(s"Can't convert $schema") - } + var jsonSchema: JsonSchema = schema match { + case schema if schema.ref.isDefined => + RefSchema(schema.ref.get) + case schema if schema.schemaType.contains(TypeOrTypes.Type("number")) => + JsonSchema.Number( + NumberFormat.fromString(schema.format.getOrElse("double")), + schema.minimum.map(_.fold(identity, _.toDouble)), + schema.exclusiveMinimum.map(_.map(_.fold(identity, _.toDouble))), + schema.maximum.map(_.fold(identity, _.toDouble)), + schema.exclusiveMaximum.map(_.map(_.fold(identity, _.toDouble))), + ) + case schema if schema.schemaType.contains(TypeOrTypes.Type("integer")) => + JsonSchema.Integer( + IntegerFormat.fromString(schema.format.getOrElse("int64")), + schema.minimum.map(_.fold(_.toLong, identity)), + schema.exclusiveMinimum.map(_.map(_.fold(_.toLong, identity))), + schema.maximum.map(_.fold(_.toLong, identity)), + schema.exclusiveMaximum.map(_.map(_.fold(_.toLong, identity))), + ) + case schema if schema.schemaType.contains(TypeOrTypes.Type("string")) && schema.enumValues.isEmpty => + JsonSchema.String( + schema.format.map(StringFormat.fromString), + schema.pattern.map(Pattern.apply), + schema.maxLength, + schema.minLength, + ) + case schema if schema.schemaType.contains(TypeOrTypes.Type("boolean")) => + JsonSchema.Boolean + case schema if schema.schemaType.contains(TypeOrTypes.Type("array")) => + JsonSchema.ArrayType( + schema.items.map(fromSerializableSchema), + schema.minItems, + schema.uniqueItems.contains(true), + ) + case schema if schema.enumValues.isDefined => + JsonSchema.Enum(schema.enumValues.get.map(EnumValue.fromJson)) + case schema if schema.oneOf.isDefined => + OneOfSchema(schema.oneOf.get.map(fromSerializableSchema)) + case schema if schema.allOf.isDefined => + AllOfSchema(schema.allOf.get.map(fromSerializableSchema)) + case schema if schema.anyOf.isDefined => + AnyOfSchema(schema.anyOf.get.map(fromSerializableSchema)) + case schema if schema.schemaType.contains(TypeOrTypes.Type("null")) => + JsonSchema.Null + case schema if schema.schemaType.contains(TypeOrTypes.Type("object")) || schema.schemaType.isEmpty => + JsonSchema.Object( + schema.properties + .map(_.map { case (name, schema) => name -> fromSerializableSchema(schema) }) + .getOrElse(Map.empty), + additionalProperties, + schema.required.getOrElse(Chunk.empty), + ) + case _ => + throw new IllegalArgumentException(s"Can't convert $schema") + } - val examples = Chunk.fromIterable(schema.example) ++ schema.examples.getOrElse(Chunk.empty) - if (examples.nonEmpty) jsonSchema = jsonSchema.examples(examples) + val examples = Chunk.fromIterable(schema.example) ++ schema.examples.getOrElse(Chunk.empty) + if (examples.nonEmpty) jsonSchema = jsonSchema.examples(examples) - schema.description match { - case Some(value) => jsonSchema = jsonSchema.description(value) - case None => () - } + schema.description match { + case Some(value) => jsonSchema = jsonSchema.description(value) + case None => () + } - schema.nullable match { - case Some(value) => jsonSchema = jsonSchema.nullable(value) - case None => () - } + schema.nullable match { + case Some(value) => jsonSchema = jsonSchema.nullable(value) + case None => () + } - schema.discriminator match { - case Some(value) => jsonSchema = jsonSchema.discriminator(value) - case None => () - } + schema.discriminator match { + case Some(value) => jsonSchema = jsonSchema.discriminator(value) + case None => () + } - schema.contentEncoding.flatMap(ContentEncoding.fromString) match { - case Some(value) => jsonSchema = jsonSchema.contentEncoding(value) - case None => () - } + schema.contentEncoding.flatMap(ContentEncoding.fromString) match { + case Some(value) => jsonSchema = jsonSchema.contentEncoding(value) + case None => () + } - schema.contentMediaType match { - case Some(value) => jsonSchema = jsonSchema.contentMediaType(value) - case None => () - } + schema.contentMediaType match { + case Some(value) => jsonSchema = jsonSchema.contentMediaType(value) + case None => () + } - jsonSchema = jsonSchema.default(schema.default) + jsonSchema = jsonSchema.default(schema.default) - jsonSchema = jsonSchema.deprecated(schema.deprecated.getOrElse(false)) + jsonSchema = jsonSchema.deprecated(schema.deprecated.getOrElse(false)) - jsonSchema + jsonSchema + } } def fromTextCodec(codec: TextCodec[_]): JsonSchema =