2021-07-29 21:43:35 +03:00
|
|
|
{-# LANGUAGE DeriveDataTypeable #-}
|
|
|
|
{-# LANGUAGE FlexibleInstances #-}
|
2021-07-30 17:43:21 +03:00
|
|
|
{-# LANGUAGE TemplateHaskell #-}
|
2021-07-29 21:43:35 +03:00
|
|
|
{-# LANGUAGE TypeApplications #-}
|
|
|
|
|
|
|
|
module Analyzer.EvaluatorTest where
|
|
|
|
|
2022-05-03 21:34:25 +03:00
|
|
|
import qualified Data.Aeson as Aeson
|
2021-07-29 21:43:35 +03:00
|
|
|
import Data.Data (Data)
|
2021-12-07 16:44:01 +03:00
|
|
|
import Data.List.Split (splitOn)
|
2022-01-20 13:45:14 +03:00
|
|
|
import Data.Maybe (fromJust)
|
|
|
|
import qualified StrongPath as SP
|
2021-07-29 21:43:35 +03:00
|
|
|
import Test.Tasty.Hspec
|
2021-12-07 16:44:01 +03:00
|
|
|
import Text.Read (readMaybe)
|
2021-11-11 15:26:20 +03:00
|
|
|
import Wasp.Analyzer.Evaluator
|
2021-12-07 16:44:01 +03:00
|
|
|
import qualified Wasp.Analyzer.Evaluator.Evaluation as E
|
|
|
|
import qualified Wasp.Analyzer.Evaluator.EvaluationError as EvaluationError
|
2022-03-30 02:25:56 +03:00
|
|
|
import Wasp.Analyzer.Parser (parseStatements)
|
2021-12-07 16:44:01 +03:00
|
|
|
import qualified Wasp.Analyzer.Type as T
|
2021-11-11 15:26:20 +03:00
|
|
|
import Wasp.Analyzer.TypeChecker (typeCheck)
|
2021-12-07 16:44:01 +03:00
|
|
|
import qualified Wasp.Analyzer.TypeChecker.AST as TypedAST
|
2021-11-11 15:26:20 +03:00
|
|
|
import qualified Wasp.Analyzer.TypeDefinitions as TD
|
2021-12-07 16:44:01 +03:00
|
|
|
import Wasp.Analyzer.TypeDefinitions.Class.HasCustomEvaluation (HasCustomEvaluation (..))
|
2021-11-25 01:18:25 +03:00
|
|
|
import Wasp.Analyzer.TypeDefinitions.TH
|
2021-11-11 15:26:20 +03:00
|
|
|
import Wasp.AppSpec.Core.Decl (IsDecl)
|
|
|
|
import Wasp.AppSpec.Core.Ref (Ref (..))
|
2021-11-25 12:47:50 +03:00
|
|
|
import Wasp.AppSpec.ExtImport (ExtImport (..), ExtImportName (..))
|
|
|
|
import Wasp.AppSpec.JSON (JSON (..))
|
2021-07-29 21:43:35 +03:00
|
|
|
|
|
|
|
fromRight :: Show a => Either a b -> b
|
|
|
|
fromRight (Right x) = x
|
|
|
|
fromRight (Left e) = error $ show e
|
|
|
|
|
2021-12-07 16:44:01 +03:00
|
|
|
------- Simple -------
|
|
|
|
|
2021-07-29 21:43:35 +03:00
|
|
|
newtype Simple = Simple String deriving (Eq, Show, Data)
|
|
|
|
|
2021-11-10 18:27:27 +03:00
|
|
|
instance IsDecl Simple
|
|
|
|
|
2021-09-14 18:08:53 +03:00
|
|
|
makeDeclType ''Simple
|
2021-07-29 21:43:35 +03:00
|
|
|
|
2021-12-07 16:44:01 +03:00
|
|
|
------- Fields -------
|
|
|
|
|
2021-07-29 21:43:35 +03:00
|
|
|
data Fields = Fields {a :: String, b :: Maybe Double} deriving (Eq, Show, Data)
|
|
|
|
|
2021-11-10 18:27:27 +03:00
|
|
|
instance IsDecl Fields
|
|
|
|
|
2021-09-14 18:08:53 +03:00
|
|
|
makeDeclType ''Fields
|
2021-07-29 21:43:35 +03:00
|
|
|
|
2021-12-07 16:44:01 +03:00
|
|
|
------ Business ------
|
|
|
|
|
2021-07-29 21:43:35 +03:00
|
|
|
data Person = Person {name :: String, age :: Integer} deriving (Eq, Show, Data)
|
|
|
|
|
2021-11-10 18:27:27 +03:00
|
|
|
instance IsDecl Person
|
|
|
|
|
2021-09-14 18:08:53 +03:00
|
|
|
makeDeclType ''Person
|
2021-07-30 17:43:21 +03:00
|
|
|
|
2021-07-29 21:43:35 +03:00
|
|
|
data BusinessType = Manufacturer | Seller | Store deriving (Eq, Show, Data)
|
|
|
|
|
2021-09-14 18:08:53 +03:00
|
|
|
makeEnumType ''BusinessType
|
2021-07-30 17:43:21 +03:00
|
|
|
|
2021-07-29 21:43:35 +03:00
|
|
|
data Business = Business
|
2021-10-22 12:59:51 +03:00
|
|
|
{ employees :: [Ref Person],
|
2021-07-29 21:43:35 +03:00
|
|
|
worth :: Double,
|
|
|
|
businessType :: BusinessType,
|
|
|
|
location :: Maybe String
|
|
|
|
}
|
|
|
|
deriving (Eq, Show, Data)
|
|
|
|
|
2021-11-10 18:27:27 +03:00
|
|
|
instance IsDecl Business
|
|
|
|
|
2021-09-14 18:08:53 +03:00
|
|
|
makeDeclType ''Business
|
2021-07-29 21:43:35 +03:00
|
|
|
|
2021-12-07 16:44:01 +03:00
|
|
|
-------- Special --------
|
|
|
|
|
|
|
|
data Special = Special {imps :: [ExtImport], json :: JSON} deriving (Eq, Show)
|
|
|
|
|
|
|
|
instance IsDecl Special
|
|
|
|
|
|
|
|
makeDeclType ''Special
|
|
|
|
|
|
|
|
------ HasCustomEvaluation ------
|
|
|
|
|
|
|
|
data SemanticVersion = SemanticVersion Int Int Int
|
|
|
|
deriving (Eq, Show, Data)
|
|
|
|
|
|
|
|
instance HasCustomEvaluation SemanticVersion where
|
|
|
|
waspType = T.StringType
|
2022-01-09 01:52:26 +03:00
|
|
|
evaluation = E.evaluation' . TypedAST.withCtx $ \ctx -> \case
|
2021-12-07 16:44:01 +03:00
|
|
|
TypedAST.StringLiteral str -> case splitOn "." str of
|
|
|
|
[major, minor, patch] ->
|
|
|
|
maybe
|
|
|
|
( Left $
|
2022-01-09 01:52:26 +03:00
|
|
|
EvaluationError.mkEvaluationError ctx $
|
|
|
|
EvaluationError.ParseError $
|
|
|
|
EvaluationError.EvaluationParseError
|
|
|
|
"Failed parsing semantic version -> some part is not int"
|
2021-12-07 16:44:01 +03:00
|
|
|
)
|
|
|
|
pure
|
|
|
|
$ do
|
|
|
|
majorInt <- readMaybe @Int major
|
|
|
|
minorInt <- readMaybe @Int minor
|
|
|
|
patchInt <- readMaybe @Int patch
|
|
|
|
return $ SemanticVersion majorInt minorInt patchInt
|
|
|
|
_ ->
|
|
|
|
Left $
|
2022-01-09 01:52:26 +03:00
|
|
|
EvaluationError.mkEvaluationError ctx $
|
|
|
|
EvaluationError.ParseError $
|
|
|
|
EvaluationError.EvaluationParseError
|
|
|
|
"Failed parsing semantic version -> it doesn't have 3 comma separated parts."
|
|
|
|
expr ->
|
|
|
|
Left $
|
|
|
|
EvaluationError.mkEvaluationError ctx $
|
|
|
|
EvaluationError.ExpectedType T.StringType (TypedAST.exprType expr)
|
2021-12-07 16:44:01 +03:00
|
|
|
|
|
|
|
data Custom = Custom
|
|
|
|
{version :: SemanticVersion}
|
|
|
|
deriving (Eq, Show, Data)
|
|
|
|
|
|
|
|
instance IsDecl Custom
|
|
|
|
|
|
|
|
makeDeclType ''Custom
|
|
|
|
|
|
|
|
--------------------------------
|
|
|
|
|
2021-12-04 19:24:04 +03:00
|
|
|
------ Tuples ------
|
|
|
|
data Tuples = Tuples
|
|
|
|
{ pair :: (String, Integer),
|
|
|
|
triple :: (String, Integer, Integer),
|
|
|
|
quadruple :: (String, Integer, Integer, [Bool])
|
|
|
|
}
|
|
|
|
deriving (Eq, Show, Data)
|
|
|
|
|
|
|
|
instance IsDecl Tuples
|
|
|
|
|
|
|
|
makeDeclType ''Tuples
|
|
|
|
|
|
|
|
--------------------
|
|
|
|
|
2022-05-03 21:34:25 +03:00
|
|
|
-------- Special --------
|
|
|
|
|
|
|
|
data AllJson = AllJson
|
|
|
|
{ objectValue :: JSON,
|
|
|
|
arrayValue :: JSON,
|
|
|
|
stringValue :: JSON,
|
|
|
|
nullValue :: JSON,
|
|
|
|
booleanValue :: JSON
|
|
|
|
}
|
|
|
|
deriving (Eq, Show)
|
|
|
|
|
|
|
|
instance IsDecl AllJson
|
|
|
|
|
|
|
|
makeDeclType ''AllJson
|
|
|
|
|
2021-08-05 22:22:21 +03:00
|
|
|
eval :: TD.TypeDefinitions -> [String] -> Either EvaluationError [Decl]
|
2022-03-30 02:25:56 +03:00
|
|
|
eval typeDefs source = evaluate typeDefs . fromRight . typeCheck typeDefs . fromRight . parseStatements $ unlines source
|
2021-08-05 22:22:21 +03:00
|
|
|
|
2021-07-29 21:43:35 +03:00
|
|
|
spec_Evaluator :: Spec
|
|
|
|
spec_Evaluator = do
|
|
|
|
describe "Analyzer.Evaluator" $
|
|
|
|
describe "evaluate" $ do
|
|
|
|
it "Evaluates a simple declaration" $ do
|
|
|
|
let typeDefs = TD.addDeclType @Simple $ TD.empty
|
2021-08-05 22:22:21 +03:00
|
|
|
let decls = eval typeDefs ["simple Test \"hello wasp\""]
|
|
|
|
fmap takeDecls decls
|
2021-07-29 21:43:35 +03:00
|
|
|
`shouldBe` Right [("Test", Simple "hello wasp")]
|
|
|
|
it "Evaluates a declaration with a dictionary" $ do
|
|
|
|
let typeDefs = TD.addDeclType @Fields $ TD.empty
|
2021-08-05 22:22:21 +03:00
|
|
|
let decls = eval typeDefs ["fields Test { a: \"hello wasp\", b: 3.14 }"]
|
|
|
|
fmap takeDecls decls
|
2021-07-29 21:43:35 +03:00
|
|
|
`shouldBe` Right [("Test", Fields {a = "hello wasp", b = Just 3.14})]
|
|
|
|
it "Evaluates a declaration with missing optional fields" $ do
|
|
|
|
let typeDefs = TD.addDeclType @Fields $ TD.empty
|
2021-08-05 22:22:21 +03:00
|
|
|
let decls = eval typeDefs ["fields Test { a: \"hello wasp\" }"]
|
|
|
|
fmap takeDecls decls
|
2021-07-29 21:43:35 +03:00
|
|
|
`shouldBe` Right [("Test", Fields {a = "hello wasp", b = Nothing})]
|
|
|
|
it "Evaluates a complicated example" $ do
|
|
|
|
let typeDefs =
|
|
|
|
TD.addDeclType @Business $
|
|
|
|
TD.addEnumType @BusinessType $
|
|
|
|
TD.addDeclType @Person $ TD.empty
|
|
|
|
let source =
|
|
|
|
[ "person Tim { name: \"Tim Stocker\", age: 40 }",
|
|
|
|
"person John { name: \"John Cashier\", age: 23 }",
|
|
|
|
"business Grocer { employees: [Tim, John], businessType: Store, worth: 115 }"
|
|
|
|
]
|
2021-08-05 22:22:21 +03:00
|
|
|
fmap takeDecls (eval typeDefs source)
|
2021-07-29 21:43:35 +03:00
|
|
|
`shouldBe` Right
|
|
|
|
[ ( "Grocer",
|
|
|
|
Business
|
2021-10-22 12:59:51 +03:00
|
|
|
{ employees = [Ref "Tim", Ref "John"],
|
2021-07-29 21:43:35 +03:00
|
|
|
businessType = Store,
|
|
|
|
worth = 115.0,
|
|
|
|
location = Nothing
|
|
|
|
}
|
|
|
|
)
|
|
|
|
]
|
2021-11-26 17:26:03 +03:00
|
|
|
it "Evaluates ExtImports and JSON" $ do
|
2021-08-05 22:22:21 +03:00
|
|
|
let typeDefs = TD.addDeclType @Special $ TD.empty
|
|
|
|
let source =
|
|
|
|
[ "special Test {",
|
2022-11-11 19:20:49 +03:00
|
|
|
" imps: [import { field } from \"@server/main.js\", import main from \"@server/main.js\"],",
|
2022-05-03 21:34:25 +03:00
|
|
|
" json: {=json { \"key\": 1 } json=}",
|
2021-08-05 22:22:21 +03:00
|
|
|
"}"
|
|
|
|
]
|
2021-12-07 16:44:01 +03:00
|
|
|
|
2021-08-05 22:22:21 +03:00
|
|
|
fmap takeDecls (eval typeDefs source)
|
|
|
|
`shouldBe` Right
|
|
|
|
[ ( "Test",
|
|
|
|
Special
|
2022-01-20 13:45:14 +03:00
|
|
|
[ ExtImport (ExtImportField "field") (fromJust $ SP.parseRelFileP "main.js"),
|
|
|
|
ExtImport (ExtImportModule "main") (fromJust $ SP.parseRelFileP "main.js")
|
2021-12-04 19:24:04 +03:00
|
|
|
]
|
2022-05-03 21:34:25 +03:00
|
|
|
(JSON $ Aeson.object ["key" Aeson..= (1 :: Integer)])
|
2021-08-05 22:22:21 +03:00
|
|
|
)
|
|
|
|
]
|
2021-12-07 16:44:01 +03:00
|
|
|
|
2022-05-03 21:34:25 +03:00
|
|
|
it "Evaluates JSON quoters and they show correctly" $ do
|
|
|
|
let typeDefs = TD.addDeclType @AllJson $ TD.empty
|
|
|
|
let source =
|
|
|
|
[ "allJson Test {",
|
|
|
|
" objectValue: {=json { \"key\": 1 } json=},",
|
|
|
|
" arrayValue: {=json [1, 2, 3] json=},",
|
|
|
|
" stringValue: {=json \"hello\" json=},",
|
|
|
|
" nullValue: {=json null json=},",
|
|
|
|
" booleanValue: {=json false json=},",
|
|
|
|
"}"
|
|
|
|
]
|
|
|
|
let Right [("Test", allJson)] = takeDecls <$> eval typeDefs source
|
|
|
|
show (objectValue allJson) `shouldBe` "{\"key\":1}"
|
|
|
|
show (arrayValue allJson) `shouldBe` "[1,2,3]"
|
|
|
|
show (stringValue allJson) `shouldBe` "\"hello\""
|
|
|
|
show (nullValue allJson) `shouldBe` "null"
|
|
|
|
show (booleanValue allJson) `shouldBe` "false"
|
|
|
|
|
2021-12-07 16:44:01 +03:00
|
|
|
it "Evaluates a declaration with a field that has custom evaluation" $ do
|
|
|
|
let typeDefs = TD.addDeclType @Custom $ TD.empty
|
|
|
|
let decls = eval typeDefs ["custom Test { version: \"1.2.3\" }"]
|
|
|
|
fmap takeDecls decls
|
|
|
|
`shouldBe` Right [("Test", Custom {version = SemanticVersion 1 2 3})]
|
2021-12-04 19:24:04 +03:00
|
|
|
|
|
|
|
it "Evaluates a declaration with fields that are tuples" $ do
|
|
|
|
let typeDefs = TD.addDeclType @Tuples $ TD.empty
|
|
|
|
let source =
|
|
|
|
[ "tuples Tuples {",
|
|
|
|
" pair: (\"foo\", 1),",
|
|
|
|
" triple: (\"foo\", 1, 2),",
|
|
|
|
" quadruple: (\"foo\", 1, 2, [true, false])",
|
|
|
|
"}"
|
|
|
|
]
|
|
|
|
fmap takeDecls (eval typeDefs source)
|
|
|
|
`shouldBe` Right
|
|
|
|
[ ( "Tuples",
|
|
|
|
Tuples
|
|
|
|
{ pair = ("foo", 1),
|
|
|
|
triple = ("foo", 1, 2),
|
|
|
|
quadruple = ("foo", 1, 2, [True, False])
|
|
|
|
}
|
|
|
|
)
|
|
|
|
]
|