2015-09-21 12:31:00 +02:00
|
|
|
module Servant.JS.Internal
|
|
|
|
( JavaScriptGenerator
|
2015-09-22 11:21:04 +02:00
|
|
|
, CommonGeneratorOptions(..)
|
|
|
|
, defCommonGeneratorOptions
|
2015-09-21 12:31:00 +02:00
|
|
|
, AjaxReq
|
|
|
|
, jsSegments
|
|
|
|
, segmentToStr
|
|
|
|
, segmentTypeToStr
|
|
|
|
, jsParams
|
|
|
|
, jsGParams
|
|
|
|
, jsMParams
|
|
|
|
, paramToStr
|
2015-09-22 12:17:43 +02:00
|
|
|
, toValidFunctionName
|
|
|
|
, toJSHeader
|
2015-09-21 12:31:00 +02:00
|
|
|
-- re-exports
|
|
|
|
, (:<|>)(..)
|
|
|
|
, (:>)
|
|
|
|
, defReq
|
|
|
|
, reqHeaders
|
|
|
|
, HasForeign(..)
|
|
|
|
, HeaderArg(..)
|
|
|
|
, concatCase
|
|
|
|
, snakeCase
|
|
|
|
, camelCase
|
|
|
|
, ReqBody
|
|
|
|
, JSON
|
|
|
|
, FormUrlEncoded
|
|
|
|
, Post
|
|
|
|
, Get
|
|
|
|
, Raw
|
|
|
|
, Header
|
|
|
|
) where
|
2014-11-25 01:36:34 +01:00
|
|
|
|
2015-09-22 12:17:43 +02:00
|
|
|
import Control.Lens hiding (List)
|
|
|
|
import qualified Data.CharSet as Set
|
|
|
|
import qualified Data.CharSet.Unicode.Category as Set
|
|
|
|
import Data.List
|
|
|
|
import Data.Monoid
|
|
|
|
import qualified Data.Text as T
|
|
|
|
import Servant.Foreign
|
2014-11-25 01:36:34 +01:00
|
|
|
|
2015-09-21 12:31:00 +02:00
|
|
|
type AjaxReq = Req
|
2014-11-25 01:36:34 +01:00
|
|
|
|
2015-07-22 19:23:31 +02:00
|
|
|
-- A 'JavascriptGenerator' just takes the data found in the API type
|
|
|
|
-- for each endpoint and generates Javascript code in a String. Several
|
|
|
|
-- generators are available in this package.
|
2015-09-21 12:31:00 +02:00
|
|
|
type JavaScriptGenerator = [Req] -> String
|
2015-01-03 18:52:18 +01:00
|
|
|
|
2015-09-22 11:21:04 +02:00
|
|
|
-- | This structure is used by specific implementations to let you
|
|
|
|
-- customize the output
|
|
|
|
data CommonGeneratorOptions = CommonGeneratorOptions
|
|
|
|
{
|
|
|
|
functionNameBuilder :: FunctionName -> String -- ^ function generating function names
|
|
|
|
, requestBody :: String -- ^ name used when a user want to send the request body (to let you redefine it)
|
|
|
|
, successCallback :: String -- ^ name of the callback parameter when the request was successful
|
|
|
|
, errorCallback :: String -- ^ name of the callback parameter when the request reported an error
|
|
|
|
, moduleName :: String -- ^ namespace on which we define the foreign function (empty mean local var)
|
|
|
|
, urlPrefix :: String -- ^ a prefix we should add to the Url in the codegen
|
|
|
|
}
|
|
|
|
|
|
|
|
-- | Default options.
|
|
|
|
--
|
|
|
|
-- @
|
|
|
|
-- > defCommonGeneratorOptions = CommonGeneratorOptions
|
|
|
|
-- > { functionNameBuilder = camelCase
|
|
|
|
-- > , requestBody = "body"
|
|
|
|
-- > , successCallback = "onSuccess"
|
|
|
|
-- > , errorCallback = "onError"
|
|
|
|
-- > , moduleName = ""
|
|
|
|
-- > , urlPrefix = ""
|
|
|
|
-- > }
|
|
|
|
-- @
|
|
|
|
defCommonGeneratorOptions :: CommonGeneratorOptions
|
|
|
|
defCommonGeneratorOptions = CommonGeneratorOptions
|
|
|
|
{
|
|
|
|
functionNameBuilder = camelCase
|
|
|
|
, requestBody = "body"
|
|
|
|
, successCallback = "onSuccess"
|
|
|
|
, errorCallback = "onError"
|
|
|
|
, moduleName = ""
|
|
|
|
, urlPrefix = ""
|
|
|
|
}
|
|
|
|
|
2015-09-22 12:17:43 +02:00
|
|
|
-- | Attempts to reduce the function name provided to that allowed by @'Foreign'@.
|
|
|
|
--
|
|
|
|
-- https://mathiasbynens.be/notes/javascript-identifiers
|
|
|
|
-- Couldn't work out how to handle zero-width characters.
|
|
|
|
--
|
|
|
|
-- @TODO: specify better default function name, or throw error?
|
|
|
|
toValidFunctionName :: String -> String
|
|
|
|
toValidFunctionName (x:xs) = [setFirstChar x] <> filter remainder xs
|
|
|
|
where
|
|
|
|
setFirstChar c = if firstChar c then c else '_'
|
|
|
|
firstChar c = prefixOK c || any (Set.member c) firstLetterOK
|
|
|
|
remainder c = prefixOK c || any (Set.member c) remainderOK
|
|
|
|
-- Valid prefixes
|
|
|
|
prefixOK c = c `elem` ['$','_']
|
|
|
|
-- Unicode character sets
|
|
|
|
firstLetterOK = [ Set.lowercaseLetter
|
|
|
|
, Set.uppercaseLetter
|
|
|
|
, Set.titlecaseLetter
|
|
|
|
, Set.modifierLetter
|
|
|
|
, Set.otherLetter
|
|
|
|
, Set.letterNumber ]
|
|
|
|
remainderOK = firstLetterOK
|
|
|
|
<> [ Set.nonSpacingMark
|
|
|
|
, Set.spacingCombiningMark
|
|
|
|
, Set.decimalNumber
|
|
|
|
, Set.connectorPunctuation ]
|
|
|
|
toValidFunctionName [] = "_"
|
|
|
|
|
|
|
|
toJSHeader :: HeaderArg -> String
|
|
|
|
toJSHeader (HeaderArg n) = toValidFunctionName ("header" <> n)
|
|
|
|
toJSHeader (ReplaceHeaderArg n p)
|
|
|
|
| pn `isPrefixOf` p = pv <> " + \"" <> rp <> "\""
|
|
|
|
| pn `isSuffixOf` p = "\"" <> rp <> "\" + " <> pv
|
|
|
|
| pn `isInfixOf` p = "\"" <> (replace pn ("\" + " <> pv <> " + \"") p)
|
|
|
|
<> "\""
|
|
|
|
| otherwise = p
|
|
|
|
where
|
|
|
|
pv = toValidFunctionName ("header" <> n)
|
|
|
|
pn = "{" <> n <> "}"
|
|
|
|
rp = replace pn "" p
|
|
|
|
-- Use replace method from Data.Text
|
|
|
|
replace old new = T.unpack
|
|
|
|
. T.replace (T.pack old) (T.pack new)
|
|
|
|
. T.pack
|
|
|
|
|
2015-01-03 18:52:18 +01:00
|
|
|
jsSegments :: [Segment] -> String
|
|
|
|
jsSegments [] = ""
|
|
|
|
jsSegments [x] = "/" ++ segmentToStr x False
|
|
|
|
jsSegments (x:xs) = "/" ++ segmentToStr x True ++ jsSegments xs
|
|
|
|
|
|
|
|
segmentToStr :: Segment -> Bool -> String
|
|
|
|
segmentToStr (Segment st ms) notTheEnd =
|
|
|
|
segmentTypeToStr st ++ jsMParams ms ++ if notTheEnd then "" else "'"
|
|
|
|
|
|
|
|
segmentTypeToStr :: SegmentType -> String
|
|
|
|
segmentTypeToStr (Static s) = s
|
|
|
|
segmentTypeToStr (Cap s) = "' + encodeURIComponent(" ++ s ++ ") + '"
|
|
|
|
|
|
|
|
jsGParams :: String -> [QueryArg] -> String
|
2015-09-21 12:31:00 +02:00
|
|
|
jsGParams _ [] = ""
|
|
|
|
jsGParams _ [x] = paramToStr x False
|
2015-01-03 18:52:18 +01:00
|
|
|
jsGParams s (x:xs) = paramToStr x True ++ s ++ jsGParams s xs
|
|
|
|
|
2014-11-25 01:36:34 +01:00
|
|
|
jsParams :: [QueryArg] -> String
|
2015-01-03 18:52:18 +01:00
|
|
|
jsParams = jsGParams "&"
|
|
|
|
|
|
|
|
jsMParams :: [MatrixArg] -> String
|
|
|
|
jsMParams [] = ""
|
|
|
|
jsMParams xs = ";" ++ jsGParams ";" xs
|
2014-11-25 01:36:34 +01:00
|
|
|
|
|
|
|
paramToStr :: QueryArg -> Bool -> String
|
|
|
|
paramToStr qarg notTheEnd =
|
|
|
|
case qarg ^. argType of
|
|
|
|
Normal -> name
|
|
|
|
++ "=' + encodeURIComponent("
|
|
|
|
++ name
|
|
|
|
++ if notTheEnd then ") + '" else ")"
|
|
|
|
Flag -> name ++ "="
|
|
|
|
List -> name
|
|
|
|
++ "[]=' + encodeURIComponent("
|
|
|
|
++ name
|
|
|
|
++ if notTheEnd then ") + '" else ")"
|
|
|
|
where name = qarg ^. argName
|