{-# LANGUAGE DeriveDataTypeable #-}
module System.Time.Extra(
Seconds,
sleep, timeout,
showDuration,
offsetTime, offsetTimeIncrease, duration
) where
import Control.Concurrent
import System.Clock
import Numeric.Extra
import Control.Monad.IO.Class
import Control.Monad.Extra
import Control.Exception.Extra
import Data.Typeable
import Data.Unique
type Seconds = Double
sleep :: Seconds -> IO ()
sleep :: Seconds -> IO ()
sleep = (Seconds -> IO (Either Seconds ())) -> Seconds -> IO ()
forall (m :: * -> *) a b.
Monad m =>
(a -> m (Either a b)) -> a -> m b
loopM ((Seconds -> IO (Either Seconds ())) -> Seconds -> IO ())
-> (Seconds -> IO (Either Seconds ())) -> Seconds -> IO ()
forall a b. (a -> b) -> a -> b
$ \Seconds
s ->
if Seconds
s Seconds -> Seconds -> Bool
forall a. Ord a => a -> a -> Bool
< Seconds
0 then
Either Seconds () -> IO (Either Seconds ())
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Either Seconds () -> IO (Either Seconds ()))
-> Either Seconds () -> IO (Either Seconds ())
forall a b. (a -> b) -> a -> b
$ () -> Either Seconds ()
forall a b. b -> Either a b
Right ()
else if Seconds
s Seconds -> Seconds -> Bool
forall a. Ord a => a -> a -> Bool
> Seconds
2000 then do
Int -> IO ()
threadDelay Int
2000000000
Either Seconds () -> IO (Either Seconds ())
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Either Seconds () -> IO (Either Seconds ()))
-> Either Seconds () -> IO (Either Seconds ())
forall a b. (a -> b) -> a -> b
$ Seconds -> Either Seconds ()
forall a b. a -> Either a b
Left (Seconds -> Either Seconds ()) -> Seconds -> Either Seconds ()
forall a b. (a -> b) -> a -> b
$ Seconds
s Seconds -> Seconds -> Seconds
forall a. Num a => a -> a -> a
- Seconds
2000
else do
Int -> IO ()
threadDelay (Int -> IO ()) -> Int -> IO ()
forall a b. (a -> b) -> a -> b
$ Seconds -> Int
forall a b. (RealFrac a, Integral b) => a -> b
ceiling (Seconds -> Int) -> Seconds -> Int
forall a b. (a -> b) -> a -> b
$ Seconds
s Seconds -> Seconds -> Seconds
forall a. Num a => a -> a -> a
* Seconds
1000000
Either Seconds () -> IO (Either Seconds ())
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Either Seconds () -> IO (Either Seconds ()))
-> Either Seconds () -> IO (Either Seconds ())
forall a b. (a -> b) -> a -> b
$ () -> Either Seconds ()
forall a b. b -> Either a b
Right ()
newtype Timeout = Timeout Unique deriving (Timeout -> Timeout -> Bool
(Timeout -> Timeout -> Bool)
-> (Timeout -> Timeout -> Bool) -> Eq Timeout
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
/= :: Timeout -> Timeout -> Bool
$c/= :: Timeout -> Timeout -> Bool
== :: Timeout -> Timeout -> Bool
$c== :: Timeout -> Timeout -> Bool
Eq,Typeable)
instance Show Timeout where show :: Timeout -> String
show Timeout
_ = String
"<<timeout>>"
instance Exception Timeout
timeout :: Seconds -> IO a -> IO (Maybe a)
timeout :: Seconds -> IO a -> IO (Maybe a)
timeout Seconds
n IO a
f
| Seconds
n Seconds -> Seconds -> Bool
forall a. Ord a => a -> a -> Bool
<= Seconds
0 = Maybe a -> IO (Maybe a)
forall (f :: * -> *) a. Applicative f => a -> f a
pure Maybe a
forall a. Maybe a
Nothing
| Bool
otherwise = do
ThreadId
pid <- IO ThreadId
myThreadId
Timeout
ex <- (Unique -> Timeout) -> IO Unique -> IO Timeout
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap Unique -> Timeout
Timeout IO Unique
newUnique
(Timeout -> Bool)
-> (Timeout -> IO (Maybe a)) -> IO (Maybe a) -> IO (Maybe a)
forall e a.
Exception e =>
(e -> Bool) -> (e -> IO a) -> IO a -> IO a
handleBool (Timeout -> Timeout -> Bool
forall a. Eq a => a -> a -> Bool
== Timeout
ex)
(IO (Maybe a) -> Timeout -> IO (Maybe a)
forall a b. a -> b -> a
const (IO (Maybe a) -> Timeout -> IO (Maybe a))
-> IO (Maybe a) -> Timeout -> IO (Maybe a)
forall a b. (a -> b) -> a -> b
$ Maybe a -> IO (Maybe a)
forall (f :: * -> *) a. Applicative f => a -> f a
pure Maybe a
forall a. Maybe a
Nothing)
(IO ThreadId
-> (ThreadId -> IO ())
-> (ThreadId -> IO (Maybe a))
-> IO (Maybe a)
forall a b c. IO a -> (a -> IO b) -> (a -> IO c) -> IO c
bracket (((forall a. IO a -> IO a) -> IO ()) -> IO ThreadId
forkIOWithUnmask (((forall a. IO a -> IO a) -> IO ()) -> IO ThreadId)
-> ((forall a. IO a -> IO a) -> IO ()) -> IO ThreadId
forall a b. (a -> b) -> a -> b
$ \forall a. IO a -> IO a
unmask -> IO () -> IO ()
forall a. IO a -> IO a
unmask (IO () -> IO ()) -> IO () -> IO ()
forall a b. (a -> b) -> a -> b
$ Seconds -> IO ()
sleep Seconds
n IO () -> IO () -> IO ()
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> ThreadId -> Timeout -> IO ()
forall e. Exception e => ThreadId -> e -> IO ()
throwTo ThreadId
pid Timeout
ex)
ThreadId -> IO ()
killThread
(\ThreadId
_ -> (a -> Maybe a) -> IO a -> IO (Maybe a)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap a -> Maybe a
forall a. a -> Maybe a
Just IO a
f))
showDuration :: Seconds -> String
showDuration :: Seconds -> String
showDuration Seconds
x
| Seconds
x Seconds -> Seconds -> Bool
forall a. Ord a => a -> a -> Bool
>= Seconds
3600 = Seconds -> String -> ShowS
forall a. RealFrac a => a -> String -> ShowS
f (Seconds
x Seconds -> Seconds -> Seconds
forall a. Fractional a => a -> a -> a
/ Seconds
60) String
"h" String
"m"
| Seconds
x Seconds -> Seconds -> Bool
forall a. Ord a => a -> a -> Bool
>= Seconds
60 = Seconds -> String -> ShowS
forall a. RealFrac a => a -> String -> ShowS
f Seconds
x String
"m" String
"s"
| Bool
otherwise = Int -> Seconds -> String
forall a. RealFloat a => Int -> a -> String
showDP Int
2 Seconds
x String -> ShowS
forall a. [a] -> [a] -> [a]
++ String
"s"
where
f :: a -> String -> ShowS
f a
x String
m String
s = Integer -> String
forall a. Show a => a -> String
show Integer
ms String -> ShowS
forall a. [a] -> [a] -> [a]
++ String
m String -> ShowS
forall a. [a] -> [a] -> [a]
++ [Char
'0' | Integer
ss Integer -> Integer -> Bool
forall a. Ord a => a -> a -> Bool
< Integer
10] String -> ShowS
forall a. [a] -> [a] -> [a]
++ Integer -> String
forall a. Show a => a -> String
show Integer
ss String -> ShowS
forall a. [a] -> [a] -> [a]
++ String
s
where (Integer
ms,Integer
ss) = a -> Integer
forall a b. (RealFrac a, Integral b) => a -> b
round a
x Integer -> Integer -> (Integer, Integer)
forall a. Integral a => a -> a -> (a, a)
`divMod` Integer
60
offsetTime :: IO (IO Seconds)
offsetTime :: IO (IO Seconds)
offsetTime = do
TimeSpec
start <- IO TimeSpec
time
IO Seconds -> IO (IO Seconds)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (IO Seconds -> IO (IO Seconds)) -> IO Seconds -> IO (IO Seconds)
forall a b. (a -> b) -> a -> b
$ do
TimeSpec
end <- IO TimeSpec
time
Seconds -> IO Seconds
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Seconds -> IO Seconds) -> Seconds -> IO Seconds
forall a b. (a -> b) -> a -> b
$ Seconds
1e-9 Seconds -> Seconds -> Seconds
forall a. Num a => a -> a -> a
* Integer -> Seconds
forall a b. (Integral a, Num b) => a -> b
fromIntegral (TimeSpec -> Integer
toNanoSecs (TimeSpec -> Integer) -> TimeSpec -> Integer
forall a b. (a -> b) -> a -> b
$ TimeSpec
end TimeSpec -> TimeSpec -> TimeSpec
forall a. Num a => a -> a -> a
- TimeSpec
start)
where time :: IO TimeSpec
time = Clock -> IO TimeSpec
getTime Clock
Monotonic
{-# DEPRECATED offsetTimeIncrease "Use 'offsetTime' instead, which is guaranteed to always increase." #-}
offsetTimeIncrease :: IO (IO Seconds)
offsetTimeIncrease :: IO (IO Seconds)
offsetTimeIncrease = IO (IO Seconds)
offsetTime
duration :: MonadIO m => m a -> m (Seconds, a)
duration :: m a -> m (Seconds, a)
duration m a
act = do
IO Seconds
time <- IO (IO Seconds) -> m (IO Seconds)
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO IO (IO Seconds)
offsetTime
a
res <- m a
act
Seconds
time <- IO Seconds -> m Seconds
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO IO Seconds
time
(Seconds, a) -> m (Seconds, a)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Seconds
time, a
res)