rewrite in Servant

This commit is contained in:
Michele Guerini Rocco 2019-11-06 00:06:50 +01:00
parent f6f9ea1a17
commit 12fc50b89e
Signed by: rnhmjoj
GPG Key ID: BFBAF4C975F76450
4 changed files with 180 additions and 103 deletions

View File

@ -1,5 +1,5 @@
name: breve name: breve
version: 0.4.5.1 version: 0.5.0.0
synopsis: a url shortener synopsis: a url shortener
description: description:
@ -30,9 +30,10 @@ executable breve
Breve.Generator, Breve.UrlTable, Breve.Generator, Breve.UrlTable,
Paths_breve Paths_breve
other-extensions: OverloadedStrings other-extensions: OverloadedStrings
build-depends: base >=4.8 && <5.0, warp, warp-tls, tls, build-depends: base >=4.8 && <5.0,
Spock, Spock-core, blaze-html, http-types, warp, warp-tls, tls, blaze-html, blaze-markup,
wai, wai-middleware-static, wai-extra, servant-server, servant, servant-blaze, http-api-data,
wai, wai-extra, streaming-commons,
transformers, mtl, transformers, mtl,
text, aeson, bytestring, binary, text, aeson, bytestring, binary,
hashtables, cryptohash, random, hashtables, cryptohash, random,

View File

