2020-08-13 23:05:39 +02:00
|
|
|
{-# LANGUAGE DeriveAnyClass #-}
|
2020-08-13 23:22:39 +02:00
|
|
|
{-# LANGUAGE DataKinds #-}
|
|
|
|
{-# LANGUAGE TypeOperators #-}
|
2020-08-13 21:53:11 +02:00
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
module Main where
|
|
|
|
--------------------------------------------------------------------------------
|
2020-08-13 23:22:39 +02:00
|
|
|
import RIO hiding (Handler)
|
2020-08-13 21:53:11 +02:00
|
|
|
import RIO.Text
|
2020-08-13 23:05:39 +02:00
|
|
|
import RIO.Time
|
2020-08-13 23:22:39 +02:00
|
|
|
import Servant
|
2020-08-13 23:05:39 +02:00
|
|
|
import Data.Time.Clock.POSIX
|
2020-08-13 23:22:39 +02:00
|
|
|
import Prelude (read)
|
2020-08-13 21:53:11 +02:00
|
|
|
import Text.ParserCombinators.ReadP
|
2020-08-13 23:22:39 +02:00
|
|
|
|
|
|
|
import qualified Network.Wai.Handler.Warp as Warp
|
2020-08-13 21:53:11 +02:00
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
2020-08-13 23:22:39 +02:00
|
|
|
type Api = "run"
|
|
|
|
:> QueryParam' '[Required] "offset" Text
|
|
|
|
:> Get '[JSON] UTCTime
|
|
|
|
|
|
|
|
server :: Server Api
|
|
|
|
server = compute
|
|
|
|
where
|
|
|
|
compute :: Text -> Handler UTCTime
|
|
|
|
compute x = do
|
|
|
|
case parseInput x of
|
|
|
|
Nothing -> throwError err401
|
|
|
|
Just req -> do
|
|
|
|
res <- liftIO $ shiftTime req
|
|
|
|
pure res
|
|
|
|
|
2020-08-13 21:53:11 +02:00
|
|
|
data ShiftTimeRequest = ShiftTimeRequest
|
|
|
|
{ shiftSeconds :: Int
|
|
|
|
, shiftMinutes :: Int
|
|
|
|
, shiftHours :: Int
|
|
|
|
, shiftDays :: Int
|
|
|
|
, shiftWeeks :: Int
|
|
|
|
, shiftMonths :: Int
|
|
|
|
, shiftQuarters :: Int
|
|
|
|
, shiftYears :: Int
|
|
|
|
} deriving (Eq, Show)
|
|
|
|
|
2020-08-13 23:05:39 +02:00
|
|
|
instance Semigroup ShiftTimeRequest where
|
|
|
|
(ShiftTimeRequest as am ah ad aw amonths aq ay) <> (ShiftTimeRequest bs bm bh bd bw bmonths bq by) =
|
|
|
|
ShiftTimeRequest
|
|
|
|
{ shiftSeconds = as + bs
|
|
|
|
, shiftMinutes = am + bm
|
|
|
|
, shiftHours = ah + bh
|
|
|
|
, shiftDays = ad + bd
|
|
|
|
, shiftWeeks = aw + bw
|
|
|
|
, shiftMonths = amonths + bmonths
|
|
|
|
, shiftQuarters = aq + bq
|
|
|
|
, shiftYears = ay + by
|
|
|
|
}
|
|
|
|
|
|
|
|
instance Monoid ShiftTimeRequest where
|
|
|
|
mempty = defaultShiftTimeRequest
|
|
|
|
|
2020-08-13 21:53:11 +02:00
|
|
|
defaultShiftTimeRequest :: ShiftTimeRequest
|
|
|
|
defaultShiftTimeRequest = ShiftTimeRequest
|
|
|
|
{ shiftSeconds = 0
|
|
|
|
, shiftMinutes = 0
|
|
|
|
, shiftHours = 0
|
|
|
|
, shiftDays = 0
|
|
|
|
, shiftWeeks = 0
|
|
|
|
, shiftMonths = 0
|
|
|
|
, shiftQuarters = 0
|
|
|
|
, shiftYears = 0
|
|
|
|
}
|
|
|
|
|
2020-08-13 23:05:39 +02:00
|
|
|
-- This basically broken because it doesn't account for:
|
|
|
|
-- Exhales... time stuff
|
|
|
|
-- - Leap seconds, leap days, leap years...
|
|
|
|
-- - Months like February having 28 days and others having 31
|
|
|
|
-- - other things that I'm probably not considering
|
|
|
|
toSeconds :: ShiftTimeRequest -> NominalDiffTime
|
|
|
|
toSeconds ShiftTimeRequest{..} = do
|
|
|
|
let minutes = 60
|
|
|
|
hours = minutes * 60
|
|
|
|
days = hours * 24
|
|
|
|
weeks = days * 7
|
|
|
|
months = weeks * 4
|
|
|
|
quarters = months * 3
|
|
|
|
years = days * 365
|
|
|
|
fromIntegral $ shiftSeconds +
|
|
|
|
shiftMinutes * minutes +
|
|
|
|
shiftHours * hours +
|
|
|
|
shiftDays * days +
|
|
|
|
shiftWeeks * weeks +
|
|
|
|
shiftMonths * months +
|
|
|
|
shiftQuarters * quarters +
|
|
|
|
shiftYears * years
|
|
|
|
|
|
|
|
shiftTime :: ShiftTimeRequest -> IO UTCTime
|
|
|
|
shiftTime req = do
|
|
|
|
t <- getPOSIXTime
|
|
|
|
let t' = t + toSeconds req
|
|
|
|
pure $ posixSecondsToUTCTime t'
|
2020-08-13 21:53:11 +02:00
|
|
|
|
|
|
|
data Unit = Second
|
2020-08-13 23:05:39 +02:00
|
|
|
| Minute
|
|
|
|
| Hour
|
|
|
|
| Day
|
|
|
|
| Week
|
|
|
|
| Month
|
|
|
|
| Quarter
|
|
|
|
| Year
|
|
|
|
deriving (Eq, Show)
|
2020-08-13 21:53:11 +02:00
|
|
|
|
|
|
|
digit :: ReadP Char
|
|
|
|
digit =
|
|
|
|
satisfy (\c -> c >= '0' && c <= '9')
|
|
|
|
|
|
|
|
unit :: ReadP Unit
|
|
|
|
unit = do
|
2020-08-13 23:05:39 +02:00
|
|
|
c <- get
|
|
|
|
case c of
|
|
|
|
's' -> pure Second
|
|
|
|
'm' -> pure Minute
|
|
|
|
'h' -> pure Hour
|
|
|
|
'd' -> pure Day
|
|
|
|
'w' -> pure Week
|
|
|
|
'M' -> pure Month
|
|
|
|
'q' -> pure Quarter
|
|
|
|
'y' -> pure Year
|
|
|
|
_ -> fail $ "We don't support this unit: " ++ show c
|
2020-08-13 21:53:11 +02:00
|
|
|
|
|
|
|
request :: ReadP ShiftTimeRequest
|
|
|
|
request = do
|
|
|
|
negative <- option Nothing $ fmap Just (satisfy (== '-'))
|
|
|
|
n <- read <$> many1 digit
|
2020-08-13 23:05:39 +02:00
|
|
|
u <- unit
|
|
|
|
let amt = if isJust negative then -1 * n else n
|
|
|
|
case u of
|
|
|
|
Second -> pure $ defaultShiftTimeRequest { shiftSeconds = amt }
|
|
|
|
Minute -> pure $ defaultShiftTimeRequest { shiftMinutes = amt }
|
|
|
|
Hour -> pure $ defaultShiftTimeRequest { shiftHours = amt }
|
|
|
|
Day -> pure $ defaultShiftTimeRequest { shiftDays = amt }
|
|
|
|
Week -> pure $ defaultShiftTimeRequest { shiftWeeks = amt }
|
|
|
|
Month -> pure $ defaultShiftTimeRequest { shiftMonths = amt }
|
|
|
|
Quarter -> pure $ defaultShiftTimeRequest { shiftQuarters = amt }
|
|
|
|
Year -> pure $ defaultShiftTimeRequest { shiftYears = amt }
|
2020-08-13 21:53:11 +02:00
|
|
|
|
2020-08-13 23:05:39 +02:00
|
|
|
parseInput :: Text -> Maybe ShiftTimeRequest
|
|
|
|
parseInput x =
|
|
|
|
case readP_to_S (manyTill request eof) (unpack x) of
|
|
|
|
[(xs, "")] -> Just $ mconcat xs
|
2020-08-13 21:53:11 +02:00
|
|
|
_ -> Nothing
|
|
|
|
|
|
|
|
main :: IO ()
|
2020-08-13 23:22:39 +02:00
|
|
|
main = Warp.run 8000 $ serve (Proxy @ Api) server
|