rewrite in Servant
This commit is contained in:
parent
f6f9ea1a17
commit
12fc50b89e
@ -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,
|
||||||
|
@ -1,79 +1,144 @@
|
|||||||
|
{-# LANGUAGE DataKinds #-}
|
||||||
|
{-# LANGUAGE KindSignatures #-}
|
||||||
|
{-# LANGUAGE TypeOperators #-}
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# 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
|
||||||
logStr = liftIO . T.putStrLn
|
--
|
||||||
|
-- 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)
|
||||||
|
|
||||||
|
|
||||||
app :: Url -> UrlTable -> SpockT IO ()
|
-- * Handlers
|
||||||
app url' table = do
|
|
||||||
static <- liftIO (getDataFileName "static/")
|
|
||||||
|
|
||||||
middleware (serveStatic static)
|
-- | Breve server
|
||||||
middleware logStdout
|
--
|
||||||
|
-- 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
|
||||||
|
|
||||||
get "/" $ render index
|
-- | Serves the homepage
|
||||||
|
homepage :: Handler Html
|
||||||
|
homepage = pure index
|
||||||
|
|
||||||
get var $ \name -> do
|
-- | Resolves a 'Name' to the full 'Url'
|
||||||
|
resolver :: UrlTable -> Name -> Handler Redirection
|
||||||
|
resolver table name = do
|
||||||
url <- liftIO (extract table name)
|
url <- liftIO (extract table name)
|
||||||
case url of
|
case url of
|
||||||
Nothing -> reply status404 "404: does not exist"
|
Nothing ->
|
||||||
|
throwError $ err404 { errBody = renderMarkup (message "404: not found") }
|
||||||
Just url -> do
|
Just url -> do
|
||||||
logStr ("Resolved " <> name <> " -> " <> url)
|
logStr ("Resolved " <> name <> " -> " <> url)
|
||||||
redirect url
|
pure (addHeader url NoContent)
|
||||||
|
|
||||||
post "/" $ do
|
|
||||||
url <- param "url"
|
-- | Takes a 'Url' via POST
|
||||||
case url of
|
-- and prints the shortned one
|
||||||
Nothing -> reply status400 "400: bad request"
|
uploader :: Url -> UrlTable -> Url -> Handler Html
|
||||||
Just url -> do
|
uploader bindUrl table url = do
|
||||||
name <- liftIO (insert table url)
|
name <- liftIO (insert table url)
|
||||||
logStr ("Registered " <> url <> " -> " <> name)
|
logStr ("Registered " <> url <> " -> " <> name)
|
||||||
render (done $ url' <> name)
|
pure (done $ bindUrl <> name)
|
||||||
|
|
||||||
post "api" $ do
|
-- Takes a 'Url' via POST and returns
|
||||||
url <- param "url"
|
-- the shortned one in an 'ApiReply' as JSON.
|
||||||
case url of
|
api :: Url -> UrlTable -> Url -> Handler ApiReply
|
||||||
Nothing -> do
|
api bindUrl table url = do
|
||||||
setStatus status400
|
|
||||||
json $ object [ "error" .= ("bad request" :: Text )
|
|
||||||
, "msg" .= ("missing url field" :: Text ) ]
|
|
||||||
Just url -> do
|
|
||||||
name <- liftIO (insert table url)
|
name <- liftIO (insert table url)
|
||||||
logStr ("Registered " <> url <> " -> " <> name)
|
logStr ("Registered " <> url <> " -> " <> name)
|
||||||
json $ object [ "link" .= (url' <> name)
|
pure $ ApiReply { link = (bindUrl <> name)
|
||||||
, "name" .= name
|
, name = name
|
||||||
, "original" .= url ]
|
, original = url
|
||||||
|
}
|
||||||
|
|
||||||
|
-- * Misc
|
||||||
|
|
||||||
toTLS :: Text -> SpockT IO ()
|
-- | Handy function to log to stdout
|
||||||
toTLS host = do
|
logStr :: Text -> Handler ()
|
||||||
get var (redirect . new)
|
logStr = liftIO . T.putStrLn
|
||||||
get "/" (redirect $ new "")
|
|
||||||
where new url = "https://" <> host <> "/" <> url
|
|
||||||
|
55
src/Main.hs
55
src/Main.hs
@ -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)
|
||||||
|
|
||||||
|
-- Data conversions
|
||||||
import Data.Text (Text, unpack)
|
import Data.Text (Text, unpack)
|
||||||
|
import Data.String (IsString(..))
|
||||||
import Data.Maybe (listToMaybe)
|
import Data.Maybe (listToMaybe)
|
||||||
import Control.Concurrent (forkIO)
|
|
||||||
import Control.Monad
|
-- IO
|
||||||
import System.Environment (getArgs)
|
import System.Environment (getArgs)
|
||||||
|
import Data.Text.IO as T
|
||||||
|
|
||||||
import Web.Spock.Core
|
-- Web server
|
||||||
|
import Servant (Application)
|
||||||
|
import Network.Wai.Handler.Warp (defaultSettings, setPort, setHost)
|
||||||
import Network.Wai.Handler.WarpTLS (runTLS, TLSSettings)
|
import Network.Wai.Handler.WarpTLS (runTLS, TLSSettings)
|
||||||
import Network.Wai.Handler.Warp (run, defaultSettings, setPort)
|
|
||||||
|
|
||||||
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)
|
||||||
|
14
src/Views.hs
14
src/Views.hs
@ -3,14 +3,8 @@
|
|||||||
module Views where
|
module Views where
|
||||||
|
|
||||||
import Data.Text (Text)
|
import Data.Text (Text)
|
||||||
import Data.Text.Lazy (toStrict)
|
|
||||||
import Text.Blaze.Html.Renderer.Text (renderHtml)
|
|
||||||
import Text.Blaze.Html5 as H
|
import Text.Blaze.Html5 as H
|
||||||
import Text.Blaze.Html5.Attributes as A
|
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"
|
||||||
|
Loading…
Reference in New Issue
Block a user