summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--Utility/Misc.hs2
-rw-r--r--Utility/Scheduled.hs116
2 files changed, 117 insertions, 1 deletions
diff --git a/Utility/Misc.hs b/Utility/Misc.hs
index 804a9e487..a2c9c8184 100644
--- a/Utility/Misc.hs
+++ b/Utility/Misc.hs
@@ -29,7 +29,7 @@ hGetContentsStrict = hGetContents >=> \s -> length s `seq` return s
readFileStrict :: FilePath -> IO String
readFileStrict = readFile >=> \s -> length s `seq` return s
-{- Like break, but the character matching the condition is not included
+{- Like break, but the item matching the condition is not included
- in the second result list.
-
- separate (== ':') "foo:bar" = ("foo", "bar")
diff --git a/Utility/Scheduled.hs b/Utility/Scheduled.hs
new file mode 100644
index 000000000..90d346280
--- /dev/null
+++ b/Utility/Scheduled.hs
@@ -0,0 +1,116 @@
+{- scheduled activities
+ -
+ - Copyright 2013 Joey Hess <joey@kitenet.net>
+ -
+ - Licensed under the GNU GPL version 3 or higher.
+ -}
+
+module Utility.Scheduled (
+ Schedule(..),
+ Recurrance(..),
+ TimeOfDay(..),
+ fromSchedule,
+ toSchedule
+) where
+
+import Common
+
+{- Some sort of scheduled event. -}
+data Schedule = Schedule Recurrance TimeOfDay Duration
+ deriving (Show)
+
+data Recurrance
+ = Daily
+ | Weekly WeekDay
+ | Monthly MonthDay
+ | Yearly YearDay
+ -- Divisible 3 Daily is every day of the year evenly divisible by 3
+ | Divisable Int Recurrance
+ deriving (Show)
+
+type WeekDay = Int
+type MonthDay = Int
+type YearDay = Int
+
+data TimeOfDay
+ = AnyTime
+ | Hour Int
+ deriving (Show)
+
+data Duration = MinutesDuration Int
+ deriving (Show)
+
+fromRecurrance :: Recurrance -> String
+fromRecurrance (Divisable n r) =
+ fromRecurrance' (++ "s divisible by " ++ show n) r
+fromRecurrance r = fromRecurrance' ("every " ++) r
+
+fromRecurrance' :: (String -> String) -> Recurrance -> String
+fromRecurrance' a Daily = a "day"
+fromRecurrance' a (Weekly n) = onday n (a "week")
+fromRecurrance' a (Monthly n) = onday n (a "month")
+fromRecurrance' a (Yearly n) = onday n (a "year")
+fromRecurrance' a (Divisable _n r) = fromRecurrance' a r -- not used
+
+onday :: Int -> String -> String
+onday n s = s ++ " on day " ++ show n
+
+toRecurrance :: String -> Maybe Recurrance
+toRecurrance s = case words s of
+ ("every":something:l) -> parse something l
+ (something:"divisible":"by":sn:l) -> do
+ r <- parse something l
+ n <- readish sn
+ if n > 0
+ then Just $ Divisable n r
+ else Nothing
+ _ -> Nothing
+ where
+ parse "day" [] = Just Daily
+ parse "week" l = withday Weekly l
+ parse "month" l = withday Monthly l
+ parse "year" l = withday Yearly l
+ parse v l
+ | "s" `isSuffixOf` v = parse (reverse $ drop 1 $ reverse v) l
+ | otherwise = Nothing
+ withday a ("on":"day":n:[]) = a <$> readish n
+ withday _ _ = Nothing
+
+fromTimeOfDay :: TimeOfDay -> String
+fromTimeOfDay AnyTime = "any time"
+fromTimeOfDay (Hour n) = "hour " ++ show n
+
+toTimeOfDay :: String -> Maybe TimeOfDay
+toTimeOfDay s = case words s of
+ ("any":"time":[]) -> Just AnyTime
+ ("hour":n:[]) -> Hour <$> readish n
+ _ -> Nothing
+
+fromDuration :: Duration -> String
+fromDuration (MinutesDuration n) = show n ++ " minutes"
+
+toDuration :: String -> Maybe Duration
+toDuration s = case words s of
+ (n:"minutes":[]) -> MinutesDuration <$> readish n
+ (n:"minute":[]) -> MinutesDuration <$> readish n
+ _ -> Nothing
+
+fromSchedule :: Schedule -> String
+fromSchedule (Schedule recurrance timeofday duration) = unwords
+ [ fromRecurrance recurrance
+ , "at"
+ , fromTimeOfDay timeofday
+ , "for"
+ , fromDuration duration
+ ]
+
+toSchedule :: String -> Maybe Schedule
+toSchedule s = Schedule
+ <$> toRecurrance (unwords recurrance)
+ <*> toTimeOfDay (unwords timeofday)
+ <*> toDuration (unwords duration)
+ where
+ ws = words s
+ (recurrance, ws') = separate (== "at") ws
+ (timeofday, duration) = separate (== "for") ws'
+