Use Data.Text everywhere

This commit is contained in:
rnhmjoj 2015-08-11 04:02:10 +02:00
parent 2f8adbefe9
commit 0401ccd128
6 changed files with 57 additions and 46 deletions

View File

@ -33,7 +33,7 @@ executable breve
Spock, blaze-html, http-types, Spock, blaze-html, http-types,
wai, wai-middleware-static, wai-extra, wai, wai-middleware-static, wai-extra,
transformers, mtl, transformers, mtl,
text, aeson, bytestring, binary, text, text-format, aeson, bytestring, binary,
hashtables, cryptohash, random, hashtables, cryptohash, random,
xdg-basedir, configurator, directory xdg-basedir, configurator, directory
ghc-options: -threaded -O2 ghc-options: -threaded -O2

View File

@ -6,13 +6,11 @@ import Breve.UrlTable
import Paths_breve (getDataFileName) import Paths_breve (getDataFileName)
import Views import Views
import Control.Monad.IO.Class (liftIO)
import Text.Printf (printf)
import Data.Aeson hiding (json)
import Data.Monoid import Data.Monoid
import Data.Text (pack, unpack) import Control.Monad.IO.Class (liftIO)
import Data.Text.Lazy.Encoding (decodeUtf8) import Data.Aeson hiding (json)
import Data.Text.Lazy (toStrict) import Data.Text (Text)
import qualified Data.Text.Format as F
import Web.Spock.Safe import Web.Spock.Safe
import Network.HTTP.Types.Status import Network.HTTP.Types.Status
@ -20,15 +18,12 @@ import Network.Wai (Middleware)
import Network.Wai.Middleware.Static import Network.Wai.Middleware.Static
import Network.Wai.Middleware.RequestLogger import Network.Wai.Middleware.RequestLogger
logStr :: String -> ActionT IO ()
logStr = liftIO . putStrLn
serveStatic :: FilePath -> Middleware serveStatic :: FilePath -> Middleware
serveStatic = staticPolicy . addBase serveStatic = staticPolicy . addBase
reply :: Status -> String -> ActionT IO () reply :: Status -> Text -> ActionT IO ()
reply code text = setStatus code >> render (message text) reply code text = setStatus code >> render (message text)
@ -46,29 +41,29 @@ app url' table = do
case url of case url of
Nothing -> reply status404 "404: does not exist" Nothing -> reply status404 "404: does not exist"
Just url -> do Just url -> do
logStr (printf "Resolved %s -> %s" name url) F.print "Resolved {} -> {} " (name, url)
redirect (pack url) redirect url
post "/" $ do post "/" $ do
url <- param "url" url <- param "url"
case unpack <$> url of case url of
Nothing -> reply status400 "400: bad request" Nothing -> reply status400 "400: bad request"
Just url -> do Just url -> do
name <- liftIO (insert table url) name <- liftIO (insert table url)
logStr (printf "Registered %s -> %s " url name) F.print "Registered {} -> {} " (url, name)
let link = url' <> name let link = url' <> name
render (done link) render (done link)
post "api" $ do post "api" $ do
url <- param "url" url <- param "url"
case unpack <$> url of case url of
Nothing -> do Nothing -> do
setStatus status400 setStatus status400
json $ object [ "error" .= pack "bad request" json $ object [ "error" .= ("bad request" :: Text )
, "msg" .= pack "missing url field" ] , "msg" .= ("missing url field" :: Text ) ]
Just url -> do Just url -> do
name <- liftIO (insert table url) name <- liftIO (insert table url)
logStr (printf "Registered %s -> %s " url name) F.print "Registered {} -> {} " (url, name)
json $ object [ "link" .= pack (url' <> name) json $ object [ "link" .= (url' <> name)
, "name" .= name , "name" .= name
, "original" .= url ] , "original" .= url ]

View File

@ -9,11 +9,12 @@ import Control.Monad.State
import System.Random import System.Random
import Crypto.Hash.SHA256 (hash) import Crypto.Hash.SHA256 (hash)
import Data.Binary (decode) import Data.Binary (decode)
import Data.ByteString.Char8 (pack)
import Data.ByteString.Lazy (fromStrict) import Data.ByteString.Lazy (fromStrict)
import Data.Text (Text, pack)
import Data.Text.Encoding (encodeUtf8)
type Name = String type Name = Text
type Url = String type Url = Text
-- Choose a random element of a list -- Choose a random element of a list
choice :: [a] -> State StdGen a choice :: [a] -> State StdGen a
@ -22,14 +23,14 @@ choice xs = (xs !!) <$> randomSt (0, length xs - 1)
-- Generate a random phonetic string -- Generate a random phonetic string
word :: State StdGen Name word :: State StdGen Name
word = replicateM 10 letter where word = pack <$> replicateM 10 letter where
vowels = "aeiou" vowels = "aeiou"
consonants = "bcdfghjklmnpqrstvwxyz" consonants = "bcdfghjklmnpqrstvwxyz"
letter = choice [vowels, consonants] >>= choice letter = choice [vowels, consonants] >>= choice
-- SHA256 hash to seed a generator -- SHA256 hash to seed a generator
intHash :: Url -> Int intHash :: Url -> Int
intHash = decode . fromStrict . hash . pack intHash = decode . fromStrict . hash . encodeUtf8
-- Assign a unique name to the url -- Assign a unique name to the url
nameHash :: Url -> Name nameHash :: Url -> Name

