2020-07-28 19:40:17 +02:00
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
|
|
|
{-# LANGUAGE ScopedTypeVariables #-}
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
module Sessions where
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
import Database.SQLite.Simple
|
|
|
|
|
|
|
|
import qualified Data.Time.Clock as Clock
|
|
|
|
import qualified Types as T
|
|
|
|
import qualified System.Random as Random
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
-- | Return True if `session` was created at most three hours ago.
|
|
|
|
isValid :: T.StoredSession -> IO Bool
|
|
|
|
isValid session = do
|
|
|
|
t1 <- Clock.getCurrentTime
|
|
|
|
let t0 = T.storedSessionTsCreated session in
|
|
|
|
pure $ Clock.diffUTCTime t1 t0 <= 3 * 60 * 60
|
|
|
|
|
2020-07-29 21:21:56 +02:00
|
|
|
-- | Lookup the session by UUID.
|
|
|
|
get :: FilePath -> T.SessionUUID -> IO (Maybe T.StoredSession)
|
|
|
|
get dbFile uuid = withConnection dbFile $ \conn -> do
|
2020-07-30 20:52:04 +02:00
|
|
|
res <- query conn "SELECT uuid,username,tsCreated FROM Sessions WHERE uuid = ?" (Only uuid)
|
2020-07-29 21:21:56 +02:00
|
|
|
case res of
|
|
|
|
[x] -> pure (Just x)
|
|
|
|
_ -> pure Nothing
|
|
|
|
|
2020-07-28 19:40:17 +02:00
|
|
|
-- | Lookup the session stored under `username` in `dbFile`.
|
|
|
|
find :: FilePath -> T.Username -> IO (Maybe T.StoredSession)
|
|
|
|
find dbFile username = withConnection dbFile $ \conn -> do
|
2020-07-30 20:52:04 +02:00
|
|
|
res <- query conn "SELECT uuid,username,tsCreated FROM Sessions WHERE username = ?" (Only username)
|
2020-07-28 19:40:17 +02:00
|
|
|
case res of
|
|
|
|
[x] -> pure (Just x)
|
|
|
|
_ -> pure Nothing
|
|
|
|
|
|
|
|
-- | Create a session under the `username` key in `dbFile`.
|
|
|
|
create :: FilePath -> T.Username -> IO T.SessionUUID
|
|
|
|
create dbFile username = withConnection dbFile $ \conn -> do
|
|
|
|
now <- Clock.getCurrentTime
|
|
|
|
uuid <- Random.randomIO
|
|
|
|
execute conn "INSERT INTO Sessions (uuid,username,tsCreated) VALUES (?,?,?)"
|
|
|
|
(T.SessionUUID uuid, username, now)
|
|
|
|
pure (T.SessionUUID uuid)
|
|
|
|
|
|
|
|
-- | Reset the tsCreated field to the current time to ensure the token is valid.
|
|
|
|
refresh :: FilePath -> T.SessionUUID -> IO ()
|
|
|
|
refresh dbFile uuid = withConnection dbFile $ \conn -> do
|
|
|
|
now <- Clock.getCurrentTime
|
|
|
|
execute conn "UPDATE Sessions SET tsCreated = ? WHERE uuid = ?"
|
|
|
|
(now, uuid)
|
|
|
|
pure ()
|
|
|
|
|
|
|
|
-- | Delete the session under `username` from `dbFile`.
|
2020-07-29 21:21:29 +02:00
|
|
|
delete :: FilePath -> T.SessionUUID -> IO ()
|
|
|
|
delete dbFile uuid = withConnection dbFile $ \conn ->
|
|
|
|
execute conn "DELETE FROM Sessions WHERE uuid = ?" (Only uuid)
|
2020-07-28 19:40:17 +02:00
|
|
|
|
|
|
|
-- | Find or create a session in the Sessions table. If a session exists,
|
|
|
|
-- refresh the token's validity.
|
|
|
|
findOrCreate :: FilePath -> T.Account -> IO T.SessionUUID
|
2020-08-04 10:19:48 +02:00
|
|
|
findOrCreate dbFile account =
|
2020-07-28 19:40:17 +02:00
|
|
|
let username = T.accountUsername account in do
|
|
|
|
mSession <- find dbFile username
|
|
|
|
case mSession of
|
|
|
|
Nothing -> create dbFile username
|
|
|
|
Just session ->
|
|
|
|
let uuid = T.storedSessionUUID session in do
|
|
|
|
refresh dbFile uuid
|
|
|
|
pure uuid
|
|
|
|
|
|
|
|
-- | Return a list of all sessions in the Sessions table.
|
|
|
|
list :: FilePath -> IO [T.StoredSession]
|
|
|
|
list dbFile = withConnection dbFile $ \conn ->
|
2020-07-30 20:52:04 +02:00
|
|
|
query_ conn "SELECT uuid,username,tsCreated FROM Sessions"
|