module Data.TimeSeries.Series
( DataPoint
, Series
, emptySeries
, series
, size
, slice
, toList
, tsSeries
, valueAt
, values
) where
import Prelude hiding (max, min)
import Data.Time (UTCTime)
import Data.Time.Clock.POSIX (posixSecondsToUTCTime)
data DataPoint a = DP !UTCTime a
deriving (Show, Eq)
instance Functor DataPoint where
fmap f (DP x y) = DP x (f y)
instance Foldable DataPoint where
foldMap f (DP _ y) = f y
data Series a = Series [DataPoint a]
deriving (Show, Eq)
instance Functor Series where
fmap f (Series xs) = Series (map (fmap f) xs)
instance Foldable Series where
foldMap f (Series xs) = foldMap (foldMap f) xs
length = size
emptySeries :: Series a
emptySeries = Series []
series :: [(UTCTime, a)] -> Series a
series xs = Series $ map (\(x, y) -> DP x y) xs
tsSeries :: [Integer] -> [a] -> Series a
tsSeries ts vs = Series (zipWith DP idx vs)
where idx = map (posixSecondsToUTCTime . fromIntegral) ts
toList :: Series a -> [(UTCTime, a)]
toList (Series xs) = map (\(DP x y) -> (x, y)) xs
values :: Series a -> [a]
values ts = map (\(_, y) -> y) (toList ts)
size :: Series a -> Int
size (Series xs) = length xs
valueAt :: Series a -> UTCTime -> Maybe a
valueAt (Series xs) ts = safeHead [y | DP x y <- xs, x == ts]
where safeHead [] = Nothing
safeHead (i:_) = Just i
slice :: Series a -> UTCTime -> UTCTime -> Series a
slice (Series xs) start end = Series [DP x y | DP x y <- xs, x >= start && x <= end]