View File

@ -6,13 +6,16 @@ import System.Environment (lookupEnv)
import System.Environment.XDG.BaseDir import System.Environment.XDG.BaseDir
import System.Directory (doesFileExist) import System.Directory (doesFileExist)
import Data.Configurator import Data.Configurator
import Data.Monoid
import Data.Text (Text, pack)
import Network.Wai.Handler.WarpTLS (tlsSettings, TLSSettings) import Network.Wai.Handler.WarpTLS (tlsSettings, TLSSettings)
data AppSettings = AppSettings data AppSettings = AppSettings
{ bindPort :: Int { bindHost :: Text
, bindUrl :: String , bindPort :: Int
, urlTable :: FilePath , bindUrl :: Text
, tlsSetts :: TLSSettings , urlTable :: FilePath
, tlsSetts :: TLSSettings
} }
@ -36,14 +39,14 @@ settings = do
createEmptyIfMissing urls createEmptyIfMissing urls
let base = "https://" ++ host let base = "https://" <> host
url = if port == 443 url = if port == 443
then base then base
else base ++ ":" ++ show port else base <> ":" <> pack (show port)
return AppSettings return AppSettings
{ bindPort = port { bindPort = port
, bindUrl = url ++ "/" , bindUrl = url <> "/"
, urlTable = urls , urlTable = urls
, tlsSetts = tlsSettings cert key , tlsSetts = tlsSettings cert key
} }

View File

@ -1,25 +1,37 @@
{-# LANGUAGE OverloadedStrings, NamedFieldPuns #-} {-# LANGUAGE OverloadedStrings, RecordWildCards #-}
import Application import Application
import Breve.Settings import Breve.Settings
import Breve.UrlTable import Breve.UrlTable
import Data.Text (Text, unpack)
import Control.Concurrent (forkIO)
import Control.Monad
import Web.Spock.Safe import Web.Spock.Safe
import Network.Wai.Handler.WarpTLS (runTLS, TLSSettings) import Network.Wai.Handler.WarpTLS (runTLS, TLSSettings)
import Network.Wai.Handler.Warp (defaultSettings, setPort) import Network.Wai.Handler.Warp (run, defaultSettings, setPort)
runBreve :: TLSSettings -> Int -> SpockT IO () -> IO () runBreve :: TLSSettings -> Int -> SpockT IO () -> IO ()
runBreve tls port spock = runBreve tls port spock =
spockAsApp (spockT id spock) >>= runTLS tls settings spockAsApp (spockT id spock) >>= runTLS tls settings
where settings = setPort port defaultSettings where settings = setPort port defaultSettings
runTLSRedirect :: Text -> IO ()
runTLSRedirect = spockAsApp . spockT id . toTLS >=> run 80
forkIO' :: IO () -> IO ()
forkIO' = fmap (const ()) . forkIO
main :: IO () main :: IO ()
main = do main = do
AppSettings { bindUrl AppSettings {..} <- settings
, bindPort table <- load urlTable
, urlTable
, tlsSetts } <- settings
table <- load urlTable
putStrLn ("Serving on " ++ bindUrl)
runBreve tlsSetts bindPort (app bindUrl table)
putStrLn ("Serving on " ++ unpack bindUrl)
when (bindPort == 443) (forkIO' $ runTLSRedirect bindHost)
runBreve tlsSetts bindPort (app bindUrl table)

View File

@ -12,7 +12,7 @@ import qualified Web.Spock.Safe as S
render :: Html -> S.ActionT IO () render :: Html -> S.ActionT IO ()
render = S.html . toStrict . renderHtml render = S.html . toStrict . renderHtml
done :: String -> Html done :: Text -> Html
done url = template $ do done url = template $ do
"here's your new link: " "here's your new link: "
a ! href (toValue url) $ (toHtml url) a ! href (toValue url) $ (toHtml url)
@ -24,7 +24,7 @@ index = template $ do
input ! type_ "text" ! name "url" input ! type_ "text" ! name "url"
input ! type_ "submit" ! value "go" input ! type_ "submit" ! value "go"
message :: String -> Html message :: Text -> Html
message = template . toHtml message = template . toHtml
template :: Html -> Html template :: Html -> Html