Skip to content

Commit

Permalink
Add export of 'Typ' to OpenAPI schema
Browse files Browse the repository at this point in the history
  • Loading branch information
HeinrichApfelmus committed Aug 29, 2023
1 parent 1276c09 commit 8d5f28a
Show file tree
Hide file tree
Showing 5 changed files with 417 additions and 2 deletions.
26 changes: 26 additions & 0 deletions custom.ghci
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
-- Generated by scripts/gen-ghci.sh for version 0.1.0.0
-- mar 29 ago 2023, 09:38:17, CEST

:set -XOverloadedStrings
:set -XNoImplicitPrelude
:set -XTypeApplications
:set -XDataKinds
:set -fwarn-unused-binds
:set -fwarn-unused-imports
:set -fwarn-orphans
:set -fprint-potential-instances
:set -Wno-missing-home-modules
:set -Wredundant-constraints
:set -ilib/fine-types/src
:set -ilib/fine-types/test
:set -idist-newstyle/build/x86_64-linux/ghc-9.2.8/fine-types-0.1.0.0/noopt/build/autogen

:set prompt "λ "
:set -package pretty-simple
:set -interactive-print=Text.Pretty.Simple.pPrint

import Prelude
import System.Environment (setEnv, getEnv)
import System.Directory (getCurrentDirectory)

getCurrentDirectory >>= \d -> getEnv "PATH" >>= \p -> setEnv "PATH" (d ++ ".:" ++ p)
9 changes: 7 additions & 2 deletions lib/fine-types/fine-types.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -48,18 +48,22 @@ library
hs-source-dirs:
src
build-depends:
, aeson ^>= 2.1.2
, base >=4.14.3.0
, bytestring
, containers
, deepseq >= 1.4.4
, megaparsec ^>= 9.2.1
, parser-combinators
, pretty-simple
, prettyprinter
, QuickCheck
, text
, transformers
, prettyprinter
, yaml
exposed-modules:
Language.FineTypes
Language.FineTypes.Export.OpenAPI.Typ
Language.FineTypes.Module
Language.FineTypes.Module.Gen
Language.FineTypes.Module.PrettyPrinter
Expand All @@ -79,13 +83,14 @@ test-suite unit
build-depends:
base
, containers
, deepseq
, fine-types
, hspec ^>= 2.11.0
, QuickCheck
, pretty-simple
main-is:
Spec.hs
other-modules:
Language.FineTypes.Export.OpenAPI.TypSpec
Language.FineTypes.ParserSpec
Language.FineTypes.ValueSpec

270 changes: 270 additions & 0 deletions lib/fine-types/src/Language/FineTypes/Export/OpenAPI/Typ.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,270 @@
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE OverloadedStrings #-}

-- | Export type definitions to OpenAPI JSON schemas.
--
-- https://www.openapis.org
module Language.FineTypes.Export.OpenAPI.Typ
( OpenAPISchema (..)
, schemaFromModule
, supportsJSON
, convertToJSON
) where

import Prelude

import Control.DeepSeq
( NFData
)
import Data.Aeson
( (.=)
)
import Data.Text
( Text
)
import GHC.Generics
( Generic
)
import Language.FineTypes.Module
( Declarations
, Module (..)
, resolveVars
)
import Language.FineTypes.Typ
( ConstructorName
, FieldName
, OpOne (..)
, OpTwo (..)
, Typ (..)
, TypConst (..)
, everything
, everywhere
)

import qualified Data.Aeson as JS
import qualified Data.Aeson.Key as JS.Key
import qualified Data.Aeson.Types as JS
import qualified Data.Map as Map
import qualified Data.Text as T

{-----------------------------------------------------------------------------
OpenAPI
------------------------------------------------------------------------------}
newtype OpenAPISchema = OpenAPISchema {getOpenAPISchema :: JS.Value}
deriving (Eq, Ord, Show, Generic)

instance NFData OpenAPISchema

-- | Export
--
-- Assumes that the argument satisfies 'supportsJSON'.
schemaFromModule :: Module -> OpenAPISchema
schemaFromModule m =
OpenAPISchema
$ wrapSchemasInHeader
(T.pack $ moduleName m)
[ (T.pack name, schemaFromTyp typ)
| (name, typ) <- Map.toList declarations
]
where
declarations = moduleDeclarations m

-- | Test whether a 'Module' only uses types supported by JSON.
--
-- JSON does not support finite maps such as @↦@, @↦0@, @→∗@.
supportsJSON :: Module -> Bool
supportsJSON =
and . Map.map isSupportedTyp . moduleDeclarations
where
isSupportedTyp = everything (&&) isSupported
isSupported (Two fun _ _) =
fun `notElem` [PartialFunction, FiniteSupport]
isSupported _ = True

-- | Convert 'Typ' definitions to JSON.
--
-- The result satisfies 'supportsJSON'.
--
-- Note: We don't recommend that you use this function,
-- because it does a lot of conversions under the hood.
-- Instead, if you want to export a 'Typ' to JSON,
-- we recommend that you explicitly define a second 'Typ'
-- which is apparently compatible with JSON,
-- and show that the first 'Typ' can be embedded into the second 'Typ'.
convertToJSON :: Declarations -> Declarations
convertToJSON declarations = Map.map (jsonify declarations) declarations

