Skip to content

Latest commit

 

History

History
189 lines (154 loc) · 5.17 KB

README.md

File metadata and controls

189 lines (154 loc) · 5.17 KB

simple-scala-generator

Interpretation of the OpenAPI Specification

This generator is intended to target API specification files adhering to the OpenAPI version 2.0 specification.

Treatment of enum

The OpenAPI specification defers to the [JSON-Schema Draft 4] for its treatment of enum. We currently support both standalone definitions defined as enums themselves as well as inline enums.

Enum values must be heterogeneous. Supported value types are a subset of that allowed in JSON-schema, for simplicity's sake. Specifically, these are: boolean, integer, number, and string.

The code generation strategy for both standalone and embedded enums is the same:

  • sealed type hierarchy
  • instances as case objects with both the toString and value members rendering the original value.
  • a companion with the instances available as a Set, as well as a function, fromValue, for translating a raw value
  • into an instance

Consider this standalone example:

  psp:
    description: The payment service provider.
    type: string
    enum:
      - apple
      - stripe

This would result in the following scala code:

sealed abstract class Psp(val value: String) extends Product with Serializable
case object Apple extends Psp("apple")
case object Stripe extends Psp("stripe")

object Psp {
  private val valueMap =
    Map(
      "apple" -> Apple,
      "stripe" -> Stripe
    )

  val values: Set[Psp] = valueMap.values.toSet

  def fromValue(value: String): Option[Psp] = valueMap.get(value)
}

Consider this embedded example:

  subscription:
    description: A Subscription.
    properties:
      psp:
        type: string
        enum:
          - apple
          - stripe

This would result in the following scala code:

final case class Subscription(psp: Option[Subscription.Psp])

object Subscription {
  sealed abstract class Psp(val value: String) extends Product with Serializable
  case object Apple extends Psp("apple")
  case object Stripe extends Psp("stripe")

  object Psp {
    private val valueMap =
      Map(
        "apple" -> Apple,
        "stripe" -> Stripe
      )

    val values: Set[Psp] = valueMap.values.toSet

    def fromValue(value: String): Option[Psp] = valueMap.get(value)
  }
}

Note that we infer the root enum type from the property name. We also place the hierarchy within the companion object of the class generated to represent the containing model in the specification.

Handling Subtype Polymorphism

Note: proper support is still pending - please refrain from use at this time.

The OpenAPI version 2.0 specification explicitly supports subtype polymorphism, or "inheritance", through a combination of its interpretation of the JSON-Schema allOf key word and the OpenAPI discriminator field. The following defines an subtype relationship between to sibling models, Triangle and Rectangle, and their parent, Shape:

  Shape:
    type: object
    discriminator: shapeType
    properties:
      name:
        type: string
      shapeType:
        type: string
    required:
      - name
      - shapeType
  Triangle:
    description: A representation of a triangle
    allOf:
    - $ref: "#/definitions/Shape"
    - type: object
      properties:
        base:
          type: integer
          description: The base length
        height:
          type: integer
          description: The height
      required:
        - base
        - height
  Rectangle:
    description: A representation of a rectangle
    allOf:
    - $ref: "#/definitions/Shape"
    - type: object
      properties:
        length:
          type: integer
          description: The length of the rectangle
        width:
          type: integer
          description: The width of the rectangle
      required:
        - length
        - width

Which will result in the following class hierarchy:

trait Shape {
  def name: String

  def shapeType: String
}
/**
 * A representation of a triangle
 */
final case class Triangle (
  name: String,
  shapeType: String,
  /* The base length */
  base: Int,
  /* The height */
  height: Int) extends Shape
/**
 * A representation of a rectangle
 */
final case class Rectangle (
  name: String,
  shapeType: String,
  /* The length of the rectangle */
  length: Int,
  /* The width of the rectangle */
  width: Int) extends Shape

Additional Date and Time Types

Data types in swagger have an optional modifier, format. This is an open string-valued property that can be used to further inform the generator of a more specific type. This generator exploits this modifier for string properties to add the following additional date/time properties: timestamp, local-time, and local-date-time.

Further, these types are mapped to java.time.{Instant, LocalTime, LocalDateTime}, respectively.