Use Data.Text everywhere
This commit is contained in:
parent
2f8adbefe9
commit
0401ccd128
@ -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
|
||||||
|
@ -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 ]
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
module Breve.Generator
|
module Breve.Generator
|
||||||
( nameHash
|
( nameHash
|
||||||
, intHash
|
, intHash
|
||||||
, Name
|
, Name
|
||||||
@ -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
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -27,7 +30,7 @@ settings = do
|
|||||||
urlsPath <- getUserDataFile "breve" ""
|
urlsPath <- getUserDataFile "breve" ""
|
||||||
configPath <- getUserConfigFile "breve" ""
|
configPath <- getUserConfigFile "breve" ""
|
||||||
|
|
||||||
config <- load [Required configPath]
|
config <- load [Required configPath]
|
||||||
host <- lookupDefault "localhost" config "hostname"
|
host <- lookupDefault "localhost" config "hostname"
|
||||||
port <- lookupDefault 3000 config "port"
|
port <- lookupDefault 3000 config "port"
|
||||||
cert <- lookupDefault "/usr/share/tls/breve.crt" config "cert"
|
cert <- lookupDefault "/usr/share/tls/breve.crt" config "cert"
|
||||||
@ -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
|
||||||
}
|
}
|
||||||
|
32
src/Main.hs
32
src/Main.hs
@ -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
|
putStrLn ("Serving on " ++ unpack bindUrl)
|
||||||
table <- load urlTable
|
|
||||||
putStrLn ("Serving on " ++ bindUrl)
|
when (bindPort == 443) (forkIO' $ runTLSRedirect bindHost)
|
||||||
runBreve tlsSetts bindPort (app bindUrl table)
|
runBreve tlsSetts bindPort (app bindUrl table)
|
||||||
|
|
@ -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
|
||||||
|
Loading…
Reference in New Issue
Block a user