{-----------------------------------------------------------------------------
Convert Typ to JSON schema
------------------------------------------------------------------------------}
wrapSchemasInHeader :: Text -> [(Text, JS.Value)] -> JS.Value
wrapSchemasInHeader title xs =
object
[ "openapi" .= s "3.0.3"
, "info"
.= object
[ "title" .= s title
, "version" .= s "1"
]
, "components"
.= object
[ "schemas"
.= object
[ key name .= x
| (name, x) <- xs
]
]
, "paths" .= object []
]

schemaFromTyp :: Typ -> JS.Value
schemaFromTyp = go
where
go Abstract =
object
["type" .= s "object"]
go (Var name) =
object
["$ref" .= s (T.pack $ "#/components/schemas/" <> name)]
go (Zero Bool) =
object
["type" .= s "boolean"]
go (Zero Bytes) =
object
[ "type" .= s "string"
, "format" .= s "base16"
]
go (Zero Integer) =
object
["type" .= s "integer"]
go (Zero Natural) =
object
[ "type" .= s "integer"
, "minimum" .= JS.toJSON (0 :: Int)
]
go (Zero Text) =
object
["type" .= s "string"]
go (Zero Unit) =
object
["type" .= s "null"]
go (One Option a) =
object
[ "type" .= s "object"
, "properties" .= object ["0" .= go a]
]
go (One Sequence a) =
object
[ "type" .= s "array"
, "items" .= go a
]
go (One PowerSet a) =
go (One Sequence a)
go (Two Sum2 a b) =
schemaFromSumN [("0", a), ("1", b)]
go (Two Product2 a b) =
object
[ "type" .= s "object"
, "properties" .= object ["0" .= go a, "1" .= go b]
, "required" .= array [s "0", s "1"]
, "additionalProperties" .= false
]
go (Two PartialFunction _ _) =
error "PartialFunction is not supported by JSON schema"
go (Two FiniteSupport _ _) =
error "FiniteSupport is not supported by JSON schema"
go (ProductN fields) =
schemaFromProductN fields
go (SumN constructors) =
schemaFromSumN constructors

-- | Map a record type to a JSON schema.
--
-- Field that are option types (@?@) will be mapped to optional fields.
schemaFromProductN :: [(FieldName, Typ)] -> JS.Value
schemaFromProductN fields =
object
[ "type" .= s "object"
, "properties"
.= object
[ key (T.pack name) .= schemaFromTyp (stripOption typ)
| (name, typ) <- fields
]
, "required" .= array required
, "additionalProperties" .= false
]
where
required =
[ s (T.pack name)
| (name, typ) <- fields
, not (isOption typ)
]

stripOption :: Typ -> Typ
stripOption (One Option a) = a
stripOption a = a

isOption :: Typ -> Bool
isOption (One Option _) = True
isOption _ = False

-- | Map a union type to a JSON.
--
-- The encoding corresponds to the 'ObjectWithSingleField' encoding.
schemaFromSumN :: [(ConstructorName, Typ)] -> JS.Value
schemaFromSumN constructors =
object ["oneOf" .= array (map fromConstructor constructors)]
where
fromConstructor (name, typ) =
object
[ "type" .= s "object"
, "title" .= s (T.pack name)
, "properties"
.= object [key (T.pack name) .= schemaFromTyp typ]
, "required" .= array [s (T.pack name)]
, "additionalProperties" .= false
]

{-----------------------------------------------------------------------------
Preprocessing
------------------------------------------------------------------------------}

-- | Modify the 'Typ' to be closer to JSON.
jsonify :: Declarations -> Typ -> Typ
jsonify declarations =
mergeRecords . representFiniteMaps . resolveVars declarations

representFiniteMaps :: Typ -> Typ
representFiniteMaps = everywhere represent
where
represent x@(Two op a b)
| op == FiniteSupport || op == PartialFunction =
One Sequence (Two Product2 a b)
| otherwise =
x
represent x = x

mergeRecords :: Typ -> Typ
mergeRecords = everywhere merge
where
merge (Two Product2 (ProductN a) (ProductN b)) =
ProductN (a <> b)
merge x = x

{-----------------------------------------------------------------------------
JSON helpers
------------------------------------------------------------------------------}
key :: Text -> JS.Key
key = JS.Key.fromText

s :: Text -> JS.Value
s = JS.String

object :: [JS.Pair] -> JS.Value
object = JS.object

array :: [JS.Value] -> JS.Value
array = JS.toJSON

false :: JS.Value
false = JS.toJSON False
47 changes: 47 additions & 0 deletions lib/fine-types/test/Language/FineTypes/Export/OpenAPI/TypSpec.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
module Language.FineTypes.Export.OpenAPI.TypSpec
( spec
) where

import Prelude

import Control.DeepSeq
( NFData
, rnf
)
import Language.FineTypes.Export.OpenAPI.Typ
( schemaFromModule
, supportsJSON
)
import Language.FineTypes.Parser
( parseFineTypes
)
import Test.Hspec
( Expectation
, Spec
, describe
, it
, shouldBe
)

{-----------------------------------------------------------------------------
Tests
------------------------------------------------------------------------------}
spec :: Spec
spec = do
describe "OpenAPI export on JsonUTxO.fine" $ do
let readModule = do
file <- readFile "test/data/JsonUTxO.fine"
Just m <- pure $ parseFineTypes file
pure m

it "has a 'Typ' that supports JSON" $ do
m <- readModule
supportsJSON m `shouldBe` True

it "works, i.e. does not contain ⊥" $ do
m <- readModule
let schema = schemaFromModule m
hasNormalForm schema

hasNormalForm :: NFData a => a -> Expectation
hasNormalForm x = rnf x `shouldBe` ()
Loading

0 comments on commit 8d5f28a

Please sign in to comment.