2022-06-06 09:40:23 +00:00
|
|
|
module Examples.JsonParsing.Main where
|
|
|
|
|
|
|
|
import Prelude
|
|
|
|
|
|
|
|
import Data.Either (Either)
|
|
|
|
import Data.Either as Either
|
|
|
|
import Data.Generic.Rep (class Generic)
|
|
|
|
import Data.Maybe (Maybe)
|
|
|
|
import Effect.Console (log)
|
|
|
|
import HTTPurple (JsonDecoder(..), JsonEncoder(..), ServerM, fromJson, notFound, serve, toJson, usingCont)
|
|
|
|
import HTTPurple as Json
|
|
|
|
import HTTPurple.Method (Method(..))
|
|
|
|
import HTTPurple.Response (ok')
|
|
|
|
import Routing.Duplex as RD
|
|
|
|
import Routing.Duplex.Generic as RG
|
|
|
|
|
|
|
|
data Route = SayHello
|
|
|
|
|
|
|
|
derive instance Generic Route _
|
|
|
|
|
|
|
|
route :: RD.RouteDuplex' Route
|
|
|
|
route = RD.root $ RG.sum
|
|
|
|
{ "SayHello": RG.noArgs
|
|
|
|
}
|
|
|
|
|
|
|
|
type HelloWorldRequest = { name :: String }
|
|
|
|
type HelloWorldResponse = { hello :: String }
|
|
|
|
|
2022-06-06 09:53:05 +00:00
|
|
|
-- the following test decoder/encoder code is just for testing. in your project you will want to use
|
|
|
|
-- jsonEncoder and jsonDecoder from httpurple-argonaut or httpurple-yoga-json
|
2022-06-06 09:40:23 +00:00
|
|
|
foreign import data Json :: Type
|
|
|
|
|
|
|
|
foreign import parseJson :: String -> Maybe Json
|
|
|
|
|
|
|
|
foreign import getName :: Json -> Maybe String
|
|
|
|
|
|
|
|
testDecoder :: JsonDecoder String HelloWorldRequest
|
|
|
|
testDecoder = JsonDecoder fromJsonString
|
|
|
|
where
|
|
|
|
fromJsonString :: String -> Either String HelloWorldRequest
|
|
|
|
fromJsonString = (parseJson >=> getName) >>> map { name: _ } >>> Either.note "Invalid json"
|
|
|
|
|
|
|
|
testEncoder :: JsonEncoder HelloWorldResponse
|
|
|
|
testEncoder = JsonEncoder $ \{ hello } -> "{\"hello\": \"" <> hello <> "\" }"
|
|
|
|
|
|
|
|
-- | Boot up the server
|
|
|
|
main :: ServerM
|
|
|
|
main =
|
2022-08-24 17:59:06 +00:00
|
|
|
serve { hostname: "localhost", port: 8080, onStarted } { route, router }
|
2022-06-06 09:40:23 +00:00
|
|
|
where
|
|
|
|
router { route: SayHello, method: Post, body } = usingCont do
|
2022-06-06 09:53:05 +00:00
|
|
|
-- in your project you will want to use Argonaut.jsonDecoder from httpurple-argonaut
|
|
|
|
-- or Yoga.jsonDecoder from httpurple-yoga-json here instead of the testDecoder
|
2022-06-06 09:40:23 +00:00
|
|
|
{ name } :: HelloWorldRequest <- fromJson testDecoder body
|
2022-06-06 09:53:05 +00:00
|
|
|
ok' Json.jsonHeaders $ toJson testEncoder $ { hello: name } -- same here for the encoder
|
2022-06-06 09:40:23 +00:00
|
|
|
router { route: SayHello } = notFound
|
|
|
|
|
|
|
|
onStarted = do
|
|
|
|
log " ┌────────────────────────────────────────────┐"
|
|
|
|
log " │ Server now up on port 8080 │"
|
|
|
|
log " │ │"
|
|
|
|
log " │ To test, run: │"
|
|
|
|
log " │ > http -v POST localhost:8080 hello=world │"
|
|
|
|
log " | # => { \"hello\": \"world\" } │"
|
|
|
|
log " └────────────────────────────────────────────┘"
|