{-# LANGUAGE TypeApplications #-} module AppSpec.ValidTest where import Data.Maybe (fromJust) import Fixtures (systemSPRoot) import qualified StrongPath as SP import Test.Tasty.Hspec import qualified Wasp.AppSpec as AS import qualified Wasp.AppSpec.App as AS.App import qualified Wasp.AppSpec.App.Auth as AS.Auth import qualified Wasp.AppSpec.Core.Decl as AS.Decl import qualified Wasp.AppSpec.Core.Ref as AS.Core.Ref import qualified Wasp.AppSpec.Entity as AS.Entity import qualified Wasp.AppSpec.ExtImport as AS.ExtImport import qualified Wasp.AppSpec.Page as AS.Page import qualified Wasp.AppSpec.Valid as ASV import qualified Wasp.Psl.Ast.Model as PslM spec_AppSpecValid :: Spec spec_AppSpecValid = do describe "validateAppSpec" $ do describe "should validate that AppSpec has exactly 1 'app' declaration." $ do it "returns no error if there is exactly 1 'app' declaration." $ do ASV.validateAppSpec (basicAppSpec {AS.decls = [basicAppDecl]}) `shouldBe` [] it "returns an error if there is no 'app' declaration." $ do ASV.validateAppSpec (basicAppSpec {AS.decls = []}) `shouldBe` [ ASV.GenericValidationError "You are missing an 'app' declaration in your Wasp app." ] it "returns an error if there are 2 'app' declarations." $ do ASV.validateAppSpec ( basicAppSpec { AS.decls = [ AS.Decl.makeDecl "app1" basicApp, AS.Decl.makeDecl "app2" basicApp ] } ) `shouldBe` [ ASV.GenericValidationError "You have more than one 'app' declaration in your Wasp app. You have 2." ] describe "auth-related validation" $ do let userEntityName = "User" let validUserEntity = AS.Entity.makeEntity ( PslM.Body [ PslM.ElementField $ makeBasicPslField "email" PslM.String, PslM.ElementField $ makeBasicPslField "password" PslM.String ] ) let validAppAuth = AS.Auth.Auth { AS.Auth.userEntity = AS.Core.Ref.Ref userEntityName, AS.Auth.methods = [AS.Auth.EmailAndPassword], AS.Auth.onAuthFailedRedirectTo = "/", AS.Auth.onAuthSucceededRedirectTo = Nothing } describe "should validate that when a page has authRequired, app.auth is also set." $ do let makeSpec = \appAuth pageAuthRequired -> basicAppSpec { AS.decls = [ AS.Decl.makeDecl "TestApp" $ basicApp {AS.App.auth = appAuth}, AS.Decl.makeDecl "TestPage" $ basicPage {AS.Page.authRequired = pageAuthRequired}, AS.Decl.makeDecl userEntityName validUserEntity ] } it "returns no error if there is no page with authRequired and app.auth is not set" $ do ASV.validateAppSpec (makeSpec Nothing Nothing) `shouldBe` [] ASV.validateAppSpec (makeSpec Nothing (Just False)) `shouldBe` [] it "returns no error if there is a page with authRequired and app.auth is set" $ do ASV.validateAppSpec (makeSpec (Just validAppAuth) (Just True)) `shouldBe` [] it "returns an error if there is a page with authRequired and app.auth is not set" $ do ASV.validateAppSpec (makeSpec Nothing (Just True)) `shouldBe` [ ASV.GenericValidationError "Expected app.auth to be defined since there are Pages with authRequired set to true." ] describe "should validate that when app.auth is using EmailAndPassword, user entity is of valid shape." $ do let makeSpec = \appAuth userEntity -> basicAppSpec { AS.decls = [ AS.Decl.makeDecl "TestApp" $ basicApp {AS.App.auth = appAuth}, AS.Decl.makeDecl userEntityName (userEntity :: AS.Entity.Entity) ] } let invalidUserEntity = AS.Entity.makeEntity ( PslM.Body [ PslM.ElementField $ makeBasicPslField "username" PslM.String, PslM.ElementField $ makeBasicPslField "password" PslM.String ] ) let invalidUserEntity2 = AS.Entity.makeEntity ( PslM.Body [ PslM.ElementField $ makeBasicPslField "email" PslM.String ] ) it "returns no error if app.auth is not set, regardless of shape of user entity" $ do ASV.validateAppSpec (makeSpec Nothing invalidUserEntity) `shouldBe` [] ASV.validateAppSpec (makeSpec Nothing validUserEntity) `shouldBe` [] it "returns no error if app.auth is set and user entity is of valid shape" $ do ASV.validateAppSpec (makeSpec (Just validAppAuth) validUserEntity) `shouldBe` [] it "returns an error if app.auth is set and user entity is of invalid shape" $ do ASV.validateAppSpec (makeSpec (Just validAppAuth) invalidUserEntity) `shouldBe` [ ASV.GenericValidationError "Expected an Entity referenced by app.auth.userEntity to have field 'email' of type 'string'." ] ASV.validateAppSpec (makeSpec (Just validAppAuth) invalidUserEntity2) `shouldBe` [ ASV.GenericValidationError "Expected an Entity referenced by app.auth.userEntity to have field 'password' of type 'string'." ] where makeBasicPslField name typ = PslM.Field { PslM._name = name, PslM._type = typ, PslM._typeModifiers = [], PslM._attrs = [] } basicApp = AS.App.App { AS.App.title = "Test App", AS.App.db = Nothing, AS.App.server = Nothing, AS.App.auth = Nothing, AS.App.dependencies = Nothing, AS.App.head = Nothing } basicAppDecl = AS.Decl.makeDecl "TestApp" basicApp basicAppSpec = AS.AppSpec { AS.decls = [basicAppDecl], AS.externalCodeDirPath = systemSPRoot SP. [SP.reldir|test/src|], AS.externalCodeFiles = [], AS.isBuild = False, AS.migrationsDir = Nothing, AS.dotEnvFile = Nothing } basicPage = AS.Page.Page { AS.Page.component = AS.ExtImport.ExtImport (AS.ExtImport.ExtImportModule "Home") (fromJust $ SP.parseRelFileP "pages/Main"), AS.Page.authRequired = Nothing }