From 2e8dcfd11d0cc4e787227b139544cde754231d9f Mon Sep 17 00:00:00 2001 From: Jeff Lewis Date: Thu, 30 Nov 2023 13:04:02 -0700 Subject: [PATCH 1/2] Fix handling of date format examples --- .../openapi/src/internals/GetExtensions.scala | 25 +++++--- modules/openapi/tests/src/TimestampSpec.scala | 62 +++++++++++++++++++ 2 files changed, 78 insertions(+), 9 deletions(-) diff --git a/modules/openapi/src/internals/GetExtensions.scala b/modules/openapi/src/internals/GetExtensions.scala index f51eff2..cdf1a1c 100644 --- a/modules/openapi/src/internals/GetExtensions.scala +++ b/modules/openapi/src/internals/GetExtensions.scala @@ -21,6 +21,8 @@ import smithytranslate.openapi.internals.OpenApiPattern import scala.language.reflectiveCalls import com.fasterxml.jackson.databind.JsonNode import com.fasterxml.jackson.databind.{node => jackson} +import java.time.format.DateTimeFormatter +import java.time.ZoneId object GetExtensions { @@ -49,16 +51,21 @@ object GetExtensions { } .toList + private val formatter = + DateTimeFormatter.ofPattern("yyyy-MM-dd").withZone(ZoneId.of("UTC")) + protected[smithytranslate] def anyToNode(input: Any): Node = input match { - case null => Node.nullNode() - case b: Boolean => Node.from(b) - case s: String => Node.from(s) - case i: Int => Node.from(i) - case d: Double => Node.from(d) - case s: Short => Node.from(s) - case l: Long => Node.from(l) - case f: Float => Node.from(f) - case n: Number => Node.from(n) + case null => Node.nullNode() + case b: Boolean => Node.from(b) + case s: String => Node.from(s) + case i: Int => Node.from(i) + case d: Double => Node.from(d) + case s: Short => Node.from(s) + case l: Long => Node.from(l) + case f: Float => Node.from(f) + case n: Number => Node.from(n) + case d: java.time.OffsetDateTime => Node.from(d.toString()) + case d: java.util.Date => Node.from(formatter.format(d.toInstant())) case u: java.util.UUID => Node.from(u.toString) case m: java.util.Map[_, _] => diff --git a/modules/openapi/tests/src/TimestampSpec.scala b/modules/openapi/tests/src/TimestampSpec.scala index 2623744..3f81f6f 100644 --- a/modules/openapi/tests/src/TimestampSpec.scala +++ b/modules/openapi/tests/src/TimestampSpec.scala @@ -92,4 +92,66 @@ final class TimestampSpec extends munit.FunSuite { TestUtils.runConversionTest(openapiString, expectedString) } + + test("date-time with example") { + val openapiString = """|openapi: '3.0.' + |info: + | title: test + | version: '1.0' + |paths: {} + |components: + | schemas: + | MyTimestamp: + | type: string + | format: date-time + | example: '2017-07-21T17:32:28Z' + |""".stripMargin + + val expectedString = """|namespace foo + | + |use alloy#dataExamples + | + |@dataExamples([ + | { + | json: "2017-07-21T17:32:28Z" + | } + |]) + | + |@timestampFormat("date-time") + |timestamp MyTimestamp + |""".stripMargin + + TestUtils.runConversionTest(openapiString, expectedString) + } + + test("simple date with example") { + val openapiString = """|openapi: '3.0.' + |info: + | title: test + | version: '1.0' + |paths: {} + |components: + | schemas: + | MyDate: + | type: string + | format: date + | example: '2017-07-21' + |""".stripMargin + + val expectedString = """|namespace foo + | + |use alloy#dateFormat + |use alloy#dataExamples + | + |@dataExamples([ + | { + | json: "2017-07-21" + | } + |]) + |@dateFormat + |string MyDate + |""".stripMargin + + TestUtils.runConversionTest(openapiString, expectedString) + } } From 94d1d3c0eb56a50e69d5b6630e3d282362df329e Mon Sep 17 00:00:00 2001 From: Jeff Lewis Date: Thu, 30 Nov 2023 13:04:54 -0700 Subject: [PATCH 2/2] return null node to prevent match errors --- modules/openapi/src/internals/GetExtensions.scala | 1 + 1 file changed, 1 insertion(+) diff --git a/modules/openapi/src/internals/GetExtensions.scala b/modules/openapi/src/internals/GetExtensions.scala index cdf1a1c..65ab214 100644 --- a/modules/openapi/src/internals/GetExtensions.scala +++ b/modules/openapi/src/internals/GetExtensions.scala @@ -77,6 +77,7 @@ object GetExtensions { case c: java.util.Collection[_] => Node.fromNodes(c.asScala.map(anyToNode).toList.asJava) case j: JsonNode => jacksonToSmithy(j) + case _ => Node.nullNode() // if nothing is found, to prevent match errors } private def jacksonToSmithy(jn: JsonNode): Node = jn match {