souplesse/lib/Store.hs

100 lines
2.7 KiB
Haskell
Raw Normal View History

2024-11-04 23:37:48 +00:00
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE DerivingStrategies #-}
{-# LANGUAGE EmptyDataDecls #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE QuasiQuotes #-}
{-# LANGUAGE StandaloneDeriving #-}
{-# LANGUAGE TemplateHaskell #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE UndecidableInstances #-}
2024-11-10 16:18:18 +00:00
module Store
( save,
fetch,
migrateTrkpt,
module Session,
)
where
2024-11-04 23:37:48 +00:00
2024-11-07 18:59:56 +00:00
import Control.Exception
2024-11-05 23:32:01 +00:00
import Control.Monad.IO.Class (MonadIO, liftIO)
2024-11-04 23:37:48 +00:00
import Control.Monad.Trans.Reader (ReaderT)
2024-11-07 18:59:56 +00:00
import Data.Maybe (isJust)
import Data.Time.Clock (DiffTime, NominalDiffTime, UTCTime, addUTCTime)
2024-11-04 23:37:48 +00:00
import Database.Persist
2024-11-05 23:32:01 +00:00
import Database.Persist.Class
2024-11-04 23:37:48 +00:00
import Database.Persist.Postgresql
( ConnectionString,
SqlBackend,
createPostgresqlPool,
pgConnStr,
pgPoolSize,
2024-11-10 16:18:18 +00:00
rawExecute,
2024-11-04 23:37:48 +00:00
runMigration,
runSqlPool,
)
import Database.Persist.TH
import Session qualified
import Text.Read (readMaybe)
2024-11-05 23:32:01 +00:00
import Track (Point (..), Pos (..))
2024-11-04 23:37:48 +00:00
import Track as T
connString :: ConnectionString
connString = "host=127.0.0.1 port=5432 user=souplesse dbname=souplesse password=secret"
share
[mkPersist sqlSettings, mkMigrate "migrateTrkpt"]
2024-11-04 23:37:48 +00:00
[persistLowerCase|
Trkpt
lat Double
lon Double
ele Double Maybe
time UTCTime
cadence Int Maybe
power Int Maybe
heartRate Int Maybe
|]
fromPoint :: Point -> Trkpt
fromPoint p =
let Pos lat lon ele = T.pos p
in Trkpt lat lon ele (T.time p) (T.cadence p) (T.power p) (T.heartRate p)
2024-11-05 23:32:01 +00:00
toPoint :: Entity Trkpt -> Point
toPoint entity =
let tkp = (\(Entity _ tkp) -> tkp) entity
pos = Pos (trkptLat tkp) (trkptLon tkp) (trkptEle tkp)
in Point
pos
(trkptTime tkp)
2024-11-06 19:13:47 +00:00
(trkptCadence tkp)
(trkptPower tkp)
(trkptHeartRate tkp)
2024-11-05 23:32:01 +00:00
2024-11-07 18:59:56 +00:00
data OverlapExists = OverlapExists String deriving (Show)
instance Exception OverlapExists
save :: (MonadIO m) => Track -> ReaderT SqlBackend m (Either OverlapExists Track)
save track = do
let start = T.startTime track
finish = addUTCTime (T.duration track) (T.startTime track)
2024-11-07 18:59:56 +00:00
priors <- selectFirst [TrkptTime >. start, TrkptTime <. finish] []
if isJust priors
then return $ Left (OverlapExists "track overlaps with existing data")
else do
mapM_ (insert . fromPoint) track
Session.refreshDrafts
2024-11-07 18:59:56 +00:00
return $ Right track
2024-11-05 23:32:01 +00:00
fetch :: (MonadIO m) => UTCTime -> NominalDiffTime -> ReaderT SqlBackend m [Point]
fetch start duration = do
let finish = addUTCTime duration start
trkpts <- selectList [TrkptTime >. start, TrkptTime <. finish] []
2024-11-05 23:32:01 +00:00
return $ map toPoint trkpts