@ -1,79 +1,144 @@
{-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE DataKinds #-}
{-# LANGUAGE KindSignatures #-}
{-# LANGUAGE TypeOperators #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE TypeSynonymInstances #-}
module Application where module Application where
-- Breve modules
import Breve.Generator import Breve.Generator
import Breve.UrlTable import Breve.UrlTable
import Paths_breve (getDataFileName)
import Views import Views
-- Misc
import Data.Monoid import Data.Monoid
import Control.Monad.IO.Class (liftIO) import Control.Monad.IO.Class (liftIO)
import Data.Aeson hiding (json)
import Data.Text (Text)
import qualified Data.Text.IO as T import qualified Data.Text.IO as T
import Web.Spock.Core -- JSON conversion
import Network.HTTP.Types.Status import Data.Text (Text)
import Network.Wai (Middleware) import Data.Aeson (ToJSON)
import Network.Wai.Middleware.Static import GHC.Generics (Generic)
import Network.Wai.Middleware.RequestLogger
-- HTML replies
import Text.Blaze.Html5 (Html)
import Text.Blaze.Renderer.Utf8 (renderMarkup)
-- API definition
import Servant
import Servant.HTML.Blaze (HTML)
import Web.FormUrlEncoded (FromForm(..), parseUnique)
import GHC.TypeNats (Nat)
serveStatic :: FilePath -> Middleware -- * Types
serveStatic = staticPolicy . addBase
-- | API successful reply
-- This is the reply returned by the API
-- handler when the url has been shortned
-- successfully.
data ApiReply = ApiReply
{ link :: Url -- ^ shortened url
, name :: Name -- ^ just the name
, original :: Url -- ^ original url
} deriving Generic
instance ToJSON ApiReply
instance FromForm Url where
fromForm f = parseUnique "url" f
reply :: Status -> Text -> ActionT IO () -- * Breve API
reply code text = setStatus code >> render (message text)
-- | API spec
--
-- Breve has three main components:
-- 1. the web app
-- 2. the JSON API
-- 3. the static files server
type Breve = API :<|> App
-- | Web app spec
type App =
Get '[HTML] Html
:<|> "static" :> Raw
:<|> Capture "name" Name :> Redirect
:<|> ReqBody '[FormUrlEncoded] Url :> Post '[HTML] Html
-- | JSON API spec
type API =
"api" :> ReqBody '[FormUrlEncoded] Url :> Post '[JSON] ApiReply
-- | Verb that encodes an HTTP 302 redirection
type Redirect =
Verb 'GET 302 '[PlainText] Redirection
-- | Reply with Location redirect header
type Redirection =
Headers '[Header "Location" Text] NoContent
logStr :: Text -> ActionT IO () -- | Breve application
--
-- Breve takes as parameters the bind url and the urls table
breve :: FilePath -> Url -> UrlTable -> Application
breve static url table = serve (Proxy :: Proxy Breve) (breveServer static url table)
-- * Handlers
-- | Breve server
--
-- This is just an ordered collection of handlers
-- following the 'Breve' API spec.
breveServer :: FilePath -> Url -> UrlTable -> Server Breve
breveServer static url table =
api url table :<|> app
where app = homepage :<|>
serveDirectoryWebApp static :<|>
resolver table :<|>
uploader url table
-- | Serves the homepage
homepage :: Handler Html
homepage = pure index
-- | Resolves a 'Name' to the full 'Url'
resolver :: UrlTable -> Name -> Handler Redirection
resolver table name = do
url <- liftIO (extract table name)
case url of
Nothing ->
throwError $ err404 { errBody = renderMarkup (message "404: not found") }
Just url -> do
logStr ("Resolved " <> name <> " -> " <> url)
pure (addHeader url NoContent)
-- | Takes a 'Url' via POST
-- and prints the shortned one
uploader :: Url -> UrlTable -> Url -> Handler Html
uploader bindUrl table url = do
name <- liftIO (insert table url)
logStr ("Registered " <> url <> " -> " <> name)
pure (done $ bindUrl <> name)
-- Takes a 'Url' via POST and returns
-- the shortned one in an 'ApiReply' as JSON.
api :: Url -> UrlTable -> Url -> Handler ApiReply
api bindUrl table url = do
name <- liftIO (insert table url)
logStr ("Registered " <> url <> " -> " <> name)
pure $ ApiReply { link = (bindUrl <> name)
, name = name
, original = url
}
-- * Misc
-- | Handy function to log to stdout
logStr :: Text -> Handler ()
logStr = liftIO . T.putStrLn logStr = liftIO . T.putStrLn
app :: Url -> UrlTable -> SpockT IO ()
app url' table = do
static <- liftIO (getDataFileName "static/")
middleware (serveStatic static)
middleware logStdout
get "/" $ render index
get var $ \name -> do
url <- liftIO (extract table name)
case url of
Nothing -> reply status404 "404: does not exist"
Just url -> do
logStr ("Resolved " <> name <> " -> " <> url)
redirect url
post "/" $ do
url <- param "url"
case url of
Nothing -> reply status400 "400: bad request"
Just url -> do
name <- liftIO (insert table url)
logStr ("Registered " <> url <> " -> " <> name)
render (done $ url' <> name)
post "api" $ do
url <- param "url"
case url of
Nothing -> do
setStatus status400
json $ object [ "error" .= ("bad request" :: Text )
, "msg" .= ("missing url field" :: Text ) ]
Just url -> do
name <- liftIO (insert table url)
logStr ("Registered " <> url <> " -> " <> name)
json $ object [ "link" .= (url' <> name)
, "name" .= name
, "original" .= url ]
toTLS :: Text -> SpockT IO ()
toTLS host = do
get var (redirect . new)
get "/" (redirect $ new "")
where new url = "https://" <> host <> "/" <> url

View File

@ -1,40 +1,57 @@
{-# LANGUAGE OverloadedStrings, RecordWildCards #-} {-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE RecordWildCards #-}
-- Breve modules
import Application import Application
import Breve.Settings import Breve.Settings
import Breve.UrlTable import Breve.UrlTable
import Paths_breve (getDataFileName)
import Data.Text (Text, unpack) -- Data conversions
import Data.Maybe (listToMaybe) import Data.Text (Text, unpack)
import Control.Concurrent (forkIO) import Data.String (IsString(..))
import Control.Monad import Data.Maybe (listToMaybe)
-- IO
import System.Environment (getArgs) import System.Environment (getArgs)
import Data.Text.IO as T
import Web.Spock.Core -- Web server
import Network.Wai.Handler.WarpTLS (runTLS, TLSSettings) import Servant (Application)
import Network.Wai.Handler.Warp (run, defaultSettings, setPort) import Network.Wai.Handler.Warp (defaultSettings, setPort, setHost)
import Network.Wai.Handler.WarpTLS (runTLS, TLSSettings)
runBreve :: TLSSettings -> Int -> SpockT IO () -> IO () -- Middlewares
runBreve tlsSettings port spock = import Network.Wai.Middleware.ForceSSL (forceSSL)
spockAsApp (spockT id spock) >>= runTLS tlsSettings settings import Network.Wai.Middleware.RequestLogger (logStdout)
where settings = setPort port defaultSettings
runTLSRedirect :: Text -> IO () -- * Helpers
runTLSRedirect = spockAsApp . spockT id . toTLS >=> run 80
forkIO' :: IO () -> IO ()
forkIO' = fmap (const ()) . forkIO
-- | Runs Breve on the Warp webserver
run :: AppSettings -> Application -> IO ()
run (AppSettings{..}) =
runTLS tlsSettings warpSettings
where
host = unpack bindHost
warpSettings = setPort bindPort $
setHost (fromString host) defaultSettings
-- | Main
--
-- Reads the config (given as the unique argument)
-- and runs the breve web app
main :: IO () main :: IO ()
main = do main = do
configPath <- fmap listToMaybe getArgs configPath <- fmap listToMaybe getArgs
AppSettings {..} <- settings configPath config@(AppSettings{..}) <- settings configPath
table <- load urlTable table <- load urlTable
static <- getDataFileName "static/"
when (bindPort == 443) (forkIO' $ runTLSRedirect bindHost) let
middlewares =
logStdout .
(if bindPort == 433 then forceSSL else id)
putStrLn ("Serving on " ++ unpack bindUrl) T.putStrLn ("Serving on " <> bindUrl)
runBreve tlsSettings bindPort (app bindUrl table) run config (middlewares $ breve static bindUrl table)

View File

@ -2,15 +2,9 @@
module Views where module Views where
import Data.Text (Text) import Data.Text (Text)
import Data.Text.Lazy (toStrict) import Text.Blaze.Html5 as H
import Text.Blaze.Html.Renderer.Text (renderHtml) import Text.Blaze.Html5.Attributes as A
import Text.Blaze.Html5 as H
import Text.Blaze.Html5.Attributes as A
import qualified Web.Spock.Core as S
render :: Html -> S.ActionT IO ()
render = S.html . toStrict . renderHtml
done :: Text -> Html done :: Text -> Html
done url = template $ do done url = template $ do
@ -36,10 +30,10 @@ template fill =
meta ! name "keywords" ! content "url, shortener" meta ! name "keywords" ! content "url, shortener"
meta ! name "author" ! content "Michele Guerini Rocco" meta ! name "author" ! content "Michele Guerini Rocco"
meta ! charset "utf-8" meta ! charset "utf-8"
link ! rel "stylesheet" ! href "main.css" ! type_ "text/css" link ! rel "stylesheet" ! href "/static/main.css" ! type_ "text/css"
link ! rel "apple-touch-icon" ! href "icon-big.png" link ! rel "apple-touch-icon" ! href "static/icon-big.png"
link ! rel "icon" ! type_ "image/png" ! href "/icon-medium.png" ! sizes "96x96" link ! rel "icon" ! type_ "image/png" ! href "/static/icon-medium.png" ! sizes "96x96"
link ! rel "icon" ! type_ "image/png" ! href "/icon-small.png" ! sizes "16x16" link ! rel "icon" ! type_ "image/png" ! href "/static/icon-small.png" ! sizes "16x16"
body $ do body $ do
header $ do header $ do
h1 $ a ! href "/" $ "BREVE" h1 $ a ! href "/" $ "BREVE"