Skip to content

Commit

Permalink
[ADP-3335] Change Cardano.Wallet.Deposit.IO.DB to depend on `sqlite…
Browse files Browse the repository at this point in the history
…-simple` (#4526)

This pull request changes `Cardano.Wallet.Deposit.IO.DB` to use
`sqlite-simple` instead of `persistent`.

The reason for this change is that `withDBHandleInMemory` throws an
exception when called *twice* in some circumstances.

It appears that this bug is located in a dependency. Rumor has it that
there might be a bug in `persistent` where SQL statements are not
finalized. It seems likely that the SQL statement which is not finalized
is the `BEGIN TRANSACTION` statement that `persistent` reuses in each
call to `runSqlConn`. I did not investigate this further, as I want to
move away from `persistent` anyway, and using `sqlite-simple` fixes the
issue.

### Comment

This pull request prepares the implementation of a mock environment for
the Deposit Wallet. In turn, this enables execution of user scenarios.

### Issue Number

ADP-3335
  • Loading branch information
HeinrichApfelmus authored May 13, 2024
2 parents eafaf8f + f186998 commit 348cac6
Show file tree
Hide file tree
Showing 3 changed files with 22 additions and 71 deletions.
2 changes: 2 additions & 0 deletions lib/customer-deposit-wallet/customer-deposit-wallet.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -58,7 +58,9 @@ library
, delta-types
, iohk-monitoring-extra ^>=0.1
, persistent >= 2.13 && < 2.15
, sqlite-simple >= 0.4.19.0 && < 0.5
, text
, transformers
, time
exposed-modules:
Cardano.Wallet.Deposit.IO
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,7 @@ onWalletState
-> IO r
onWalletState WalletInstance{env,walletState} update' =
atomically env $ Delta.onDBVar walletState update'
-- FIXME: Propagation of exceptions from Pure to IO.

-- | Convenience to read the 'WalletState'.
--
Expand Down
90 changes: 19 additions & 71 deletions lib/customer-deposit-wallet/src/Cardano/Wallet/Deposit/IO/DB.hs
Original file line number Diff line number Diff line change
Expand Up @@ -16,30 +16,27 @@ import Cardano.BM.Extra
)
import Cardano.DB.Sqlite
( DBLog (..)
, dbBackend
, withDBHandle
, withDBHandleInMemory
)
import Control.Concurrent.MVar
( newMVar
, withMVar
, withMVarMasked
)
import Control.Monad.Trans.Reader
( ReaderT (..)
)
import Control.Tracer
( Tracer
, contramap
)
import Data.Time.Clock
( NominalDiffTime
, traceWith
)

import qualified Database.Persist.Sql as Persistent
import qualified Database.SQLite.Simple as Sqlite

{-----------------------------------------------------------------------------
SqlContext
------------------------------------------------------------------------------}
-- | Monad to run SQL queries in.
type SqlM = Persistent.SqlPersistT IO
type SqlM = ReaderT Sqlite.Connection IO

-- | A facility to run 'SqlM' computations.
-- Importantly, computations are not run in parallel, but sequenced.
Expand All @@ -54,75 +51,26 @@ withSqlContextInMemory
-> (SqlContext -> IO a)
-- ^ Action to run
-> IO a
withSqlContextInMemory tr action = do
withDBHandleInMemory tr $ \dbhandle -> do
-- Lock ensures that database operations are sequenced.
lock <- newMVar (dbBackend dbhandle)
let runSqlM :: SqlM a -> IO a
runSqlM cmd =
withMVarMasked lock (observe . Persistent.runSqlConn cmd)
action $ SqlContext{runSqlM}
where
observe :: IO a -> IO a
observe = bracketTracer (contramap MsgRun tr)
withSqlContextInMemory tr = withSqliteFile tr ":memory:"

-- | Open an .sqlite database file
-- | Use sqlite to open a database file
-- and provide an 'SqlContext' for running 'SqlM' actions.
withSqliteFile
:: Tracer IO DBLog
-- ^ Logging
-- ^ Logging
-> FilePath
-- ^ Database file
-- ^ Database file
-> (SqlContext -> IO a)
-- ^ Action to run
-- ^ Action to run
-> IO a
withSqliteFile tr fp action = do
-- Lock ensures that database operations are sequenced.
lock <- newMVar ()
withDBHandle tr fp $ \dbHandle ->
let
-- Run a query on the open database,
-- but retry on busy.
runSqlM :: SqlM a -> IO a
runSqlM cmd =
observe
. retryOnBusy tr retryOnBusyTimeout
$ withMVar lock
$ const
$ Persistent.runSqlConn cmd
$ dbBackend dbHandle
in
action $ SqlContext{runSqlM}
withSqliteFile tr filepath action =
Sqlite.withConnection filepath $ \connection0 -> do
traceWith tr $ MsgOpenSingleConnection filepath
-- The lock ensures that database operations are sequenced.
lock <- newMVar connection0
let runSqlM :: SqlM a -> IO a
runSqlM cmd = withMVar lock (observe . runReaderT cmd)
action SqlContext{runSqlM}
where
observe :: IO a -> IO a
observe = bracketTracer (contramap MsgRun tr)

-- | Retry an action if the database yields an 'SQLITE_BUSY' error.
--
-- From <https://www.sqlite.org/rescode.html#busy>
--
-- The SQLITE_BUSY result code indicates that the database file could not be
-- written (or in some cases read) because of concurrent activity by some
-- other database connection, usually a database connection in a separate
-- process.
--
-- For example, if process A is in the middle of a large write transaction
-- and at the same time process B attempts to start a new write transaction,
-- process B will get back an SQLITE_BUSY result because SQLite only supports
-- one writer at a time. Process B will need to wait for process A to finish
-- its transaction before starting a new transaction. The sqlite3_busy_timeout()
-- and sqlite3_busy_handler() interfaces and the busy_timeout pragma are
-- available to process B to help it deal with SQLITE_BUSY errors.
retryOnBusy
:: Tracer IO DBLog
-- ^ Logging
-> NominalDiffTime
-- ^ Timeout
-> IO a
-- ^ Action to retry
-> IO a
retryOnBusy _ _ action = action -- FIXME

-- | Default timeout for `retryOnBusy`
retryOnBusyTimeout :: NominalDiffTime
retryOnBusyTimeout = 60

0 comments on commit 348cac6

Please sign in to comment.