2018-06-29 22:08:26 +03:00
|
|
|
{-# LANGUAGE ConstraintKinds #-}
|
|
|
|
{-# LANGUAGE DataKinds #-}
|
|
|
|
{-# LANGUAGE FlexibleContexts #-}
|
|
|
|
{-# LANGUAGE FlexibleInstances #-}
|
|
|
|
{-# LANGUAGE MultiParamTypeClasses #-}
|
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
|
|
|
{-# LANGUAGE PolyKinds #-}
|
|
|
|
{-# LANGUAGE ScopedTypeVariables #-}
|
|
|
|
{-# LANGUAGE TypeFamilies #-}
|
|
|
|
{-# LANGUAGE TypeOperators #-}
|
2015-11-04 11:16:14 +00:00
|
|
|
|
|
|
|
module Servant.ForeignSpec where
|
|
|
|
|
2018-06-29 22:08:26 +03:00
|
|
|
import Data.Proxy
|
|
|
|
import Servant.Foreign
|
2020-11-18 21:57:20 +03:00
|
|
|
import Servant.Test.ComprehensiveAPI
|
2018-06-26 20:11:28 +03:00
|
|
|
import Servant.Types.SourceT
|
|
|
|
(SourceT)
|
2015-11-04 11:16:14 +00:00
|
|
|
|
2018-06-29 22:08:26 +03:00
|
|
|
import Test.Hspec
|
2015-11-04 11:16:14 +00:00
|
|
|
|
2017-01-20 00:09:54 +02:00
|
|
|
|
2015-11-04 11:16:14 +00:00
|
|
|
spec :: Spec
|
|
|
|
spec = describe "Servant.Foreign" $ do
|
|
|
|
camelCaseSpec
|
2015-12-02 12:28:04 +00:00
|
|
|
listFromAPISpec
|
2015-11-04 11:16:14 +00:00
|
|
|
|
|
|
|
camelCaseSpec :: Spec
|
|
|
|
camelCaseSpec = describe "camelCase" $ do
|
|
|
|
it "converts FunctionNames to camelCase" $ do
|
2016-02-18 00:47:30 +03:00
|
|
|
camelCase (FunctionName ["post", "counter", "inc"])
|
|
|
|
`shouldBe` "postCounterInc"
|
2017-01-16 11:51:35 +02:00
|
|
|
camelCase (FunctionName ["get", "hyphen-ated", "counter"])
|
|
|
|
`shouldBe` "getHyphen-atedCounter"
|
|
|
|
|
2017-01-20 00:09:54 +02:00
|
|
|
----------------------------------------------------------------------
|
|
|
|
|
|
|
|
-- This declaration simply checks that all instances are in place.
|
|
|
|
_ = listFromAPI (Proxy :: Proxy LangX) (Proxy :: Proxy String) comprehensiveAPIWithoutRaw
|
2015-12-02 12:28:04 +00:00
|
|
|
|
|
|
|
----------------------------------------------------------------------
|
|
|
|
|
|
|
|
data LangX
|
|
|
|
|
2016-07-08 09:11:34 +02:00
|
|
|
instance HasForeignType LangX String NoContent where
|
2016-03-14 10:21:36 +13:00
|
|
|
typeFor _ _ _ = "voidX"
|
2016-02-11 13:41:34 +03:00
|
|
|
|
2017-01-20 00:09:54 +02:00
|
|
|
instance HasForeignType LangX String (Headers ctyps NoContent) where
|
|
|
|
typeFor _ _ _ = "voidX"
|
|
|
|
|
2016-03-14 10:21:36 +13:00
|
|
|
instance HasForeignType LangX String Int where
|
|
|
|
typeFor _ _ _ = "intX"
|
2016-02-11 13:41:34 +03:00
|
|
|
|
2018-06-26 20:11:28 +03:00
|
|
|
instance HasForeignType LangX String (SourceT m a) where
|
|
|
|
typeFor _ _ _ = "streamTX"
|
|
|
|
|
2016-03-14 10:21:36 +13:00
|
|
|
instance HasForeignType LangX String Bool where
|
|
|
|
typeFor _ _ _ = "boolX"
|
2016-02-11 13:41:34 +03:00
|
|
|
|
2018-07-11 01:39:38 +03:00
|
|
|
instance {-# OVERLAPPING #-} HasForeignType LangX String String where
|
2016-03-14 10:21:36 +13:00
|
|
|
typeFor _ _ _ = "stringX"
|
2016-02-11 13:41:34 +03:00
|
|
|
|
2018-07-11 01:39:38 +03:00
|
|
|
instance {-# OVERLAPPABLE #-} HasForeignType LangX String a => HasForeignType LangX String [a] where
|
2016-03-14 10:21:36 +13:00
|
|
|
typeFor lang ftype _ = "listX of " <> typeFor lang ftype (Proxy :: Proxy a)
|
2015-12-02 12:28:04 +00:00
|
|
|
|
2017-11-27 00:23:55 +01:00
|
|
|
instance (HasForeignType LangX String a) => HasForeignType LangX String (Maybe a) where
|
|
|
|
typeFor lang ftype _ = "maybe " <> typeFor lang ftype (Proxy :: Proxy a)
|
|
|
|
|
2019-09-30 17:59:56 -07:00
|
|
|
data ContactForm = ContactForm {
|
|
|
|
name :: String
|
|
|
|
, message :: String
|
|
|
|
, email :: String
|
|
|
|
} deriving (Eq, Show)
|
|
|
|
|
|
|
|
instance HasForeignType LangX String ContactForm where
|
|
|
|
typeFor _ _ _ = "contactFormX"
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-12-02 12:28:04 +00:00
|
|
|
type TestApi
|
|
|
|
= "test" :> Header "header" [String] :> QueryFlag "flag" :> Get '[JSON] Int
|
2016-07-08 09:11:34 +02:00
|
|
|
:<|> "test" :> QueryParam "param" Int :> ReqBody '[JSON] [String] :> Post '[JSON] NoContent
|
2019-10-02 20:47:54 -07:00
|
|
|
:<|> "test" :> QueryParamForm ContactForm :> Post '[JSON] NoContent
|
2016-07-08 09:11:34 +02:00
|
|
|
:<|> "test" :> QueryParams "params" Int :> ReqBody '[JSON] String :> Put '[JSON] NoContent
|
|
|
|
:<|> "test" :> Capture "id" Int :> Delete '[JSON] NoContent
|
2016-05-26 20:51:28 +01:00
|
|
|
:<|> "test" :> CaptureAll "ids" Int :> Get '[JSON] [Int]
|
2017-05-16 10:31:33 +00:00
|
|
|
:<|> "test" :> EmptyAPI
|
2015-12-02 12:28:04 +00:00
|
|
|
|
2016-03-14 10:21:36 +13:00
|
|
|
testApi :: [Req String]
|
|
|
|
testApi = listFromAPI (Proxy :: Proxy LangX) (Proxy :: Proxy String) (Proxy :: Proxy TestApi)
|
2015-12-02 12:28:04 +00:00
|
|
|
|
|
|
|
listFromAPISpec :: Spec
|
|
|
|
listFromAPISpec = describe "listFromAPI" $ do
|
2017-05-16 10:31:02 +00:00
|
|
|
it "generates 5 endpoints for TestApi" $ do
|
2019-09-30 17:59:56 -07:00
|
|
|
length testApi `shouldBe` 6
|
2016-02-11 13:41:34 +03:00
|
|
|
|
2019-09-30 17:59:56 -07:00
|
|
|
let [getReq, postReq, contactReq, putReq, deleteReq, captureAllReq] = testApi
|
2016-02-11 13:41:34 +03:00
|
|
|
|
|
|
|
it "collects all info for get request" $ do
|
|
|
|
shouldBe getReq $ defReq
|
|
|
|
{ _reqUrl = Url
|
|
|
|
[ Segment $ Static "test" ]
|
2016-02-18 00:47:30 +03:00
|
|
|
[ QueryArg (Arg "flag" "boolX") Flag ]
|
2020-11-18 21:57:20 +03:00
|
|
|
Nothing
|
2016-02-11 13:41:34 +03:00
|
|
|
, _reqMethod = "GET"
|
2017-11-27 00:23:55 +01:00
|
|
|
, _reqHeaders = [HeaderArg $ Arg "header" "maybe listX of stringX"]
|
2016-02-11 13:41:34 +03:00
|
|
|
, _reqBody = Nothing
|
2016-03-14 10:21:36 +13:00
|
|
|
, _reqReturnType = Just "intX"
|
2016-02-18 00:47:30 +03:00
|
|
|
, _reqFuncName = FunctionName ["get", "test"]
|
2016-02-11 13:41:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
it "collects all info for post request" $ do
|
|
|
|
shouldBe postReq $ defReq
|
|
|
|
{ _reqUrl = Url
|
|
|
|
[ Segment $ Static "test" ]
|
2017-12-10 14:25:14 +02:00
|
|
|
[ QueryArg (Arg "param" "maybe intX") Normal ]
|
2020-11-18 21:57:20 +03:00
|
|
|
Nothing
|
2016-02-11 13:41:34 +03:00
|
|
|
, _reqMethod = "POST"
|
|
|
|
, _reqHeaders = []
|
|
|
|
, _reqBody = Just "listX of stringX"
|
2016-03-14 10:21:36 +13:00
|
|
|
, _reqReturnType = Just "voidX"
|
2016-02-18 00:47:30 +03:00
|
|
|
, _reqFuncName = FunctionName ["post", "test"]
|
2016-02-11 13:41:34 +03:00
|
|
|
}
|
|
|
|
|
2019-09-30 17:59:56 -07:00
|
|
|
it "collects all info for a queryparamform" $ do
|
|
|
|
shouldBe contactReq $ defReq
|
|
|
|
{ _reqUrl = Url
|
|
|
|
[ Segment $ Static "test" ]
|
|
|
|
[ QueryArg (Arg "" "maybe contactFormX") Form ]
|
2021-11-17 18:30:22 -08:00
|
|
|
Nothing
|
2019-09-30 17:59:56 -07:00
|
|
|
, _reqMethod = "POST"
|
|
|
|
, _reqHeaders = []
|
2021-11-17 18:30:22 -08:00
|
|
|
, _reqBody = Nothing
|
2019-09-30 17:59:56 -07:00
|
|
|
, _reqReturnType = Just "voidX"
|
|
|
|
, _reqFuncName = FunctionName ["post", "test"]
|
|
|
|
}
|
|
|
|
|
2016-02-11 13:41:34 +03:00
|
|
|
it "collects all info for put request" $ do
|
|
|
|
shouldBe putReq $ defReq
|
|
|
|
{ _reqUrl = Url
|
|
|
|
[ Segment $ Static "test" ]
|
2020-06-06 06:43:51 +02:00
|
|
|
-- Should this be |intX| or |listX of intX| ?
|
2016-02-18 00:47:30 +03:00
|
|
|
[ QueryArg (Arg "params" "listX of intX") List ]
|
2020-11-18 21:57:20 +03:00
|
|
|
Nothing
|
2016-02-11 13:41:34 +03:00
|
|
|
, _reqMethod = "PUT"
|
|
|
|
, _reqHeaders = []
|
|
|
|
, _reqBody = Just "stringX"
|
2016-03-14 10:21:36 +13:00
|
|
|
, _reqReturnType = Just "voidX"
|
2016-02-18 00:47:30 +03:00
|
|
|
, _reqFuncName = FunctionName ["put", "test"]
|
2016-02-11 13:41:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
it "collects all info for delete request" $ do
|
|
|
|
shouldBe deleteReq $ defReq
|
|
|
|
{ _reqUrl = Url
|
|
|
|
[ Segment $ Static "test"
|
2016-02-18 00:47:30 +03:00
|
|
|
, Segment $ Cap (Arg "id" "intX") ]
|
2016-02-11 13:41:34 +03:00
|
|
|
[]
|
2020-11-18 21:57:20 +03:00
|
|
|
Nothing
|
2016-02-11 13:41:34 +03:00
|
|
|
, _reqMethod = "DELETE"
|
|
|
|
, _reqHeaders = []
|
|
|
|
, _reqBody = Nothing
|
2016-03-14 10:21:36 +13:00
|
|
|
, _reqReturnType = Just "voidX"
|
2016-02-18 00:47:30 +03:00
|
|
|
, _reqFuncName = FunctionName ["delete", "test", "by", "id"]
|
2016-02-11 13:41:34 +03:00
|
|
|
}
|
2016-05-26 20:51:28 +01:00
|
|
|
|
|
|
|
it "collects all info for capture all request" $ do
|
|
|
|
shouldBe captureAllReq $ defReq
|
|
|
|
{ _reqUrl = Url
|
|
|
|
[ Segment $ Static "test"
|
|
|
|
, Segment $ Cap (Arg "ids" "listX of intX") ]
|
|
|
|
[]
|
2020-11-18 21:57:20 +03:00
|
|
|
Nothing
|
2016-05-26 20:51:28 +01:00
|
|
|
, _reqMethod = "GET"
|
|
|
|
, _reqHeaders = []
|
|
|
|
, _reqBody = Nothing
|
|
|
|
, _reqReturnType = Just "listX of intX"
|
|
|
|
, _reqFuncName = FunctionName ["get", "test", "by", "ids"]
|
|
|
|
}
|
2019-09-30 17:59:56 -07:00
|
|
|
|