From 1edc198376f70a1f2a395b9b8893758a0b450326 Mon Sep 17 00:00:00 2001 From: Matt Parsons Date: Thu, 23 Nov 2017 11:49:57 -0700 Subject: [PATCH 01/20] Add Generic instance and remove constraint --- persistent/Database/Persist/Class/PersistEntity.hs | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/persistent/Database/Persist/Class/PersistEntity.hs b/persistent/Database/Persist/Class/PersistEntity.hs index 9b8412054..9c077ec03 100644 --- a/persistent/Database/Persist/Class/PersistEntity.hs +++ b/persistent/Database/Persist/Class/PersistEntity.hs @@ -1,4 +1,5 @@ {-# LANGUAGE CPP #-} +{-# LANGUAGE DeriveGeneric #-} {-# LANGUAGE ExistentialQuantification #-} {-# LANGUAGE TypeFamilies #-} {-# LANGUAGE OverloadedStrings #-} @@ -43,6 +44,7 @@ import Data.Monoid (mappend) import qualified Data.HashMap.Strict as HM import Data.Typeable (Typeable) import Data.Maybe (isJust) +import GHC.Generics -- | Persistent serialized Haskell records to the database. -- A Database 'Entity' (A row in SQL, a document in MongoDB, etc) @@ -170,10 +172,11 @@ data Filter record = forall typ. PersistField typ => Filter -- your query returns two entities (i.e. @(Entity backend a, -- Entity backend b)@), then you must you use @SELECT ??, ?? -- WHERE ...@, and so on. -data Entity record = PersistEntity record => +data Entity record = Entity { entityKey :: Key record , entityVal :: record } +deriving instance (PersistEntity record, Generic (Key record), Generic record) => Generic (Entity record) deriving instance (PersistEntity record, Eq (Key record), Eq record) => Eq (Entity record) deriving instance (PersistEntity record, Ord (Key record), Ord record) => Ord (Entity record) deriving instance (PersistEntity record, Show (Key record), Show record) => Show (Entity record) From 868a628abb5cc55d3ad32a40804980ec6f0e6723 Mon Sep 17 00:00:00 2001 From: Matt Parsons Date: Sun, 17 Dec 2017 09:24:35 -0700 Subject: [PATCH 02/20] rename --- persistent-mysql/Database/Persist/MySQL.hs | 34 +++++++++++++--------- 1 file changed, 20 insertions(+), 14 deletions(-) diff --git a/persistent-mysql/Database/Persist/MySQL.hs b/persistent-mysql/Database/Persist/MySQL.hs index 39b8bed97..1ff442dc8 100644 --- a/persistent-mysql/Database/Persist/MySQL.hs +++ b/persistent-mysql/Database/Persist/MySQL.hs @@ -18,7 +18,7 @@ module Database.Persist.MySQL , mockMigration , insertOnDuplicateKeyUpdate , insertManyOnDuplicateKeyUpdate - , SomeField(SomeField) + , HandleUpdateCollision(HandleUpdateCollision) , copyUnlessNull , copyUnlessEmpty , copyUnlessEq @@ -1046,41 +1046,47 @@ insertOnDuplicateKeyUpdate record = -- 'insertManyOnDuplicateKeyUpdate' in the library. -- -- @since 2.6.2 -data SomeField record where +data HandleUpdateCollision record where -- | Copy the field directly from the record. - SomeField :: EntityField record typ -> SomeField record + CopyField :: EntityField record typ -> HandleUpdateCollision record -- | Only copy the field if it is not equal to the provided value. - CopyUnlessEq :: PersistField typ => EntityField record typ -> typ -> SomeField record + CopyUnlessEq :: PersistField typ => EntityField record typ -> typ -> HandleUpdateCollision record -- | Copy the field into the database only if the value in the -- corresponding record is non-@NULL@. -- -- @since 2.6.2 -copyUnlessNull :: PersistField typ => EntityField record (Maybe typ) -> SomeField record +copyUnlessNull :: PersistField typ => EntityField record (Maybe typ) -> HandleUpdateCollision record copyUnlessNull field = CopyUnlessEq field Nothing -- | Copy the field into the database only if the value in the -- corresponding record is non-empty, where "empty" means the Monoid -- definition for 'mempty'. Useful for 'Text', 'String', 'ByteString', etc. -- --- The resulting 'SomeField' type is useful for the +-- The resulting 'HandleUpdateCollision' type is useful for the -- 'insertManyOnDuplicateKeyUpdate' function. -- -- @since 2.6.2 -copyUnlessEmpty :: (Monoid.Monoid typ, PersistField typ) => EntityField record typ -> SomeField record +copyUnlessEmpty :: (Monoid.Monoid typ, PersistField typ) => EntityField record typ -> HandleUpdateCollision record copyUnlessEmpty field = CopyUnlessEq field Monoid.mempty -- | Copy the field into the database only if the field is not equal to the -- provided value. This is useful to avoid copying weird nullary data into -- the database. -- --- The resulting 'SomeField' type is useful for the +-- The resulting 'HandleUpdateCollision' type is useful for the -- 'insertManyOnDuplicateKeyUpdate' function. -- -- @since 2.6.2 -copyUnlessEq :: PersistField typ => EntityField record typ -> typ -> SomeField record +copyUnlessEq :: PersistField typ => EntityField record typ -> typ -> HandleUpdateCollision record copyUnlessEq = CopyUnlessEq +-- | Copy the field directly from the record. +-- +-- @since 3.0 +copyField :: PersistField typ => EntityField record typ -> typ -> HandleUpdateCollision record +copyField = CopyField + -- | Do a bulk insert on the given records in the first parameter. In the event -- that a key conflicts with a record currently in the database, the second and -- third parameters determine what will happen. @@ -1093,9 +1099,9 @@ copyUnlessEq = CopyUnlessEq -- the value that is provided. You can use this to increment a counter value. -- These updates only occur if the original record is present in the database. -- --- === __More details on 'SomeField' usage__ +-- === __More details on 'HandleUpdateCollision' usage__ -- --- The @['SomeField']@ parameter allows you to specify which fields (and +-- The @['HandleUpdateCollision']@ parameter allows you to specify which fields (and -- under which conditions) will be copied from the inserted rows. For -- a brief example, consider the following data model and existing data set: -- @@ -1177,7 +1183,7 @@ insertManyOnDuplicateKeyUpdate , MonadIO m ) => [record] -- ^ A list of the records you want to insert, or update - -> [SomeField record] -- ^ A list of the fields you want to copy over. + -> [HandleUpdateCollision record] -- ^ A list of the fields you want to copy over. -> [Update record] -- ^ A list of the updates to apply that aren't dependent on the record being inserted. -> ReaderT backend m () insertManyOnDuplicateKeyUpdate [] _ _ = return () @@ -1192,14 +1198,14 @@ insertManyOnDuplicateKeyUpdate records fieldValues updates = mkBulkInsertQuery :: PersistEntity record => [record] -- ^ A list of the records you want to insert, or update - -> [SomeField record] -- ^ A list of the fields you want to copy over. + -> [HandleUpdateCollision record] -- ^ A list of the fields you want to copy over. -> [Update record] -- ^ A list of the updates to apply that aren't dependent on the record being inserted. -> (Text, [PersistValue]) mkBulkInsertQuery records fieldValues updates = (q, recordValues <> updsValues <> copyUnlessValues) where mfieldDef x = case x of - SomeField rec -> Right (fieldDbToText (persistFieldDef rec)) + HandleUpdateCollision rec -> Right (fieldDbToText (persistFieldDef rec)) CopyUnlessEq rec val -> Left (fieldDbToText (persistFieldDef rec), toPersistValue val) (fieldsToMaybeCopy, updateFieldNames) = partitionEithers $ map mfieldDef fieldValues fieldDbToText = T.pack . escapeDBName . fieldDB From 8b54a5edb807f0b9636e6056ae70964b8940d418 Mon Sep 17 00:00:00 2001 From: Matt Parsons Date: Sun, 17 Dec 2017 09:42:00 -0700 Subject: [PATCH 03/20] MAke the query type opaque, deprecate old type/constructors --- persistent-mysql/ChangeLog.md | 3 +++ persistent-mysql/Database/Persist/MySQL.hs | 27 +++++++++++++++++----- stack.yaml | 4 ++++ 3 files changed, 28 insertions(+), 6 deletions(-) diff --git a/persistent-mysql/ChangeLog.md b/persistent-mysql/ChangeLog.md index dc8e6be41..e658fe272 100644 --- a/persistent-mysql/ChangeLog.md +++ b/persistent-mysql/ChangeLog.md @@ -4,6 +4,9 @@ -- This can be released as a minor change on the next update. Currently persistent-mysql can't be released because 2.6.2.2 depends on persistent-2.7.2 being released. +* The `SomeField` type was renamed to `HandleUpdateCollision` and deprecated. Please migrate to using `HandleUpdateCollision`. +* The `SomeField` constructor was deprecated, and a temporary pattern synonym introduced. Please migrate to using `copyField`. + ## 2.6.2.2 [UNRELEASED ON HACKAGE] -- This version depends on persistent 2.7.2, which introduced breaking changes and is deprecated on hackage. diff --git a/persistent-mysql/Database/Persist/MySQL.hs b/persistent-mysql/Database/Persist/MySQL.hs index 1ff442dc8..79857dfe4 100644 --- a/persistent-mysql/Database/Persist/MySQL.hs +++ b/persistent-mysql/Database/Persist/MySQL.hs @@ -16,9 +16,10 @@ module Database.Persist.MySQL , MySQLBase.defaultSSLInfo , MySQLConf(..) , mockMigration + -- * @ON DUPLICATE KEY UPDATE@ Functionality , insertOnDuplicateKeyUpdate , insertManyOnDuplicateKeyUpdate - , HandleUpdateCollision(HandleUpdateCollision) + , HandleUpdateCollision , copyUnlessNull , copyUnlessEmpty , copyUnlessEq @@ -1026,8 +1027,10 @@ mockMigration mig = do resp <- result sqlbackend mapM_ T.putStrLn $ map snd $ snd resp --- | MySQL specific 'upsert'. This will prevent multiple queries, when one will --- do. +-- | MySQL specific 'upsert_'. This will prevent multiple queries, when one will +-- do. The record will be inserted into the database. In the event that the +-- record already exists in the database, the record will have the +-- relevant updates performed. insertOnDuplicateKeyUpdate :: ( backend ~ PersistEntityBackend record , PersistEntity record @@ -1042,16 +1045,28 @@ insertOnDuplicateKeyUpdate record = insertManyOnDuplicateKeyUpdate [record] [] -- | This type is used to determine how to update rows using MySQL's --- @INSERT ON DUPLICATE KEY UPDATE@ functionality, exposed via --- 'insertManyOnDuplicateKeyUpdate' in the library. +-- @INSERT ... ON DUPLICATE KEY UPDATE@ functionality, exposed via +-- 'insertManyOnDuplicateKeyUpdate' in this library. -- --- @since 2.6.2 +-- @since 3.0.0 data HandleUpdateCollision record where -- | Copy the field directly from the record. CopyField :: EntityField record typ -> HandleUpdateCollision record -- | Only copy the field if it is not equal to the provided value. CopyUnlessEq :: PersistField typ => EntityField record typ -> typ -> HandleUpdateCollision record +-- | An alias for 'HandleUpdateCollision'. The type previously was only +-- used to copy a single value, but was expanded to be handle more complex +-- queries. +-- +-- @since 2.6.2 +type SomeField = HandleUpdateCollision +{-# DEPRECATED SomeField "The type SomeField was renamed to HandleUpdateCollision. Please migrate to that name." } + +pattern SomeField :: EntityField record typ -> SomeField record +pattern SomeField x = CopyField x +{-# DEPRECATED SomeField "The constructor SomeField is deprecated. Use the function copyField instead."} + -- | Copy the field into the database only if the value in the -- corresponding record is non-@NULL@. -- diff --git a/stack.yaml b/stack.yaml index f87b07dab..a5e39b6f4 100644 --- a/stack.yaml +++ b/stack.yaml @@ -8,3 +8,7 @@ packages: - ./persistent-mysql - ./persistent-postgresql - ./persistent-redis + + +extra-deps: +- mysql-simple-0.4.4 From 21ba9bdfc47788ddc4e2e7b39faf9246274d732b Mon Sep 17 00:00:00 2001 From: Matt Parsons Date: Sun, 17 Dec 2017 10:02:27 -0700 Subject: [PATCH 04/20] Update tests --- persistent-mysql/Database/Persist/MySQL.hs | 11 ++++++----- persistent-test/src/InsertDuplicateUpdate.hs | 4 ++-- 2 files changed, 8 insertions(+), 7 deletions(-) diff --git a/persistent-mysql/Database/Persist/MySQL.hs b/persistent-mysql/Database/Persist/MySQL.hs index 79857dfe4..cd029eef7 100644 --- a/persistent-mysql/Database/Persist/MySQL.hs +++ b/persistent-mysql/Database/Persist/MySQL.hs @@ -1,4 +1,5 @@ {-# LANGUAGE ScopedTypeVariables #-} +{-# LANGUAGE PatternSynonyms #-} {-# LANGUAGE ConstraintKinds #-} {-# LANGUAGE GADTs #-} {-# LANGUAGE OverloadedStrings #-} @@ -19,7 +20,8 @@ module Database.Persist.MySQL -- * @ON DUPLICATE KEY UPDATE@ Functionality , insertOnDuplicateKeyUpdate , insertManyOnDuplicateKeyUpdate - , HandleUpdateCollision + , HandleUpdateCollision(SomeField) + , copyField , copyUnlessNull , copyUnlessEmpty , copyUnlessEq @@ -1061,11 +1063,10 @@ data HandleUpdateCollision record where -- -- @since 2.6.2 type SomeField = HandleUpdateCollision -{-# DEPRECATED SomeField "The type SomeField was renamed to HandleUpdateCollision. Please migrate to that name." } pattern SomeField :: EntityField record typ -> SomeField record pattern SomeField x = CopyField x -{-# DEPRECATED SomeField "The constructor SomeField is deprecated. Use the function copyField instead."} +{-# DEPRECATED SomeField "The type SomeField is deprecated. Use the type HandleUpdateCollision instead, and use the function copyField instead of the data constructor." #-} -- | Copy the field into the database only if the value in the -- corresponding record is non-@NULL@. @@ -1099,7 +1100,7 @@ copyUnlessEq = CopyUnlessEq -- | Copy the field directly from the record. -- -- @since 3.0 -copyField :: PersistField typ => EntityField record typ -> typ -> HandleUpdateCollision record +copyField :: PersistField typ => EntityField record typ -> HandleUpdateCollision record copyField = CopyField -- | Do a bulk insert on the given records in the first parameter. In the event @@ -1220,7 +1221,7 @@ mkBulkInsertQuery records fieldValues updates = (q, recordValues <> updsValues <> copyUnlessValues) where mfieldDef x = case x of - HandleUpdateCollision rec -> Right (fieldDbToText (persistFieldDef rec)) + CopyField rec -> Right (fieldDbToText (persistFieldDef rec)) CopyUnlessEq rec val -> Left (fieldDbToText (persistFieldDef rec), toPersistValue val) (fieldsToMaybeCopy, updateFieldNames) = partitionEithers $ map mfieldDef fieldValues fieldDbToText = T.pack . escapeDBName . fieldDB diff --git a/persistent-test/src/InsertDuplicateUpdate.hs b/persistent-test/src/InsertDuplicateUpdate.hs index 06b9761e6..ef9bee8e2 100644 --- a/persistent-test/src/InsertDuplicateUpdate.hs +++ b/persistent-test/src/InsertDuplicateUpdate.hs @@ -60,7 +60,7 @@ specs = describe "DuplicateKeyUpdate" $ do let newItem = Item "item3" "fresh" Nothing Nothing insertManyOnDuplicateKeyUpdate (newItem : items) - [SomeField ItemDescription] + [copyField ItemDescription] [] dbItems <- map entityVal <$> selectList [] [] sort dbItems @== sort (newItem : items) @@ -79,7 +79,7 @@ specs = describe "DuplicateKeyUpdate" $ do insertManyOnDuplicateKeyUpdate newItems [ copyUnlessEq ItemQuantity (Just 0) - , SomeField ItemPrice + , copyField ItemPrice ] [] dbItems <- sort . fmap entityVal <$> selectList [] [] From f0525021d28eb7096915cdf58921096b281e35a0 Mon Sep 17 00:00:00 2001 From: Matt Parsons Date: Sun, 17 Dec 2017 11:21:00 -0700 Subject: [PATCH 05/20] Some CPP Hackery --- persistent-mysql/Database/Persist/MySQL.hs | 97 +++++++++++++--------- 1 file changed, 57 insertions(+), 40 deletions(-) diff --git a/persistent-mysql/Database/Persist/MySQL.hs b/persistent-mysql/Database/Persist/MySQL.hs index cd029eef7..09bf2fcda 100644 --- a/persistent-mysql/Database/Persist/MySQL.hs +++ b/persistent-mysql/Database/Persist/MySQL.hs @@ -1,4 +1,5 @@ {-# LANGUAGE ScopedTypeVariables #-} +{-# LANGUAGE CPP #-} {-# LANGUAGE PatternSynonyms #-} {-# LANGUAGE ConstraintKinds #-} {-# LANGUAGE GADTs #-} @@ -20,7 +21,13 @@ module Database.Persist.MySQL -- * @ON DUPLICATE KEY UPDATE@ Functionality , insertOnDuplicateKeyUpdate , insertManyOnDuplicateKeyUpdate +#if MIN_VERSION_base(4,7,0) + , HandleUpdateCollision + , pattern SomeField +#elif MIN_VERSION_BASE(4,9,0) , HandleUpdateCollision(SomeField) +#endif + , SomeField , copyField , copyUnlessNull , copyUnlessEmpty @@ -468,44 +475,50 @@ getColumns :: MySQL.ConnectInfo ) getColumns connectInfo getter def = do -- Find out ID column. - stmtIdClmn <- getter "SELECT COLUMN_NAME, \ - \IS_NULLABLE, \ - \DATA_TYPE, \ - \COLUMN_DEFAULT \ - \FROM INFORMATION_SCHEMA.COLUMNS \ - \WHERE TABLE_SCHEMA = ? \ - \AND TABLE_NAME = ? \ - \AND COLUMN_NAME = ?" + stmtIdClmn <- getter $ T.concat + [ "SELECT COLUMN_NAME, " + , "IS_NULLABLE, " + , "DATA_TYPE, " + , "COLUMN_DEFAULT " + , "FROM INFORMATION_SCHEMA.COLUMNS " + , "WHERE TABLE_SCHEMA = ? " + , "AND TABLE_NAME = ? " + , "AND COLUMN_NAME = ?" + ] inter1 <- with (stmtQuery stmtIdClmn vals) ($$ CL.consume) ids <- runResourceT $ CL.sourceList inter1 $$ helperClmns -- avoid nested queries -- Find out all columns. - stmtClmns <- getter "SELECT COLUMN_NAME, \ - \IS_NULLABLE, \ - \DATA_TYPE, \ - \COLUMN_TYPE, \ - \CHARACTER_MAXIMUM_LENGTH, \ - \NUMERIC_PRECISION, \ - \NUMERIC_SCALE, \ - \COLUMN_DEFAULT \ - \FROM INFORMATION_SCHEMA.COLUMNS \ - \WHERE TABLE_SCHEMA = ? \ - \AND TABLE_NAME = ? \ - \AND COLUMN_NAME <> ?" + stmtClmns <- getter $ T.concat + [ "SELECT COLUMN_NAME, " + , "IS_NULLABLE, " + , "DATA_TYPE, " + , "COLUMN_TYPE, " + , "CHARACTER_MAXIMUM_LENGTH, " + , "NUMERIC_PRECISION, " + , "NUMERIC_SCALE, " + , "COLUMN_DEFAULT " + , "FROM INFORMATION_SCHEMA.COLUMNS " + , "WHERE TABLE_SCHEMA = ? " + , "AND TABLE_NAME = ? " + , "AND COLUMN_NAME <> ?" + ] inter2 <- with (stmtQuery stmtClmns vals) ($$ CL.consume) cs <- runResourceT $ CL.sourceList inter2 $$ helperClmns -- avoid nested queries -- Find out the constraints. - stmtCntrs <- getter "SELECT CONSTRAINT_NAME, \ - \COLUMN_NAME \ - \FROM INFORMATION_SCHEMA.KEY_COLUMN_USAGE \ - \WHERE TABLE_SCHEMA = ? \ - \AND TABLE_NAME = ? \ - \AND COLUMN_NAME <> ? \ - \AND CONSTRAINT_NAME <> 'PRIMARY' \ - \AND REFERENCED_TABLE_SCHEMA IS NULL \ - \ORDER BY CONSTRAINT_NAME, \ - \COLUMN_NAME" + stmtCntrs <- getter $ T.concat + [ "SELECT CONSTRAINT_NAME, " + , "COLUMN_NAME " + , "FROM INFORMATION_SCHEMA.KEY_COLUMN_USAGE " + , "WHERE TABLE_SCHEMA = ? " + , "AND TABLE_NAME = ? " + , "AND COLUMN_NAME <> ? " + , "AND CONSTRAINT_NAME <> 'PRIMARY' " + , "AND REFERENCED_TABLE_SCHEMA IS NULL " + , "ORDER BY CONSTRAINT_NAME, " + , "COLUMN_NAME" + ] us <- with (stmtQuery stmtCntrs vals) ($$ helperCntrs) -- Return both @@ -559,16 +572,18 @@ getColumn connectInfo getter tname [ PersistText cname _ -> fail $ "Invalid default column: " ++ show default' -- Foreign key (if any) - stmt <- lift $ getter "SELECT REFERENCED_TABLE_NAME, \ - \CONSTRAINT_NAME, \ - \ORDINAL_POSITION \ - \FROM INFORMATION_SCHEMA.KEY_COLUMN_USAGE \ - \WHERE TABLE_SCHEMA = ? \ - \AND TABLE_NAME = ? \ - \AND COLUMN_NAME = ? \ - \AND REFERENCED_TABLE_SCHEMA = ? \ - \ORDER BY CONSTRAINT_NAME, \ - \COLUMN_NAME" + stmt <- lift . getter $ T.concat + [ "SELECT REFERENCED_TABLE_NAME, " + , "CONSTRAINT_NAME, " + , "ORDINAL_POSITION " + , "FROM INFORMATION_SCHEMA.KEY_COLUMN_USAGE " + , "WHERE TABLE_SCHEMA = ? " + , "AND TABLE_NAME = ? " + , "AND COLUMN_NAME = ? " + , "AND REFERENCED_TABLE_SCHEMA = ? " + , "ORDER BY CONSTRAINT_NAME, " + , "COLUMN_NAME" + ] let vars = [ PersistText $ pack $ MySQL.connectDatabase connectInfo , PersistText $ unDBName $ tname , PersistText cname @@ -1064,7 +1079,9 @@ data HandleUpdateCollision record where -- @since 2.6.2 type SomeField = HandleUpdateCollision +#if MIN_VERSION_base(4,8,0) pattern SomeField :: EntityField record typ -> SomeField record +#endif pattern SomeField x = CopyField x {-# DEPRECATED SomeField "The type SomeField is deprecated. Use the type HandleUpdateCollision instead, and use the function copyField instead of the data constructor." #-} From c882203c9cc09ba28b2012b58b4cd0fdc415e6ce Mon Sep 17 00:00:00 2001 From: Matt Parsons Date: Sun, 17 Dec 2017 11:49:13 -0700 Subject: [PATCH 06/20] love cpp --- persistent-mysql/Database/Persist/MySQL.hs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/persistent-mysql/Database/Persist/MySQL.hs b/persistent-mysql/Database/Persist/MySQL.hs index 09bf2fcda..4406d517e 100644 --- a/persistent-mysql/Database/Persist/MySQL.hs +++ b/persistent-mysql/Database/Persist/MySQL.hs @@ -24,7 +24,7 @@ module Database.Persist.MySQL #if MIN_VERSION_base(4,7,0) , HandleUpdateCollision , pattern SomeField -#elif MIN_VERSION_BASE(4,9,0) +#elif MIN_VERSION_base(4,9,0) , HandleUpdateCollision(SomeField) #endif , SomeField From 61bd5fedc3d71c5eb361843f24965b36c37a4e3f Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Sun, 31 Dec 2017 09:21:00 +0200 Subject: [PATCH 07/20] Compiling with latest conduit --- .../Database/Persist/MongoDB.hs | 16 ++++--- persistent-mongoDB/persistent-mongoDB.cabal | 2 +- persistent-mysql/Database/Persist/MySQL.hs | 10 ++-- persistent-mysql/persistent-mysql.cabal | 2 +- .../Database/Persist/Postgresql.hs | 18 ++++---- .../persistent-postgresql.cabal | 2 +- persistent-redis/persistent-redis.cabal | 2 +- persistent-sqlite/Database/Persist/Sqlite.hs | 28 ++++++----- persistent-sqlite/persistent-sqlite.cabal | 2 +- persistent-test/persistent-test.cabal | 3 +- persistent-test/src/CompositeTest.hs | 10 ---- persistent-test/src/Init.hs | 8 ++-- persistent-test/src/PersistentTest.hs | 14 +----- .../Database/Persist/Class/PersistConfig.hs | 5 +- .../Database/Persist/Class/PersistStore.hs | 2 +- persistent/Database/Persist/Sql/Migration.hs | 9 ++-- .../Persist/Sql/Orphan/PersistStore.hs | 2 +- persistent/Database/Persist/Sql/Run.hs | 46 +++++++++---------- persistent/persistent.cabal | 6 +-- 19 files changed, 81 insertions(+), 106 deletions(-) diff --git a/persistent-mongoDB/Database/Persist/MongoDB.hs b/persistent-mongoDB/Database/Persist/MongoDB.hs index 0d3374921..cde2d4c3f 100644 --- a/persistent-mongoDB/Database/Persist/MongoDB.hs +++ b/persistent-mongoDB/Database/Persist/MongoDB.hs @@ -152,7 +152,7 @@ import Data.Bits (shiftR) import Data.Word (Word16) import Data.Monoid (mappend) import Control.Monad.Trans.Reader (ask, runReaderT) -import Control.Monad.Trans.Control (MonadBaseControl) +import Control.Monad.IO.Unlift (MonadUnliftIO, withRunInIO) import Numeric (readHex) import Unsafe.Coerce (unsafeCoerce) @@ -362,17 +362,21 @@ withMongoDBPool dbname hostname port mauth poolStripes stripeConnections connect connectionReader pool -- | run a pool created with 'createMongoDBPipePool' -runMongoDBPipePool :: (Trans.MonadIO m, MonadBaseControl IO m) => DB.AccessMode -> Database -> DB.Action m a -> PipePool -> m a +runMongoDBPipePool :: MonadUnliftIO m => DB.AccessMode -> Database -> DB.Action m a -> PipePool -> m a runMongoDBPipePool accessMode db action pool = - Pool.withResource pool $ \pipe -> DB.access pipe accessMode db action + withRunInIO $ \run -> + Pool.withResource pool $ \pipe -> + run $ DB.access pipe accessMode db action -runMongoDBPool :: (Trans.MonadIO m, MonadBaseControl IO m) => DB.AccessMode -> DB.Action m a -> ConnectionPool -> m a +runMongoDBPool :: MonadUnliftIO m => DB.AccessMode -> DB.Action m a -> ConnectionPool -> m a runMongoDBPool accessMode action pool = - Pool.withResource pool $ \(Connection pipe db) -> DB.access pipe accessMode db action + withRunInIO $ \run -> + Pool.withResource pool $ \(Connection pipe db) -> + run $ DB.access pipe accessMode db action -- | use default 'AccessMode' -runMongoDBPoolDef :: (Trans.MonadIO m, MonadBaseControl IO m) => DB.Action m a -> ConnectionPool -> m a +runMongoDBPoolDef :: MonadUnliftIO m => DB.Action m a -> ConnectionPool -> m a runMongoDBPoolDef = runMongoDBPool defaultAccessMode queryByKey :: (PersistEntity record, PersistEntityBackend record ~ DB.MongoContext) diff --git a/persistent-mongoDB/persistent-mongoDB.cabal b/persistent-mongoDB/persistent-mongoDB.cabal index 8740ca964..869b30b45 100644 --- a/persistent-mongoDB/persistent-mongoDB.cabal +++ b/persistent-mongoDB/persistent-mongoDB.cabal @@ -33,12 +33,12 @@ library , cereal >= 0.3.0.0 , path-pieces >= 0.1 , http-api-data >= 0.2 && < 0.4 - , monad-control >= 0.3 , aeson >= 0.6.2 , attoparsec , time , bytestring , resource-pool < 0.3 + , unliftio-core exposed-modules: Database.Persist.MongoDB ghc-options: -Wall diff --git a/persistent-mysql/Database/Persist/MySQL.hs b/persistent-mysql/Database/Persist/MySQL.hs index 3cf1aeed3..761dc9211 100644 --- a/persistent-mysql/Database/Persist/MySQL.hs +++ b/persistent-mysql/Database/Persist/MySQL.hs @@ -67,7 +67,7 @@ import qualified Database.MySQL.Simple.Types as MySQL import qualified Database.MySQL.Base as MySQLBase import qualified Database.MySQL.Base.Types as MySQLBase -import Control.Monad.Trans.Control (MonadBaseControl) +import Control.Monad.IO.Unlift (MonadUnliftIO) import Control.Monad.Trans.Resource (runResourceT) import Prelude @@ -76,7 +76,7 @@ import Prelude -- The pool is properly released after the action finishes using -- it. Note that you should not use the given 'ConnectionPool' -- outside the action since it may be already been released. -withMySQLPool :: (MonadIO m, MonadLogger m, MonadBaseControl IO m, IsSqlBackend backend) +withMySQLPool :: (MonadLogger m, MonadUnliftIO m, IsSqlBackend backend) => MySQL.ConnectInfo -- ^ Connection information. -> Int @@ -90,7 +90,7 @@ withMySQLPool ci = withSqlPool $ open' ci -- | Create a MySQL connection pool. Note that it's your -- responsibility to properly close the connection pool when -- unneeded. Use 'withMySQLPool' for automatic resource control. -createMySQLPool :: (MonadBaseControl IO m, MonadIO m, MonadLogger m, IsSqlBackend backend) +createMySQLPool :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => MySQL.ConnectInfo -- ^ Connection information. -> Int @@ -101,7 +101,7 @@ createMySQLPool ci = createSqlPool $ open' ci -- | Same as 'withMySQLPool', but instead of opening a pool -- of connections, only one connection is opened. -withMySQLConn :: (MonadBaseControl IO m, MonadIO m, MonadLogger m, IsSqlBackend backend) +withMySQLConn :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => MySQL.ConnectInfo -- ^ Connection information. -> (backend -> m a) @@ -570,7 +570,7 @@ getColumn connectInfo getter tname [ PersistText cname , PersistText $ unDBName $ tname , PersistText cname , PersistText $ pack $ MySQL.connectDatabase connectInfo ] - cntrs <- with (stmtQuery stmt vars) ($$ CL.consume) + cntrs <- liftIO $ with (stmtQuery stmt vars) ($$ CL.consume) ref <- case cntrs of [] -> return Nothing [[PersistText tab, PersistText ref, PersistInt64 pos]] -> diff --git a/persistent-mysql/persistent-mysql.cabal b/persistent-mysql/persistent-mysql.cabal index ef41a1b8b..d5cfd6b28 100644 --- a/persistent-mysql/persistent-mysql.cabal +++ b/persistent-mysql/persistent-mysql.cabal @@ -36,7 +36,7 @@ library , containers >= 0.2 , bytestring >= 0.9 , text >= 0.11.0.6 - , monad-control >= 0.2 + , unliftio-core , aeson >= 0.6.2 , conduit >= 0.5.3 , resourcet >= 0.4.10 diff --git a/persistent-postgresql/Database/Persist/Postgresql.hs b/persistent-postgresql/Database/Persist/Postgresql.hs index 933188f8c..1aaa5f50f 100644 --- a/persistent-postgresql/Database/Persist/Postgresql.hs +++ b/persistent-postgresql/Database/Persist/Postgresql.hs @@ -105,7 +105,7 @@ instance Exception PostgresServerVersionError -- finishes using it. Note that you should not use the given -- 'ConnectionPool' outside the action since it may be already -- been released. -withPostgresqlPool :: (MonadBaseControl IO m, MonadLogger m, MonadIO m, IsSqlBackend backend) +withPostgresqlPool :: (MonadLogger m, MonadUnliftIO m, IsSqlBackend backend) => ConnectionString -- ^ Connection string to the database. -> Int @@ -121,7 +121,7 @@ withPostgresqlPool ci = withPostgresqlPoolWithVersion getServerVersion ci -- the server version (to workaround an Amazon Redshift bug). -- -- @since 2.6.2 -withPostgresqlPoolWithVersion :: (MonadBaseControl IO m, MonadLogger m, MonadIO m, IsSqlBackend backend) +withPostgresqlPoolWithVersion :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => (PG.Connection -> IO (Maybe Double)) -- ^ action to perform to get the server version -> ConnectionString @@ -139,7 +139,7 @@ withPostgresqlPoolWithVersion getVer ci = withSqlPool $ open' (const $ return () -- responsibility to properly close the connection pool when -- unneeded. Use 'withPostgresqlPool' for an automatic resource -- control. -createPostgresqlPool :: (MonadIO m, MonadBaseControl IO m, MonadLogger m, IsSqlBackend backend) +createPostgresqlPool :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => ConnectionString -- ^ Connection string to the database. -> Int @@ -157,7 +157,7 @@ createPostgresqlPool = createPostgresqlPoolModified (const $ return ()) -- -- @since 2.1.3 createPostgresqlPoolModified - :: (MonadIO m, MonadBaseControl IO m, MonadLogger m, IsSqlBackend backend) + :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => (PG.Connection -> IO ()) -- ^ action to perform after connection is created -> ConnectionString -- ^ Connection string to the database. -> Int -- ^ Number of connections to be kept open in the pool. @@ -170,7 +170,7 @@ createPostgresqlPoolModified = createPostgresqlPoolModifiedWithVersion getServer -- -- @since 2.6.2 createPostgresqlPoolModifiedWithVersion - :: (MonadIO m, MonadBaseControl IO m, MonadLogger m, IsSqlBackend backend) + :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => (PG.Connection -> IO (Maybe Double)) -- ^ action to perform to get the server version -> (PG.Connection -> IO ()) -- ^ action to perform after connection is created -> ConnectionString -- ^ Connection string to the database. @@ -181,7 +181,7 @@ createPostgresqlPoolModifiedWithVersion getVer modConn ci = -- | Same as 'withPostgresqlPool', but instead of opening a pool -- of connections, only one connection is opened. -withPostgresqlConn :: (MonadIO m, MonadBaseControl IO m, MonadLogger m, IsSqlBackend backend) +withPostgresqlConn :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => ConnectionString -> (backend -> m a) -> m a withPostgresqlConn = withPostgresqlConnWithVersion getServerVersion @@ -189,13 +189,13 @@ withPostgresqlConn = withPostgresqlConnWithVersion getServerVersion -- the server version (to workaround an Amazon Redshift bug). -- -- @since 2.6.2 -withPostgresqlConnWithVersion :: (MonadIO m, MonadBaseControl IO m, MonadLogger m, IsSqlBackend backend) +withPostgresqlConnWithVersion :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => (PG.Connection -> IO (Maybe Double)) - -> ConnectionString + -> ConnectionString -> (backend -> m a) -> m a withPostgresqlConnWithVersion getVer = withSqlConn . open' (const $ return ()) getVer - + open' :: (IsSqlBackend backend) => (PG.Connection -> IO ()) diff --git a/persistent-postgresql/persistent-postgresql.cabal b/persistent-postgresql/persistent-postgresql.cabal index a94884bf1..43a957d53 100644 --- a/persistent-postgresql/persistent-postgresql.cabal +++ b/persistent-postgresql/persistent-postgresql.cabal @@ -23,7 +23,7 @@ library , containers >= 0.2 , bytestring >= 0.9 , text >= 0.7 - , monad-control >= 0.2 + , unliftio-core , blaze-builder , time >= 1.1 , aeson >= 0.6.2 diff --git a/persistent-redis/persistent-redis.cabal b/persistent-redis/persistent-redis.cabal index 638961cae..b43618804 100644 --- a/persistent-redis/persistent-redis.cabal +++ b/persistent-redis/persistent-redis.cabal @@ -59,7 +59,7 @@ test-suite basic , text >= 1.2.0.0 , aeson >= 0.8 , binary >= 0.7 && < 0.9 - , time >= 1.4 && < 1.7 + , time >= 1.4 && < 1.9 , attoparsec >= 0.12.0.0 , template-haskell , monad-control >= 0.3.2.0 && < 1.2.0.0 diff --git a/persistent-sqlite/Database/Persist/Sqlite.hs b/persistent-sqlite/Database/Persist/Sqlite.hs index 89afa4db4..9b2f92657 100644 --- a/persistent-sqlite/Database/Persist/Sqlite.hs +++ b/persistent-sqlite/Database/Persist/Sqlite.hs @@ -39,10 +39,8 @@ import qualified Database.Sqlite as Sqlite import Control.Applicative as A import qualified Control.Exception as E import Control.Monad (when) -import Control.Monad.IO.Class (MonadIO (..)) +import Control.Monad.IO.Unlift (MonadIO (..), MonadUnliftIO, withUnliftIO, unliftIO) import Control.Monad.Logger (NoLoggingT, runNoLoggingT, MonadLogger) -import Control.Monad.Trans.Control (control) -import Control.Monad.Trans.Control (MonadBaseControl) import Control.Monad.Trans.Reader (ReaderT, runReaderT) import Control.Monad.Trans.Resource (ResourceT, runResourceT) import Control.Monad.Trans.Writer (runWriterT) @@ -67,7 +65,7 @@ import Lens.Micro.TH (makeLenses) -- Note that this should not be used with the @:memory:@ connection string, as -- the pool will regularly remove connections, destroying your database. -- Instead, use 'withSqliteConn'. -createSqlitePool :: (MonadIO m, MonadLogger m, MonadBaseControl IO m, IsSqlBackend backend) +createSqlitePool :: (MonadLogger m, MonadUnliftIO m, IsSqlBackend backend) => Text -> Int -> m (Pool backend) createSqlitePool = createSqlitePoolFromInfo . conStringToInfo @@ -78,14 +76,14 @@ createSqlitePool = createSqlitePoolFromInfo . conStringToInfo -- Instead, use 'withSqliteConn'. -- -- @since 2.6.2 -createSqlitePoolFromInfo :: (MonadIO m, MonadLogger m, MonadBaseControl IO m, IsSqlBackend backend) +createSqlitePoolFromInfo :: (MonadLogger m, MonadUnliftIO m, IsSqlBackend backend) => SqliteConnectionInfo -> Int -> m (Pool backend) createSqlitePoolFromInfo connInfo = createSqlPool $ open' connInfo -- | Run the given action with a connection pool. -- -- Like 'createSqlitePool', this should not be used with @:memory:@. -withSqlitePool :: (MonadBaseControl IO m, MonadIO m, MonadLogger m, IsSqlBackend backend) +withSqlitePool :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => Text -> Int -- ^ number of connections to open -> (Pool backend -> m a) -> m a @@ -96,18 +94,18 @@ withSqlitePool connInfo = withSqlPool . open' $ conStringToInfo connInfo -- Like 'createSqlitePool', this should not be used with @:memory:@. -- -- @since 2.6.2 -withSqlitePoolInfo :: (MonadBaseControl IO m, MonadIO m, MonadLogger m, IsSqlBackend backend) +withSqlitePoolInfo :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => SqliteConnectionInfo -> Int -- ^ number of connections to open -> (Pool backend -> m a) -> m a withSqlitePoolInfo connInfo = withSqlPool $ open' connInfo -withSqliteConn :: (MonadBaseControl IO m, MonadIO m, MonadLogger m, IsSqlBackend backend) +withSqliteConn :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => Text -> (backend -> m a) -> m a withSqliteConn = withSqliteConnInfo . conStringToInfo -- | @since 2.6.2 -withSqliteConnInfo :: (MonadBaseControl IO m, MonadIO m, MonadLogger m, IsSqlBackend backend) +withSqliteConnInfo :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => SqliteConnectionInfo -> (backend -> m a) -> m a withSqliteConnInfo = withSqlConn . open' @@ -179,7 +177,7 @@ wrapConnectionInfo connInfo conn logFunc = do -- that all log messages are discarded. -- -- @since 1.1.4 -runSqlite :: (MonadBaseControl IO m, MonadIO m, IsSqlBackend backend) +runSqlite :: (MonadUnliftIO m, IsSqlBackend backend) => Text -- ^ connection string -> ReaderT backend (NoLoggingT (ResourceT m)) a -- ^ database action -> m a @@ -193,7 +191,7 @@ runSqlite connstr = runResourceT -- that all log messages are discarded. -- -- @since 2.6.2 -runSqliteInfo :: (MonadBaseControl IO m, MonadIO m, IsSqlBackend backend) +runSqliteInfo :: (MonadUnliftIO m, IsSqlBackend backend) => SqliteConnectionInfo -> ReaderT backend (NoLoggingT (ResourceT m)) a -- ^ database action -> m a @@ -515,13 +513,13 @@ instance PersistConfig SqliteConf where runPool _ = runSqlPool loadConfig = parseJSON -finally :: MonadBaseControl IO m +finally :: MonadUnliftIO m => m a -- ^ computation to run first -> m b -- ^ computation to run afterward (even if an exception was raised) -> m a -finally a sequel = control $ \runInIO -> - E.finally (runInIO a) - (runInIO sequel) +finally a sequel = withUnliftIO $ \u -> + E.finally (unliftIO u a) + (unliftIO u sequel) {-# INLINABLE finally #-} -- | Creates a SqliteConnectionInfo from a connection string, with the -- default settings. diff --git a/persistent-sqlite/persistent-sqlite.cabal b/persistent-sqlite/persistent-sqlite.cabal index 50b6c50e5..68b6d4905 100644 --- a/persistent-sqlite/persistent-sqlite.cabal +++ b/persistent-sqlite/persistent-sqlite.cabal @@ -29,7 +29,7 @@ library , bytestring >= 0.9.1 , transformers >= 0.2.1 , persistent >= 2.6.1 && < 3 - , monad-control >= 0.2 + , unliftio-core , containers >= 0.2 , text >= 0.7 , aeson >= 0.6.2 diff --git a/persistent-test/persistent-test.cabal b/persistent-test/persistent-test.cabal index 3846aa09e..7612e8e55 100644 --- a/persistent-test/persistent-test.cabal +++ b/persistent-test/persistent-test.cabal @@ -101,7 +101,8 @@ library , http-api-data >= 0.2 , text >= 0.8 , transformers >= 0.2.1 - , monad-control >= 0.3 + , unliftio-core + , unliftio , containers >= 0.2 , bytestring >= 0.9 , base64-bytestring diff --git a/persistent-test/src/CompositeTest.hs b/persistent-test/src/CompositeTest.hs index 25a8d7cc7..3eec20571 100644 --- a/persistent-test/src/CompositeTest.hs +++ b/persistent-test/src/CompositeTest.hs @@ -19,8 +19,6 @@ module CompositeTest where import Test.Hspec.Expectations () -import qualified Control.Monad.Trans.Control -import qualified Control.Exception as E import Init #ifndef WITH_NOSQL @@ -311,11 +309,3 @@ matchParentK = (\(a:b:c:[]) -> (,,) <$> fromPersistValue a <*> fromPersistValue matchCitizenAddressK :: Key CitizenAddress -> Either Text (Int64, Int64) matchCitizenAddressK = (\(a:b:[]) -> (,) <$> fromPersistValue a <*> fromPersistValue b) . keyToValues - -catch' :: (Control.Monad.Trans.Control.MonadBaseControl IO m, E.Exception e) - => m a -- ^ The computation to run - -> (e -> m a) -- ^ Handler to invoke if an exception is raised - -> m a -catch' a handler = Control.Monad.Trans.Control.control $ \runInIO -> - E.catch (runInIO a) - (\e -> runInIO $ handler e) diff --git a/persistent-test/src/Init.hs b/persistent-test/src/Init.hs index 4d99fa9b5..d006702f8 100644 --- a/persistent-test/src/Init.hs +++ b/persistent-test/src/Init.hs @@ -110,7 +110,7 @@ import System.IO.Unsafe (unsafePerformIO) #endif import Control.Monad (unless, (>=>)) -import Control.Monad.Trans.Control (MonadBaseControl) +import Control.Monad.IO.Unlift (MonadUnliftIO) -- Data types import Data.Int (Int32, Int64) @@ -191,7 +191,7 @@ dbName = "persistent" type BackendMonad = Context #ifdef WITH_MONGODB -runConn :: (MonadIO m, MonadBaseControl IO m) => Action m backend -> m () +runConn :: MonadUnliftIO m => Action m backend -> m () runConn f = do conf <- liftIO $ applyDockerEnv $ defaultMongoConf dbName -- { mgRsPrimary = Just "replicaset" } void $ withMongoPool conf $ runMongoDBPool MongoDB.master f @@ -201,7 +201,7 @@ setupMongo = void $ MongoDB.dropDatabase dbName #endif #ifdef WITH_ZOOKEEPER -runConn :: (MonadIO m, MonadBaseControl IO m) => Action m backend -> m () +runConn :: MonadUnliftIO m => Action m backend -> m () runConn f = do let conf = defaultZookeeperConf {zCoord = "localhost:2181/" ++ T.unpack dbName} void $ withZookeeperPool conf $ runZookeeperPool f @@ -234,7 +234,7 @@ sqlite_database_file = error "Sqlite tests disabled" sqlite_database :: () sqlite_database = error "Sqlite tests disabled" # endif -runConn :: (MonadIO m, MonadBaseControl IO m) => SqlPersistT (LoggingT m) t -> m () +runConn :: MonadUnliftIO m => SqlPersistT (LoggingT m) t -> m () runConn f = do travis <- liftIO isTravis let debugPrint = not travis && _debugOn diff --git a/persistent-test/src/PersistentTest.hs b/persistent-test/src/PersistentTest.hs index cd5c61a55..5e2a406f6 100644 --- a/persistent-test/src/PersistentTest.hs +++ b/persistent-test/src/PersistentTest.hs @@ -21,6 +21,7 @@ import Test.HUnit hiding (Test) import Control.Monad.Trans.Resource (runResourceT) import Test.Hspec.Expectations () import Test.Hspec.QuickCheck(prop) +import UnliftIO (MonadUnliftIO, catch) import Database.Persist @@ -45,9 +46,6 @@ import Database.Persist.MySQL() #endif -import qualified Control.Monad.Trans.Control -import Control.Exception.Lifted (catch) - import Control.Monad.IO.Class import Web.PathPieces (PathPiece (..)) @@ -186,7 +184,7 @@ db :: Action IO () -> Assertion db = db' cleanDB #endif -catchPersistException :: Control.Monad.Trans.Control.MonadBaseControl IO m => m a -> b -> m b +catchPersistException :: MonadUnliftIO m => m a -> b -> m b catchPersistException action errValue = do Left res <- (Right `fmap` action) `catch` @@ -1189,14 +1187,6 @@ caseCommitRollback = db $ do c4 <- count filt c4 @== 4 -catch' :: (Control.Monad.Trans.Control.MonadBaseControl IO m, E.Exception e) - => m a -- ^ The computation to run - -> (e -> m a) -- ^ Handler to invoke if an exception is raised - -> m a -catch' a handler = Control.Monad.Trans.Control.control $ \runInIO -> - E.catch (runInIO a) - (\e -> runInIO $ handler e) - #endif -- Test proper polymorphism diff --git a/persistent/Database/Persist/Class/PersistConfig.hs b/persistent/Database/Persist/Class/PersistConfig.hs index ad2bf1bf5..f5a8b7a4d 100644 --- a/persistent/Database/Persist/Class/PersistConfig.hs +++ b/persistent/Database/Persist/Class/PersistConfig.hs @@ -7,8 +7,7 @@ module Database.Persist.Class.PersistConfig import Data.Aeson (Value (Object)) import Data.Aeson.Types (Parser) -import Control.Monad.IO.Class (MonadIO) -import Control.Monad.Trans.Control (MonadBaseControl) +import Control.Monad.IO.Unlift (MonadUnliftIO) import Control.Applicative as A ((<$>)) import qualified Data.HashMap.Strict as HashMap @@ -31,7 +30,7 @@ class PersistConfig c where createPoolConfig :: c -> IO (PersistConfigPool c) -- | Run a database action by taking a connection from the pool. - runPool :: (MonadBaseControl IO m, MonadIO m) + runPool :: MonadUnliftIO m => c -> PersistConfigBackend c m a -> PersistConfigPool c diff --git a/persistent/Database/Persist/Class/PersistStore.hs b/persistent/Database/Persist/Class/PersistStore.hs index 8fc8471e6..30d4bcc66 100644 --- a/persistent/Database/Persist/Class/PersistStore.hs +++ b/persistent/Database/Persist/Class/PersistStore.hs @@ -23,7 +23,7 @@ module Database.Persist.Class.PersistStore import qualified Data.Text as T import Control.Monad.IO.Class (MonadIO, liftIO) -import Control.Exception.Lifted (throwIO) +import Control.Exception (throwIO) import Control.Monad.Trans.Reader (ReaderT) import Control.Monad.Reader (MonadReader (ask), runReaderT) import Database.Persist.Class.PersistEntity diff --git a/persistent/Database/Persist/Sql/Migration.hs b/persistent/Database/Persist/Sql/Migration.hs index 16287341d..38890395e 100644 --- a/persistent/Database/Persist/Sql/Migration.hs +++ b/persistent/Database/Persist/Sql/Migration.hs @@ -13,9 +13,8 @@ module Database.Persist.Sql.Migration ) where -import Control.Monad.Trans.Control (MonadBaseControl) import Control.Monad.Trans.Class (MonadTrans (..)) -import Control.Monad.IO.Class +import Control.Monad.IO.Unlift import Control.Monad.Trans.Writer import Control.Monad.Trans.Reader (ReaderT (..), ask) import Control.Monad (liftM, unless) @@ -23,7 +22,6 @@ import Data.Text (Text, unpack, snoc, isPrefixOf, pack) import qualified Data.Text.IO import System.IO import System.IO.Silently (hSilence) -import Control.Monad.Trans.Control (liftBaseOp_) import Database.Persist.Sql.Types import Database.Persist.Sql.Raw import Database.Persist.Types @@ -79,10 +77,11 @@ runMigration m = runMigration' m False >> return () -- | Same as 'runMigration', but returns a list of the SQL commands executed -- instead of printing them to stderr. -runMigrationSilent :: (MonadBaseControl IO m, MonadIO m) +runMigrationSilent :: (MonadUnliftIO m, MonadIO m) => Migration -> ReaderT SqlBackend m [Text] -runMigrationSilent m = liftBaseOp_ (hSilence [stderr]) $ runMigration' m True +runMigrationSilent m = withRunInIO $ \run -> + hSilence [stderr] $ run $ runMigration' m True -- | Run the given migration against the database. If the migration fails -- to parse, or there are any unsafe migrations, then this will error at diff --git a/persistent/Database/Persist/Sql/Orphan/PersistStore.hs b/persistent/Database/Persist/Sql/Orphan/PersistStore.hs index f666f7930..1e1e73393 100644 --- a/persistent/Database/Persist/Sql/Orphan/PersistStore.hs +++ b/persistent/Database/Persist/Sql/Orphan/PersistStore.hs @@ -39,7 +39,7 @@ import Web.PathPieces (PathPiece) import Web.HttpApiData (ToHttpApiData, FromHttpApiData) import Database.Persist.Sql.Class (PersistFieldSql) import qualified Data.Aeson as A -import Control.Exception.Lifted (throwIO) +import Control.Exception (throwIO) import Database.Persist.Class () withRawQuery :: MonadIO m diff --git a/persistent/Database/Persist/Sql/Run.hs b/persistent/Database/Persist/Sql/Run.hs index bd63ec251..3cd4924c9 100644 --- a/persistent/Database/Persist/Sql/Run.hs +++ b/persistent/Database/Persist/Sql/Run.hs @@ -7,14 +7,12 @@ module Database.Persist.Sql.Run where import Database.Persist.Class.PersistStore import Database.Persist.Sql.Types import Database.Persist.Sql.Raw -import Control.Monad.Trans.Control import Data.Pool as P import Control.Monad.Trans.Reader hiding (local) import Control.Monad.Trans.Resource import Control.Monad.Logger -import Control.Monad.Base -import Control.Exception.Lifted (onException, bracket) -import Control.Monad.IO.Class +import Control.Exception (onException, bracket) +import Control.Monad.IO.Unlift import Control.Exception (mask) import System.Timeout (timeout) import Data.IORef (readIORef, writeIORef, newIORef) @@ -28,22 +26,22 @@ import Control.Monad (liftM) -- was buggy and caused more problems than it solved. Since version 2.1.2, it -- performs no timeout checks. runSqlPool - :: (MonadBaseControl IO m, IsSqlBackend backend) + :: (MonadUnliftIO m, IsSqlBackend backend) => ReaderT backend m a -> Pool backend -> m a -runSqlPool r pconn = withResource pconn $ runSqlConn r +runSqlPool r pconn = withRunInIO $ \run -> withResource pconn $ run . runSqlConn r -- | Like 'withResource', but times out the operation if resource -- allocation does not complete within the given timeout period. -- -- @since 2.0.0 withResourceTimeout - :: forall a m b. (MonadBaseControl IO m) + :: forall a m b. (MonadUnliftIO m) => Int -- ^ Timeout period in microseconds -> Pool a -> (a -> m b) -> m (Maybe b) {-# SPECIALIZE withResourceTimeout :: Int -> Pool a -> (a -> IO b) -> IO (Maybe b) #-} -withResourceTimeout ms pool act = control $ \runInIO -> mask $ \restore -> do +withResourceTimeout ms pool act = withRunInIO $ \runInIO -> mask $ \restore -> do mres <- timeout ms $ takeResource pool case mres of Nothing -> runInIO $ return (Nothing :: Maybe b) @@ -54,8 +52,8 @@ withResourceTimeout ms pool act = control $ \runInIO -> mask $ \restore -> do return ret {-# INLINABLE withResourceTimeout #-} -runSqlConn :: (MonadBaseControl IO m, IsSqlBackend backend) => ReaderT backend m a -> backend -> m a -runSqlConn r conn = control $ \runInIO -> mask $ \restore -> do +runSqlConn :: (MonadUnliftIO m, IsSqlBackend backend) => ReaderT backend m a -> backend -> m a +runSqlConn r conn = withRunInIO $ \runInIO -> mask $ \restore -> do let conn' = persistBackend conn getter = getStmtConn conn' restore $ connBegin conn' getter @@ -81,16 +79,18 @@ liftSqlPersistMPool liftSqlPersistMPool x pool = liftIO (runSqlPersistMPool x pool) withSqlPool - :: (MonadIO m, MonadLogger m, MonadBaseControl IO m, IsSqlBackend backend) + :: (MonadLogger m, MonadUnliftIO m, IsSqlBackend backend) => (LogFunc -> IO backend) -- ^ create a new connection -> Int -- ^ connection count -> (Pool backend -> m a) -> m a -withSqlPool mkConn connCount f = - bracket (createSqlPool mkConn connCount) (liftIO . destroyAllResources) f +withSqlPool mkConn connCount f = withUnliftIO $ \u -> bracket + (unliftIO u $ createSqlPool mkConn connCount) + destroyAllResources + (unliftIO u . f) createSqlPool - :: (MonadIO m, MonadLogger m, MonadBaseControl IO m, IsSqlBackend backend) + :: (MonadLogger m, MonadUnliftIO m, IsSqlBackend backend) => (LogFunc -> IO backend) -> Int -> m (Pool backend) @@ -104,21 +104,19 @@ createSqlPool mkConn size = do -- FIXME: in a future release, switch over to the new askLoggerIO function -- added in monad-logger 0.3.10. That function was not available at the time -- this code was written. -askLogFunc :: forall m. (MonadBaseControl IO m, MonadLogger m) => m LogFunc -askLogFunc = do - ref <- liftBase $ newIORef undefined - liftBaseWith $ \run -> writeIORef ref run - runInBase <- liftBase $ readIORef ref - return $ \a b c d -> do - _ <- runInBase (monadLoggerLog a b c d) - return () +askLogFunc :: forall m. (MonadUnliftIO m, MonadLogger m) => m LogFunc +askLogFunc = withRunInIO $ \run -> + return $ \a b c d -> run (monadLoggerLog a b c d) withSqlConn - :: (MonadIO m, MonadBaseControl IO m, MonadLogger m, IsSqlBackend backend) + :: (MonadUnliftIO m, MonadLogger m, IsSqlBackend backend) => (LogFunc -> IO backend) -> (backend -> m a) -> m a withSqlConn open f = do logFunc <- askLogFunc - bracket (liftIO $ open logFunc) (liftIO . close') f + withRunInIO $ \run -> bracket + (open logFunc) + close' + (run . f) close' :: (IsSqlBackend backend) => backend -> IO () close' conn = do diff --git a/persistent/persistent.cabal b/persistent/persistent.cabal index 3d284dd6a..7b8541387 100644 --- a/persistent/persistent.cabal +++ b/persistent/persistent.cabal @@ -31,15 +31,11 @@ library , containers >= 0.2 , conduit >= 1.0 , resourcet >= 1.1 - , exceptions >= 0.6 - , monad-control >= 0.3 - , lifted-base >= 0.1 , resource-pool >= 0.2.2.0 , path-pieces >= 0.1 , http-api-data >= 0.2 && < 0.4 , aeson >= 0.5 , monad-logger >= 0.3 - , transformers-base , base64-bytestring , unordered-containers , vector @@ -53,6 +49,7 @@ library , fast-logger >= 2.1 , scientific , tagged + , unliftio-core exposed-modules: Database.Persist Database.Persist.Quasi @@ -112,7 +109,6 @@ test-suite test , scientific , tagged , fast-logger >= 2.1 - , lifted-base >= 0.1 , mtl , template-haskell , resource-pool From 9e88459462dddbff5318193bb38816a384da5ff3 Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Mon, 15 Jan 2018 15:09:19 +0200 Subject: [PATCH 08/20] Cleanup warnings --- persistent-mysql/Database/Persist/MySQL.hs | 17 ++++++++--------- persistent-mysql/persistent-mysql.cabal | 2 +- .../Database/Persist/Postgresql.hs | 10 +++++----- .../persistent-postgresql.cabal | 2 +- persistent-sqlite/Database/Persist/Sqlite.hs | 7 ++++--- persistent-sqlite/persistent-sqlite.cabal | 2 +- .../Database/Persist/Class/DeleteCascade.hs | 4 ++-- .../Database/Persist/Class/PersistQuery.hs | 16 ++++++++-------- .../Database/Persist/Sql/Orphan/PersistQuery.hs | 4 ++-- .../Database/Persist/Sql/Orphan/PersistStore.hs | 6 +++--- persistent/Database/Persist/Sql/Raw.hs | 6 +++--- persistent/Database/Persist/Sql/Run.hs | 2 +- .../Database/Persist/Sql/Types/Internal.hs | 4 ++-- persistent/persistent.cabal | 2 +- 14 files changed, 42 insertions(+), 42 deletions(-) diff --git a/persistent-mysql/Database/Persist/MySQL.hs b/persistent-mysql/Database/Persist/MySQL.hs index 761dc9211..c4f60e515 100644 --- a/persistent-mysql/Database/Persist/MySQL.hs +++ b/persistent-mysql/Database/Persist/MySQL.hs @@ -68,7 +68,6 @@ import qualified Database.MySQL.Simple.Types as MySQL import qualified Database.MySQL.Base as MySQLBase import qualified Database.MySQL.Base.Types as MySQLBase import Control.Monad.IO.Unlift (MonadUnliftIO) -import Control.Monad.Trans.Resource (runResourceT) import Prelude @@ -178,7 +177,7 @@ withStmt' :: MonadIO m => MySQL.Connection -> MySQL.Query -> [PersistValue] - -> Acquire (Source m [PersistValue]) + -> Acquire (ConduitM () [PersistValue] m ()) withStmt' conn query vals = do result <- mkAcquire createResult MySQLBase.freeResult return $ fetchRows result >>= CL.sourceList @@ -473,8 +472,8 @@ getColumns connectInfo getter def = do \WHERE TABLE_SCHEMA = ? \ \AND TABLE_NAME = ? \ \AND COLUMN_NAME = ?" - inter1 <- with (stmtQuery stmtIdClmn vals) ($$ CL.consume) - ids <- runResourceT $ CL.sourceList inter1 $$ helperClmns -- avoid nested queries + inter1 <- with (stmtQuery stmtIdClmn vals) (\src -> runConduit $ src .| CL.consume) + ids <- runConduitRes $ CL.sourceList inter1 .| helperClmns -- avoid nested queries -- Find out all columns. stmtClmns <- getter "SELECT COLUMN_NAME, \ @@ -489,8 +488,8 @@ getColumns connectInfo getter def = do \WHERE TABLE_SCHEMA = ? \ \AND TABLE_NAME = ? \ \AND COLUMN_NAME <> ?" - inter2 <- with (stmtQuery stmtClmns vals) ($$ CL.consume) - cs <- runResourceT $ CL.sourceList inter2 $$ helperClmns -- avoid nested queries + inter2 <- with (stmtQuery stmtClmns vals) (\src -> runConduit $ src .| CL.consume) + cs <- runConduitRes $ CL.sourceList inter2 .| helperClmns -- avoid nested queries -- Find out the constraints. stmtCntrs <- getter "SELECT CONSTRAINT_NAME, \ @@ -503,7 +502,7 @@ getColumns connectInfo getter def = do \AND REFERENCED_TABLE_SCHEMA IS NULL \ \ORDER BY CONSTRAINT_NAME, \ \COLUMN_NAME" - us <- with (stmtQuery stmtCntrs vals) ($$ helperCntrs) + us <- with (stmtQuery stmtCntrs vals) (\src -> runConduit $ src .| helperCntrs) -- Return both return (ids, cs ++ us) @@ -512,7 +511,7 @@ getColumns connectInfo getter def = do , PersistText $ unDBName $ entityDB def , PersistText $ unDBName $ fieldDB $ entityId def ] - helperClmns = CL.mapM getIt =$ CL.consume + helperClmns = CL.mapM getIt .| CL.consume where getIt = fmap (either Left (Right . Left)) . liftIO . @@ -570,7 +569,7 @@ getColumn connectInfo getter tname [ PersistText cname , PersistText $ unDBName $ tname , PersistText cname , PersistText $ pack $ MySQL.connectDatabase connectInfo ] - cntrs <- liftIO $ with (stmtQuery stmt vars) ($$ CL.consume) + cntrs <- liftIO $ with (stmtQuery stmt vars) (\src -> runConduit $ src .| CL.consume) ref <- case cntrs of [] -> return Nothing [[PersistText tab, PersistText ref, PersistInt64 pos]] -> diff --git a/persistent-mysql/persistent-mysql.cabal b/persistent-mysql/persistent-mysql.cabal index d5cfd6b28..463d1d29e 100644 --- a/persistent-mysql/persistent-mysql.cabal +++ b/persistent-mysql/persistent-mysql.cabal @@ -38,7 +38,7 @@ library , text >= 0.11.0.6 , unliftio-core , aeson >= 0.6.2 - , conduit >= 0.5.3 + , conduit >= 1.2.8 , resourcet >= 0.4.10 , monad-logger , resource-pool diff --git a/persistent-postgresql/Database/Persist/Postgresql.hs b/persistent-postgresql/Database/Persist/Postgresql.hs index 1aaa5f50f..ede81efde 100644 --- a/persistent-postgresql/Database/Persist/Postgresql.hs +++ b/persistent-postgresql/Database/Persist/Postgresql.hs @@ -338,7 +338,7 @@ withStmt' :: MonadIO m => PG.Connection -> PG.Query -> [PersistValue] - -> Acquire (Source m [PersistValue]) + -> Acquire (ConduitM () [PersistValue] m ()) withStmt' conn query vals = pull `fmap` mkAcquire openS closeS where @@ -528,7 +528,7 @@ doesTableExist :: (Text -> IO Statement) -> IO Bool doesTableExist getter (DBName name) = do stmt <- getter sql - with (stmtQuery stmt vals) ($$ start) + with (stmtQuery stmt vals) (\src -> runConduit $ src .| start) where sql = "SELECT COUNT(*) FROM pg_catalog.pg_tables WHERE schemaname != 'pg_catalog'" <> " AND schemaname != 'information_schema' AND tablename=?" @@ -656,7 +656,7 @@ getColumns getter def = do [ PersistText $ unDBName $ entityDB def , PersistText $ unDBName $ fieldDB (entityId def) ] - cs <- with (stmtQuery stmt vals) ($$ helper) + cs <- with (stmtQuery stmt vals) (\src -> runConduit $ src .| helper) let sqlc = T.concat ["SELECT " ,"c.constraint_name, " ,"c.column_name " @@ -675,7 +675,7 @@ getColumns getter def = do stmt' <- getter sqlc - us <- with (stmtQuery stmt' vals) ($$ helperU) + us <- with (stmtQuery stmt' vals) (\src -> runConduit $ src .| helperU) return $ cs ++ us where getAll front = do @@ -791,7 +791,7 @@ getColumn getter tname [PersistText x, PersistText y, PersistText z, d, npre, ns with (stmtQuery stmt [ PersistText $ unDBName tname , PersistText $ unDBName ref - ]) ($$ do + ]) (\src -> runConduit $ src .| do Just [PersistInt64 i] <- CL.head return $ if i == 0 then Nothing else Just (DBName "", ref)) d' = case d of diff --git a/persistent-postgresql/persistent-postgresql.cabal b/persistent-postgresql/persistent-postgresql.cabal index 43a957d53..d9b7536ea 100644 --- a/persistent-postgresql/persistent-postgresql.cabal +++ b/persistent-postgresql/persistent-postgresql.cabal @@ -27,7 +27,7 @@ library , blaze-builder , time >= 1.1 , aeson >= 0.6.2 - , conduit >= 0.5.3 + , conduit >= 1.2.8 , resourcet >= 1.1 , monad-logger >= 0.3.4 , resource-pool diff --git a/persistent-sqlite/Database/Persist/Sqlite.hs b/persistent-sqlite/Database/Persist/Sqlite.hs index 9b2f92657..18c1b3bd3 100644 --- a/persistent-sqlite/Database/Persist/Sqlite.hs +++ b/persistent-sqlite/Database/Persist/Sqlite.hs @@ -259,7 +259,7 @@ withStmt' => Sqlite.Connection -> Sqlite.Statement -> [PersistValue] - -> Acquire (Source m [PersistValue]) + -> Acquire (ConduitM () [PersistValue] m ()) withStmt' conn stmt vals = do _ <- mkAcquire (Sqlite.bind stmt vals >> return stmt) @@ -296,7 +296,8 @@ migrate' allDefs getter val = do let (cols, uniqs, _) = mkColumns allDefs val let newSql = mkCreateTable False def (filter (not . safeToRemove val . cName) cols, uniqs) stmt <- getter "SELECT sql FROM sqlite_master WHERE type='table' AND name=?" - oldSql' <- with (stmtQuery stmt [PersistText $ unDBName table]) ($$ go) + oldSql' <- with (stmtQuery stmt [PersistText $ unDBName table]) + (\src -> runConduit $ src .| go) case oldSql' of Nothing -> return $ Right [(False, newSql)] Just oldSql -> do @@ -369,7 +370,7 @@ getCopyTable :: [EntityDef] -> IO [(Bool, Text)] getCopyTable allDefs getter def = do stmt <- getter $ T.concat [ "PRAGMA table_info(", escape table, ")" ] - oldCols' <- with (stmtQuery stmt []) ($$ getCols) + oldCols' <- with (stmtQuery stmt []) (\src -> runConduit $ src .| getCols) let oldCols = map DBName $ filter (/= "id") oldCols' -- need to update for table id attribute ? let newCols = filter (not . safeToRemove def) $ map cName cols let common = filter (`elem` oldCols) newCols diff --git a/persistent-sqlite/persistent-sqlite.cabal b/persistent-sqlite/persistent-sqlite.cabal index 68b6d4905..feda9f686 100644 --- a/persistent-sqlite/persistent-sqlite.cabal +++ b/persistent-sqlite/persistent-sqlite.cabal @@ -33,7 +33,7 @@ library , containers >= 0.2 , text >= 0.7 , aeson >= 0.6.2 - , conduit >= 0.5.3 + , conduit >= 1.2.8 , monad-logger >= 0.2.4 , microlens-th >= 0.4.1.1 , resourcet >= 1.1 diff --git a/persistent/Database/Persist/Class/DeleteCascade.hs b/persistent/Database/Persist/Class/DeleteCascade.hs index c40dfb0b3..82690ffdf 100644 --- a/persistent/Database/Persist/Class/DeleteCascade.hs +++ b/persistent/Database/Persist/Class/DeleteCascade.hs @@ -9,7 +9,7 @@ import Database.Persist.Class.PersistStore import Database.Persist.Class.PersistQuery import Database.Persist.Class.PersistEntity -import qualified Data.Conduit as C +import Data.Conduit import qualified Data.Conduit.List as CL import Control.Monad.IO.Class (MonadIO, liftIO) import Control.Monad.Reader (ReaderT, ask, runReaderT) @@ -31,4 +31,4 @@ deleteCascadeWhere :: (MonadIO m, DeleteCascade record backend, PersistQueryWrit deleteCascadeWhere filts = do srcRes <- selectKeysRes filts [] conn <- ask - liftIO $ with srcRes (C.$$ CL.mapM_ (flip runReaderT conn . deleteCascade)) + liftIO $ with srcRes (\src -> runConduit $ src .| CL.mapM_ (flip runReaderT conn . deleteCascade)) diff --git a/persistent/Database/Persist/Class/PersistQuery.hs b/persistent/Database/Persist/Class/PersistQuery.hs index 322841d08..96a99d6bd 100644 --- a/persistent/Database/Persist/Class/PersistQuery.hs +++ b/persistent/Database/Persist/Class/PersistQuery.hs @@ -15,7 +15,7 @@ import Database.Persist.Types import Control.Monad.IO.Class (MonadIO, liftIO) import Control.Monad.Reader (ReaderT, MonadReader) -import qualified Data.Conduit as C +import Data.Conduit (ConduitM, (.|), await, runConduit) import qualified Data.Conduit.List as CL import Database.Persist.Class.PersistStore import Database.Persist.Class.PersistEntity @@ -30,7 +30,7 @@ class (PersistCore backend, PersistStoreRead backend) => PersistQueryRead backen :: (PersistRecordBackend record backend, MonadIO m1, MonadIO m2) => [Filter record] -> [SelectOpt record] - -> ReaderT backend m1 (Acquire (C.Source m2 (Entity record))) + -> ReaderT backend m1 (Acquire (ConduitM () (Entity record) m2 ())) -- | Get just the first record for the criterion. selectFirst :: (MonadIO m, PersistRecordBackend record backend) @@ -39,14 +39,14 @@ class (PersistCore backend, PersistStoreRead backend) => PersistQueryRead backen -> ReaderT backend m (Maybe (Entity record)) selectFirst filts opts = do srcRes <- selectSourceRes filts (LimitTo 1 : opts) - liftIO $ with srcRes (C.$$ CL.head) + liftIO $ with srcRes (\src -> runConduit $ src .| await) -- | Get the 'Key's of all records matching the given criterion. selectKeysRes :: (MonadIO m1, MonadIO m2, PersistRecordBackend record backend) => [Filter record] -> [SelectOpt record] - -> ReaderT backend m1 (Acquire (C.Source m2 (Key record))) + -> ReaderT backend m1 (Acquire (ConduitM () (Key record) m2 ())) -- | The total number of records fulfilling the given criterion. count :: (MonadIO m, PersistRecordBackend record backend) @@ -68,7 +68,7 @@ selectSource :: (PersistQueryRead (BaseBackend backend), MonadResource m, PersistEntity record, PersistEntityBackend record ~ BaseBackend (BaseBackend backend), MonadReader backend m, HasPersistBackend backend) => [Filter record] -> [SelectOpt record] - -> C.Source m (Entity record) + -> ConduitM () (Entity record) m () selectSource filts opts = do srcRes <- liftPersist $ selectSourceRes filts opts (releaseKey, src) <- allocateAcquire srcRes @@ -79,7 +79,7 @@ selectSource filts opts = do selectKeys :: (PersistQueryRead (BaseBackend backend), MonadResource m, PersistEntity record, BaseBackend (BaseBackend backend) ~ PersistEntityBackend record, MonadReader backend m, HasPersistBackend backend) => [Filter record] -> [SelectOpt record] - -> C.Source m (Key record) + -> ConduitM () (Key record) m () selectKeys filts opts = do srcRes <- liftPersist $ selectKeysRes filts opts (releaseKey, src) <- allocateAcquire srcRes @@ -93,7 +93,7 @@ selectList :: (MonadIO m, PersistQueryRead backend, PersistRecordBackend record -> ReaderT backend m [Entity record] selectList filts opts = do srcRes <- selectSourceRes filts opts - liftIO $ with srcRes (C.$$ CL.consume) + liftIO $ with srcRes (\src -> runConduit $ src .| CL.consume) -- | Call 'selectKeys' but return the result as a list. selectKeysList :: (MonadIO m, PersistQueryRead backend, PersistRecordBackend record backend) @@ -102,4 +102,4 @@ selectKeysList :: (MonadIO m, PersistQueryRead backend, PersistRecordBackend rec -> ReaderT backend m [Key record] selectKeysList filts opts = do srcRes <- selectKeysRes filts opts - liftIO $ with srcRes (C.$$ CL.consume) + liftIO $ with srcRes (\src -> runConduit $ src .| CL.consume) diff --git a/persistent/Database/Persist/Sql/Orphan/PersistQuery.hs b/persistent/Database/Persist/Sql/Orphan/PersistQuery.hs index 017463360..9e9976cd2 100644 --- a/persistent/Database/Persist/Sql/Orphan/PersistQuery.hs +++ b/persistent/Database/Persist/Sql/Orphan/PersistQuery.hs @@ -57,7 +57,7 @@ instance PersistQueryRead SqlBackend where selectSourceRes filts opts = do conn <- ask srcRes <- rawQueryRes (sql conn) (getFiltsValues conn filts) - return $ fmap ($= CL.mapM parse) srcRes + return $ fmap (.| CL.mapM parse) srcRes where (limit, offset, orders) = limitOffsetOrder opts @@ -85,7 +85,7 @@ instance PersistQueryRead SqlBackend where selectKeysRes filts opts = do conn <- ask srcRes <- rawQueryRes (sql conn) (getFiltsValues conn filts) - return $ fmap ($= CL.mapM parse) srcRes + return $ fmap (.| CL.mapM parse) srcRes where t = entityDef $ dummyFromFilts filts cols conn = T.intercalate "," $ dbIdColumns conn t diff --git a/persistent/Database/Persist/Sql/Orphan/PersistStore.hs b/persistent/Database/Persist/Sql/Orphan/PersistStore.hs index 1e1e73393..efbe57449 100644 --- a/persistent/Database/Persist/Sql/Orphan/PersistStore.hs +++ b/persistent/Database/Persist/Sql/Orphan/PersistStore.hs @@ -23,7 +23,7 @@ import Database.Persist import Database.Persist.Sql.Types import Database.Persist.Sql.Raw import Database.Persist.Sql.Util (dbIdColumns, keyAndEntityColumnNames) -import qualified Data.Conduit as C +import Data.Conduit import qualified Data.Conduit.List as CL import qualified Data.Text as T import Data.Text (Text, unpack) @@ -45,11 +45,11 @@ import Database.Persist.Class () withRawQuery :: MonadIO m => Text -> [PersistValue] - -> C.Sink [PersistValue] IO a + -> ConduitT [PersistValue] Void IO a -> ReaderT SqlBackend m a withRawQuery sql vals sink = do srcRes <- rawQueryRes sql vals - liftIO $ with srcRes (C.$$ sink) + liftIO $ with srcRes (\src -> runConduit $ src .| sink) toSqlKey :: ToBackendKey SqlBackend record => Int64 -> Key record toSqlKey = fromBackendKey . SqlBackendKey diff --git a/persistent/Database/Persist/Sql/Raw.hs b/persistent/Database/Persist/Sql/Raw.hs index 804c7cb1b..c738f43bb 100644 --- a/persistent/Database/Persist/Sql/Raw.hs +++ b/persistent/Database/Persist/Sql/Raw.hs @@ -24,7 +24,7 @@ import Control.Monad.Trans.Resource (MonadResource,release) rawQuery :: (MonadResource m, MonadReader env m, HasPersistBackend env, BaseBackend env ~ SqlBackend) => Text -> [PersistValue] - -> Source m [PersistValue] + -> ConduitM () [PersistValue] m () rawQuery sql vals = do srcRes <- liftPersist $ rawQueryRes sql vals (releaseKey, src) <- allocateAcquire srcRes @@ -35,7 +35,7 @@ rawQueryRes :: (MonadIO m1, MonadIO m2, IsSqlBackend env) => Text -> [PersistValue] - -> ReaderT env m1 (Acquire (Source m2 [PersistValue])) + -> ReaderT env m1 (Acquire (ConduitM () [PersistValue] m2 ())) rawQueryRes sql vals = do conn <- persistBackend `liftM` ask let make = do @@ -218,7 +218,7 @@ rawSql stmt = run withStmt' colSubsts params sink = do srcRes <- rawQueryRes sql params - liftIO $ with srcRes ($$ sink) + liftIO $ with srcRes (\src -> runConduit $ src .| sink) where sql = T.concat $ makeSubsts colSubsts $ T.splitOn placeholder stmt placeholder = "??" diff --git a/persistent/Database/Persist/Sql/Run.hs b/persistent/Database/Persist/Sql/Run.hs index 3cd4924c9..e7c0636fd 100644 --- a/persistent/Database/Persist/Sql/Run.hs +++ b/persistent/Database/Persist/Sql/Run.hs @@ -15,7 +15,7 @@ import Control.Exception (onException, bracket) import Control.Monad.IO.Unlift import Control.Exception (mask) import System.Timeout (timeout) -import Data.IORef (readIORef, writeIORef, newIORef) +import Data.IORef (readIORef) import qualified Data.Map as Map import Control.Monad (liftM) diff --git a/persistent/Database/Persist/Sql/Types/Internal.hs b/persistent/Database/Persist/Sql/Types/Internal.hs index 4fd486c08..251c64aa5 100644 --- a/persistent/Database/Persist/Sql/Types/Internal.hs +++ b/persistent/Database/Persist/Sql/Types/Internal.hs @@ -28,7 +28,7 @@ import Control.Monad.Logger (LogSource, LogLevel) import Control.Monad.Trans.Class (lift) import Control.Monad.Trans.Reader (ReaderT, runReaderT, ask) import Data.Acquire (Acquire) -import Data.Conduit (Source) +import Data.Conduit (ConduitM) import Data.Int (Int64) import Data.IORef (IORef) import Data.Map (Map) @@ -57,7 +57,7 @@ data Statement = Statement , stmtExecute :: [PersistValue] -> IO Int64 , stmtQuery :: forall m. MonadIO m => [PersistValue] - -> Acquire (Source m [PersistValue]) + -> Acquire (ConduitM () [PersistValue] m ()) } data SqlBackend = SqlBackend diff --git a/persistent/persistent.cabal b/persistent/persistent.cabal index 7b8541387..0639ffad7 100644 --- a/persistent/persistent.cabal +++ b/persistent/persistent.cabal @@ -29,7 +29,7 @@ library , old-locale , text >= 0.8 , containers >= 0.2 - , conduit >= 1.0 + , conduit >= 1.2.8 , resourcet >= 1.1 , resource-pool >= 0.2.2.0 , path-pieces >= 0.1 From a677f988da975d754e5837f739b74db19b75a643 Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Mon, 15 Jan 2018 16:39:38 +0200 Subject: [PATCH 09/20] Require newer monad-logger --- persistent/persistent.cabal | 2 +- stack.yaml | 2 ++ 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/persistent/persistent.cabal b/persistent/persistent.cabal index 0639ffad7..a0c845c00 100644 --- a/persistent/persistent.cabal +++ b/persistent/persistent.cabal @@ -35,7 +35,7 @@ library , path-pieces >= 0.1 , http-api-data >= 0.2 && < 0.4 , aeson >= 0.5 - , monad-logger >= 0.3 + , monad-logger >= 0.3.28 , base64-bytestring , unordered-containers , vector diff --git a/stack.yaml b/stack.yaml index f87b07dab..38452fc8d 100644 --- a/stack.yaml +++ b/stack.yaml @@ -8,3 +8,5 @@ packages: - ./persistent-mysql - ./persistent-postgresql - ./persistent-redis +extra-deps: +- monad-logger-0.3.28 From 961ab3b6be343dec7fedd2da1b331dcf1da5e0db Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Tue, 16 Jan 2018 20:05:27 +0200 Subject: [PATCH 10/20] Major version bump for MonadUnliftIO migration --- persistent-mongoDB/ChangeLog.md | 4 ++++ persistent-mongoDB/persistent-mongoDB.cabal | 4 ++-- persistent-mysql/ChangeLog.md | 3 ++- persistent-mysql/persistent-mysql.cabal | 4 ++-- persistent-postgresql/ChangeLog.md | 4 ++++ persistent-postgresql/persistent-postgresql.cabal | 4 ++-- persistent-sqlite/ChangeLog.md | 4 ++++ persistent-sqlite/persistent-sqlite.cabal | 4 ++-- persistent/ChangeLog.md | 6 +++++- persistent/persistent.cabal | 2 +- 10 files changed, 28 insertions(+), 11 deletions(-) diff --git a/persistent-mongoDB/ChangeLog.md b/persistent-mongoDB/ChangeLog.md index a067008e5..c22956ef2 100644 --- a/persistent-mongoDB/ChangeLog.md +++ b/persistent-mongoDB/ChangeLog.md @@ -1,3 +1,7 @@ +## 2.8.0 + +* Switch from `MonadBaseControl` to `MonadUnliftIO` + ## 2.6.0 * Fix upsert behavior [#613](https://github.com/yesodweb/persistent/issues/613) diff --git a/persistent-mongoDB/persistent-mongoDB.cabal b/persistent-mongoDB/persistent-mongoDB.cabal index 869b30b45..e9f7027bd 100644 --- a/persistent-mongoDB/persistent-mongoDB.cabal +++ b/persistent-mongoDB/persistent-mongoDB.cabal @@ -1,5 +1,5 @@ name: persistent-mongoDB -version: 2.6.0 +version: 2.8.0 license: MIT license-file: LICENSE author: Greg Weber @@ -20,7 +20,7 @@ Flag high_precision_date library build-depends: base >= 4.6 && < 5 - , persistent >= 2.5 && < 3 + , persistent >= 2.8 && < 3 , text >= 0.8 , transformers >= 0.2.1 , containers >= 0.2 diff --git a/persistent-mysql/ChangeLog.md b/persistent-mysql/ChangeLog.md index dc8e6be41..d9c2be5fb 100644 --- a/persistent-mysql/ChangeLog.md +++ b/persistent-mysql/ChangeLog.md @@ -1,5 +1,6 @@ -## Unreleased +## 2.8.0 (Unreleased) +* Switch from `MonadBaseControl` to `MonadUnliftIO` * Fix duplicate migrations when using `mediumtext`, `longtext`, `mediumblob`, `longblob`, and `double`s using a custom precision. [#754](https://github.com/yesodweb/persistent/pull/754) -- This can be released as a minor change on the next update. Currently persistent-mysql can't be released because 2.6.2.2 depends on persistent-2.7.2 being released. diff --git a/persistent-mysql/persistent-mysql.cabal b/persistent-mysql/persistent-mysql.cabal index 463d1d29e..1a64b5054 100644 --- a/persistent-mysql/persistent-mysql.cabal +++ b/persistent-mysql/persistent-mysql.cabal @@ -1,5 +1,5 @@ name: persistent-mysql -version: 2.6.2.1 +version: 2.8.0 license: MIT license-file: LICENSE author: Felipe Lessa , Michael Snoyman @@ -32,7 +32,7 @@ library , mysql-simple >= 0.4.3 && < 0.5 , mysql >= 0.1.1.3 && < 0.2 , blaze-builder - , persistent >= 2.6.1 && < 3 + , persistent >= 2.8.0 && < 3 , containers >= 0.2 , bytestring >= 0.9 , text >= 0.11.0.6 diff --git a/persistent-postgresql/ChangeLog.md b/persistent-postgresql/ChangeLog.md index d0f990d3c..79cd13c3c 100644 --- a/persistent-postgresql/ChangeLog.md +++ b/persistent-postgresql/ChangeLog.md @@ -1,3 +1,7 @@ +## 2.8.0 + +* Switch from `MonadBaseControl` to `MonadUnliftIO` + ## 2.6.2.1 * Fix bug where, if a custom column width was set, the field would be migrated every time [#742](https://github.com/yesodweb/persistent/pull/742) diff --git a/persistent-postgresql/persistent-postgresql.cabal b/persistent-postgresql/persistent-postgresql.cabal index d9b7536ea..c2989897c 100644 --- a/persistent-postgresql/persistent-postgresql.cabal +++ b/persistent-postgresql/persistent-postgresql.cabal @@ -1,5 +1,5 @@ name: persistent-postgresql -version: 2.6.2.1 +version: 2.8.0 license: MIT license-file: LICENSE author: Felipe Lessa, Michael Snoyman @@ -19,7 +19,7 @@ library , transformers >= 0.2.1 , postgresql-simple >= 0.4.0 && < 0.6 , postgresql-libpq >= 0.6.1 && < 0.10 - , persistent >= 2.6.1 && < 3 + , persistent >= 2.8.0 && < 3 , containers >= 0.2 , bytestring >= 0.9 , text >= 0.7 diff --git a/persistent-sqlite/ChangeLog.md b/persistent-sqlite/ChangeLog.md index cbb9fc9fd..31f3c6b04 100644 --- a/persistent-sqlite/ChangeLog.md +++ b/persistent-sqlite/ChangeLog.md @@ -1,3 +1,7 @@ +## 2.8.0 + +* Switch from `MonadBaseControl` to `MonadUnliftIO` + ## 2.6.4 * Adds a new function `stepConn`, which uses an additional parameter to give more detailed error messages [#750](https://github.com/yesodweb/persistent/pull/750) diff --git a/persistent-sqlite/persistent-sqlite.cabal b/persistent-sqlite/persistent-sqlite.cabal index feda9f686..5d483d56f 100644 --- a/persistent-sqlite/persistent-sqlite.cabal +++ b/persistent-sqlite/persistent-sqlite.cabal @@ -1,5 +1,5 @@ name: persistent-sqlite -version: 2.6.4 +version: 2.8.0 license: MIT license-file: LICENSE author: Michael Snoyman @@ -28,7 +28,7 @@ library build-depends: base >= 4.6 && < 5 , bytestring >= 0.9.1 , transformers >= 0.2.1 - , persistent >= 2.6.1 && < 3 + , persistent >= 2.8.0 && < 3 , unliftio-core , containers >= 0.2 , text >= 0.7 diff --git a/persistent/ChangeLog.md b/persistent/ChangeLog.md index 1ecd6baa6..388f3bb54 100644 --- a/persistent/ChangeLog.md +++ b/persistent/ChangeLog.md @@ -1,4 +1,8 @@ -## 2.7.3 +## 2.8.0 + +* Switch from `MonadBaseControl` to `MonadUnliftIO` + +## 2.7.3 * Reverts [#723](https://github.com/yesodweb/persistent/pull/723), which generalized functions using the `BackendCompatible` class. These changes were an accidental breaking change. * Recommend the `PersistDbSpecific` docs if someone gets an error about converting from `PersistDbSpecific` diff --git a/persistent/persistent.cabal b/persistent/persistent.cabal index a0c845c00..d69c024f8 100644 --- a/persistent/persistent.cabal +++ b/persistent/persistent.cabal @@ -1,5 +1,5 @@ name: persistent -version: 2.7.3 +version: 2.8.0 license: MIT license-file: LICENSE author: Michael Snoyman From 4d0a6f3a4abde46c82691414e0e283a933a39f3e Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Tue, 16 Jan 2018 22:08:56 +0200 Subject: [PATCH 11/20] Revert "Revert "More BackendCompatible generalizations (#723)"" This reverts commit 8617b7b60cd0c416d5f5f285d2ce115a5bd4308d. --- persistent-mysql/Database/Persist/MySQL.hs | 5 ++--- persistent/ChangeLog.md | 1 + persistent/Database/Persist/Sql/Migration.hs | 1 + persistent/Database/Persist/Sql/Raw.hs | 12 ++++++------ persistent/Database/Persist/Sql/Types/Internal.hs | 4 +++- 5 files changed, 13 insertions(+), 10 deletions(-) diff --git a/persistent-mysql/Database/Persist/MySQL.hs b/persistent-mysql/Database/Persist/MySQL.hs index 1035b04d3..c5112e15e 100644 --- a/persistent-mysql/Database/Persist/MySQL.hs +++ b/persistent-mysql/Database/Persist/MySQL.hs @@ -39,7 +39,7 @@ import Control.Monad.Logger (MonadLogger, runNoLoggingT) import Control.Monad.IO.Class (MonadIO (..)) import Control.Monad.Trans.Class (lift) import Control.Monad.Trans.Except (runExceptT) -import Control.Monad.Trans.Reader (runReaderT, ReaderT, withReaderT) +import Control.Monad.Trans.Reader (runReaderT, ReaderT) import Control.Monad.Trans.Writer (runWriterT) import Data.Either (partitionEithers) import Data.Monoid ((<>)) @@ -1220,8 +1220,7 @@ insertManyOnDuplicateKeyUpdate -> ReaderT backend m () insertManyOnDuplicateKeyUpdate [] _ _ = return () insertManyOnDuplicateKeyUpdate records fieldValues updates = - withReaderT projectBackend - . uncurry rawExecute + uncurry rawExecute $ mkBulkInsertQuery records fieldValues updates -- | This creates the query for 'bulkInsertOnDuplicateKeyUpdate'. If you diff --git a/persistent/ChangeLog.md b/persistent/ChangeLog.md index 388f3bb54..25c1c52ea 100644 --- a/persistent/ChangeLog.md +++ b/persistent/ChangeLog.md @@ -1,6 +1,7 @@ ## 2.8.0 * Switch from `MonadBaseControl` to `MonadUnliftIO` +* Reapplies [#723](https://github.com/yesodweb/persistent/pull/723), which was reverted in version 2.7.3. ## 2.7.3 diff --git a/persistent/Database/Persist/Sql/Migration.hs b/persistent/Database/Persist/Sql/Migration.hs index 38890395e..52a33540d 100644 --- a/persistent/Database/Persist/Sql/Migration.hs +++ b/persistent/Database/Persist/Sql/Migration.hs @@ -25,6 +25,7 @@ import System.IO.Silently (hSilence) import Database.Persist.Sql.Types import Database.Persist.Sql.Raw import Database.Persist.Types +import Database.Persist.Sql.Orphan.PersistStore() allSql :: CautiousMigration -> [Sql] allSql = map snd diff --git a/persistent/Database/Persist/Sql/Raw.hs b/persistent/Database/Persist/Sql/Raw.hs index c738f43bb..b3d777b18 100644 --- a/persistent/Database/Persist/Sql/Raw.hs +++ b/persistent/Database/Persist/Sql/Raw.hs @@ -47,20 +47,20 @@ rawQueryRes sql vals = do stmtQuery stmt vals -- | Execute a raw SQL statement -rawExecute :: MonadIO m +rawExecute :: (MonadIO m, BackendCompatible SqlBackend backend) => Text -- ^ SQL statement, possibly with placeholders. -> [PersistValue] -- ^ Values to fill the placeholders. - -> ReaderT SqlBackend m () + -> ReaderT backend m () rawExecute x y = liftM (const ()) $ rawExecuteCount x y -- | Execute a raw SQL statement and return the number of -- rows it has modified. -rawExecuteCount :: (MonadIO m, IsSqlBackend backend) +rawExecuteCount :: (MonadIO m, BackendCompatible SqlBackend backend) => Text -- ^ SQL statement, possibly with placeholders. -> [PersistValue] -- ^ Values to fill the placeholders. -> ReaderT backend m Int64 rawExecuteCount sql vals = do - conn <- persistBackend `liftM` ask + conn <- projectBackend `liftM` ask runLoggingT (logDebugNS (pack "SQL") $ T.append sql $ pack $ "; " ++ show vals) (connLogFunc conn) stmt <- getStmt sql @@ -69,10 +69,10 @@ rawExecuteCount sql vals = do return res getStmt - :: (MonadIO m, IsSqlBackend backend) + :: (MonadIO m, BackendCompatible SqlBackend backend) => Text -> ReaderT backend m Statement getStmt sql = do - conn <- persistBackend `liftM` ask + conn <- projectBackend `liftM` ask liftIO $ getStmtConn conn sql getStmtConn :: SqlBackend -> Text -> IO Statement diff --git a/persistent/Database/Persist/Sql/Types/Internal.hs b/persistent/Database/Persist/Sql/Types/Internal.hs index 251c64aa5..998ffc37f 100644 --- a/persistent/Database/Persist/Sql/Types/Internal.hs +++ b/persistent/Database/Persist/Sql/Types/Internal.hs @@ -1,3 +1,4 @@ +{-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE ConstraintKinds #-} {-# LANGUAGE RankNTypes #-} {-# LANGUAGE MultiParamTypeClasses #-} @@ -39,6 +40,7 @@ import Database.Persist.Class , PersistQueryRead, PersistQueryWrite , PersistStoreRead, PersistStoreWrite , PersistUniqueRead, PersistUniqueWrite + , BackendCompatible(..) ) import Database.Persist.Class.PersistStore (IsPersistBackend (..)) import Database.Persist.Types @@ -130,7 +132,7 @@ readToUnknown ma = do -- | A constraint synonym which witnesses that a backend is SQL and can run read queries. type SqlBackendCanRead backend = - ( IsSqlBackend backend + ( BackendCompatible SqlBackend backend , PersistQueryRead backend, PersistStoreRead backend, PersistUniqueRead backend ) -- | A constraint synonym which witnesses that a backend is SQL and can run read and write queries. From 3c2416c648a6d7248de90493fa0b70cee98547d5 Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Sun, 21 Jan 2018 11:51:12 +0200 Subject: [PATCH 12/20] Include extra-deps for conduit 1.3 --- stack.yaml | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/stack.yaml b/stack.yaml index 3a46d621b..af319fab2 100644 --- a/stack.yaml +++ b/stack.yaml @@ -11,3 +11,15 @@ packages: extra-deps: - monad-logger-0.3.28 - mysql-simple-0.4.4 + +- mono-traversable-1.0.8.1 +- unliftio-0.2.4.0 +- unliftio-core-0.1.1.0 +- async-2.1.1.1 +- typed-process-0.2.1.0 +- git: https://github.com/snoyberg/conduit + commit: 7f75bfca8d479e1737861a75437a288af662a3cf + subdirs: + - conduit + - conduit-extra + - resourcet From f45653c059e34192284f787939fa962b1b456579 Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Thu, 1 Feb 2018 19:44:40 +0200 Subject: [PATCH 13/20] Remove unnecessary conduit repo in stack.yaml --- stack.yaml | 6 ------ 1 file changed, 6 deletions(-) diff --git a/stack.yaml b/stack.yaml index af319fab2..b216268a1 100644 --- a/stack.yaml +++ b/stack.yaml @@ -17,9 +17,3 @@ extra-deps: - unliftio-core-0.1.1.0 - async-2.1.1.1 - typed-process-0.2.1.0 -- git: https://github.com/snoyberg/conduit - commit: 7f75bfca8d479e1737861a75437a288af662a3cf - subdirs: - - conduit - - conduit-extra - - resourcet From 52b954d23e72e55caaa2a62307a28a50e03b5df5 Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Thu, 1 Feb 2018 19:56:28 +0200 Subject: [PATCH 14/20] Require resourcet 1.1.10 for MonadUnliftIO instance --- persistent/persistent.cabal | 2 +- stack.yaml | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/persistent/persistent.cabal b/persistent/persistent.cabal index d69c024f8..60fb465bf 100644 --- a/persistent/persistent.cabal +++ b/persistent/persistent.cabal @@ -30,7 +30,7 @@ library , text >= 0.8 , containers >= 0.2 , conduit >= 1.2.8 - , resourcet >= 1.1 + , resourcet >= 1.1.10 , resource-pool >= 0.2.2.0 , path-pieces >= 0.1 , http-api-data >= 0.2 && < 0.4 diff --git a/stack.yaml b/stack.yaml index b216268a1..3c565e089 100644 --- a/stack.yaml +++ b/stack.yaml @@ -17,3 +17,4 @@ extra-deps: - unliftio-core-0.1.1.0 - async-2.1.1.1 - typed-process-0.2.1.0 +- resourcet-1.1.11 From 6f2b99a734a00b8283825f5d18bcea299c4211da Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Thu, 1 Feb 2018 20:11:55 +0200 Subject: [PATCH 15/20] Fixes for older LTSes --- .travis.yml | 40 +++++++------------ persistent-mongoDB/persistent-mongoDB.cabal | 2 +- persistent-mysql/persistent-mysql.cabal | 2 +- .../Database/Persist/Postgresql.hs | 2 +- .../persistent-postgresql.cabal | 2 +- persistent-sqlite/Database/Persist/Sqlite.hs | 2 +- persistent-sqlite/persistent-sqlite.cabal | 2 +- .../Persist/Sql/Orphan/PersistStore.hs | 3 +- persistent/persistent.cabal | 3 +- stack.yaml | 4 +- 10 files changed, 26 insertions(+), 36 deletions(-) diff --git a/.travis.yml b/.travis.yml index 9b7a9b303..700e9e077 100644 --- a/.travis.yml +++ b/.travis.yml @@ -5,9 +5,6 @@ addons: apt: packages: - libgmp-dev - - cabal-install-1.24 - sources: - - hvr-ghc postgresql: "9.6" services: @@ -17,37 +14,28 @@ services: matrix: include: - - env: ARGS="--resolver lts-2" BACKEND=none - - env: ARGS="--resolver lts-2" BACKEND=sqlite - - env: ARGS="--resolver lts-2" BACKEND=mongodb - - env: ARGS="--resolver lts-2" BACKEND=postgresql - - env: ARGS="--resolver lts-2" BACKEND=mysql - - env: ARGS="--resolver lts-6" BACKEND=none - - env: ARGS="--resolver lts-6" BACKEND=sqlite - - env: ARGS="--resolver lts-6" BACKEND=mongodb - - env: ARGS="--resolver lts-6" BACKEND=postgresql - - env: ARGS="--resolver lts-6" BACKEND=mysql + - env: ARGS="--resolver lts-7" BACKEND=none + - env: ARGS="--resolver lts-9" BACKEND=none + - env: ARGS="--resolver lts-10" BACKEND=none + + - env: ARGS="--resolver lts-9" BACKEND=sqlite + - env: ARGS="--resolver lts-9" BACKEND=mongodb + - env: ARGS="--resolver lts-9" BACKEND=postgresql + - env: ARGS="--resolver lts-9" BACKEND=mysql - - env: ARGS="" BACKEND=none - - env: ARGS="" BACKEND=sqlite - - env: ARGS="" BACKEND=mongodb - - env: ARGS="" BACKEND=postgresql - - env: ARGS="" BACKEND=mysql + - env: ARGS="--resolver lts-10" BACKEND=sqlite + - env: ARGS="--resolver lts-10" BACKEND=mongodb + - env: ARGS="--resolver lts-10" BACKEND=postgresql + - env: ARGS="--resolver lts-10" BACKEND=mysql before_install: # Download and unpack the stack executable - mkdir -p ~/.local/bin -- export PATH=$HOME/.local/bin:/opt/cabal/1.24/bin:$PATH +- export PATH=$HOME/.local/bin:$PATH - travis_retry curl -L https://www.stackage.org/stack/linux-x86_64 | tar xz --wildcards --strip-components=1 -C ~/.local/bin '*/stack' -# *TEMPORARY* A proposed change #717 uses haskell-src-meta, which requires -# `happy` to build haskell-src-exts. This fails in lts-2. Revert to the -# commented out line if #717 is not merged, or when testing lts-2 is dropped. -# script: travis/run.sh -script: -- if [[ $ARGS =~ lts-2 ]]; then stack $ARGS --install-ghc install happy; fi -- travis/run.sh +script: travis/run.sh cache: directories: diff --git a/persistent-mongoDB/persistent-mongoDB.cabal b/persistent-mongoDB/persistent-mongoDB.cabal index e9f7027bd..29f4d11c6 100644 --- a/persistent-mongoDB/persistent-mongoDB.cabal +++ b/persistent-mongoDB/persistent-mongoDB.cabal @@ -19,7 +19,7 @@ Flag high_precision_date Default: False library - build-depends: base >= 4.6 && < 5 + build-depends: base >= 4.8 && < 5 , persistent >= 2.8 && < 3 , text >= 0.8 , transformers >= 0.2.1 diff --git a/persistent-mysql/persistent-mysql.cabal b/persistent-mysql/persistent-mysql.cabal index 1a64b5054..8dde82153 100644 --- a/persistent-mysql/persistent-mysql.cabal +++ b/persistent-mysql/persistent-mysql.cabal @@ -27,7 +27,7 @@ description: extra-source-files: ChangeLog.md library - build-depends: base >= 4.6 && < 5 + build-depends: base >= 4.8 && < 5 , transformers >= 0.2.1 , mysql-simple >= 0.4.3 && < 0.5 , mysql >= 0.1.1.3 && < 0.2 diff --git a/persistent-postgresql/Database/Persist/Postgresql.hs b/persistent-postgresql/Database/Persist/Postgresql.hs index 50a64daeb..d189f6e3c 100644 --- a/persistent-postgresql/Database/Persist/Postgresql.hs +++ b/persistent-postgresql/Database/Persist/Postgresql.hs @@ -47,7 +47,7 @@ import qualified Database.PostgreSQL.LibPQ as LibPQ import Control.Monad.Trans.Resource import Control.Exception (throw) -import Control.Monad.IO.Class (MonadIO (..)) +import Control.Monad.IO.Unlift (MonadIO (..), MonadUnliftIO) import Data.Data import Data.Typeable (Typeable) import Data.IORef diff --git a/persistent-postgresql/persistent-postgresql.cabal b/persistent-postgresql/persistent-postgresql.cabal index c2989897c..fbf1ab45f 100644 --- a/persistent-postgresql/persistent-postgresql.cabal +++ b/persistent-postgresql/persistent-postgresql.cabal @@ -15,7 +15,7 @@ bug-reports: https://github.com/yesodweb/persistent/issues extra-source-files: ChangeLog.md library - build-depends: base >= 4.6 && < 5 + build-depends: base >= 4.8 && < 5 , transformers >= 0.2.1 , postgresql-simple >= 0.4.0 && < 0.6 , postgresql-libpq >= 0.6.1 && < 0.10 diff --git a/persistent-sqlite/Database/Persist/Sqlite.hs b/persistent-sqlite/Database/Persist/Sqlite.hs index 18c1b3bd3..a406b61c3 100644 --- a/persistent-sqlite/Database/Persist/Sqlite.hs +++ b/persistent-sqlite/Database/Persist/Sqlite.hs @@ -42,7 +42,7 @@ import Control.Monad (when) import Control.Monad.IO.Unlift (MonadIO (..), MonadUnliftIO, withUnliftIO, unliftIO) import Control.Monad.Logger (NoLoggingT, runNoLoggingT, MonadLogger) import Control.Monad.Trans.Reader (ReaderT, runReaderT) -import Control.Monad.Trans.Resource (ResourceT, runResourceT) +import UnliftIO.Resource (ResourceT, runResourceT) import Control.Monad.Trans.Writer (runWriterT) import Data.Acquire (Acquire, mkAcquire, with) import Data.Aeson diff --git a/persistent-sqlite/persistent-sqlite.cabal b/persistent-sqlite/persistent-sqlite.cabal index 5d483d56f..a4feab85b 100644 --- a/persistent-sqlite/persistent-sqlite.cabal +++ b/persistent-sqlite/persistent-sqlite.cabal @@ -25,7 +25,7 @@ flag build-sanity-exe default: False library - build-depends: base >= 4.6 && < 5 + build-depends: base >= 4.8 && < 5 , bytestring >= 0.9.1 , transformers >= 0.2.1 , persistent >= 2.8.0 && < 3 diff --git a/persistent/Database/Persist/Sql/Orphan/PersistStore.hs b/persistent/Database/Persist/Sql/Orphan/PersistStore.hs index d3cd1db6f..93a555030 100644 --- a/persistent/Database/Persist/Sql/Orphan/PersistStore.hs +++ b/persistent/Database/Persist/Sql/Orphan/PersistStore.hs @@ -32,6 +32,7 @@ import Control.Monad.IO.Class import Data.ByteString.Char8 (readInteger) import Data.Maybe (isJust) import Data.List (find) +import Data.Void (Void) import Control.Monad.Trans.Reader (ReaderT, ask, withReaderT) import Data.Acquire (with) import Data.Int (Int64) @@ -45,7 +46,7 @@ import Database.Persist.Class () withRawQuery :: MonadIO m => Text -> [PersistValue] - -> ConduitT [PersistValue] Void IO a + -> ConduitM [PersistValue] Void IO a -> ReaderT SqlBackend m a withRawQuery sql vals sink = do srcRes <- rawQueryRes sql vals diff --git a/persistent/persistent.cabal b/persistent/persistent.cabal index 60fb465bf..4a8e2cd28 100644 --- a/persistent/persistent.cabal +++ b/persistent/persistent.cabal @@ -50,6 +50,7 @@ library , scientific , tagged , unliftio-core + , void exposed-modules: Database.Persist Database.Persist.Quasi @@ -86,7 +87,7 @@ test-suite test type: exitcode-stdio-1.0 main-is: test/main.hs - build-depends: base >= 4.6 && < 5 + build-depends: base >= 4.8 && < 5 , hspec >= 1.3 , containers , text diff --git a/stack.yaml b/stack.yaml index 3c565e089..4d42efc35 100644 --- a/stack.yaml +++ b/stack.yaml @@ -1,4 +1,4 @@ -resolver: lts-8.2 +resolver: lts-10.4 packages: - ./persistent - ./persistent-template @@ -9,7 +9,7 @@ packages: - ./persistent-postgresql - ./persistent-redis extra-deps: -- monad-logger-0.3.28 +- monad-logger-0.3.28.1 - mysql-simple-0.4.4 - mono-traversable-1.0.8.1 From 668a7aa02b3a0f55be78a2f1f650078b056a8a0b Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Thu, 1 Feb 2018 20:26:08 +0200 Subject: [PATCH 16/20] Don't use solver --- travis/run.sh | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/travis/run.sh b/travis/run.sh index 917089a9b..1ca949158 100755 --- a/travis/run.sh +++ b/travis/run.sh @@ -2,13 +2,10 @@ set -euxo pipefail -ARGS="$ARGS --no-terminal --install-ghc" -stack $ARGS solver --update-config - if [ "$BACKEND" = "none" ] then PACKAGES=$(stack --install-ghc query locals | grep '^ *path' | sed 's@^ *path:@@' | grep -v 'persistent-test' ) - exec stack $ARGS test --pedantic $PACKAGES + exec stack $ARGS --no-terminal test --pedantic $PACKAGES else if [ "$BACKEND" = "postgresql" ] then @@ -19,5 +16,5 @@ else fi cd persistent-test - exec stack $ARGS test --pedantic --fast persistent-test --flag persistent-test:$BACKEND --exec persistent-test + exec stack $ARGS --no-terminal test --pedantic --fast persistent-test --flag persistent-test:$BACKEND --exec persistent-test fi From 6eb401a16dbd12233061dcb964c5687060822233 Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Thu, 1 Feb 2018 20:48:04 +0200 Subject: [PATCH 17/20] Remove unneeded after_failure --- .travis.yml | 4 ---- 1 file changed, 4 deletions(-) diff --git a/.travis.yml b/.travis.yml index 700e9e077..3334c46de 100644 --- a/.travis.yml +++ b/.travis.yml @@ -48,7 +48,3 @@ cache: # packages: # - libzookeeper-mt-dev # - zookeeperd - -after_failure: -- cat ~/.cabal/logs/*log - From bac4b739d652052ac0ab9ea874ca3deb9000d0c1 Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Thu, 1 Feb 2018 20:48:14 +0200 Subject: [PATCH 18/20] Fix an unneeded import warning --- persistent-postgresql/Database/Persist/Postgresql.hs | 1 - 1 file changed, 1 deletion(-) diff --git a/persistent-postgresql/Database/Persist/Postgresql.hs b/persistent-postgresql/Database/Persist/Postgresql.hs index d189f6e3c..93d3c1a31 100644 --- a/persistent-postgresql/Database/Persist/Postgresql.hs +++ b/persistent-postgresql/Database/Persist/Postgresql.hs @@ -45,7 +45,6 @@ import Database.PostgreSQL.Simple.Ok (Ok (..)) import qualified Database.PostgreSQL.LibPQ as LibPQ -import Control.Monad.Trans.Resource import Control.Exception (throw) import Control.Monad.IO.Unlift (MonadIO (..), MonadUnliftIO) import Data.Data From 462de25cbe169e576e4425931ba94f1e0b114495 Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Thu, 1 Feb 2018 21:01:29 +0200 Subject: [PATCH 19/20] Warnings cleanup --- persistent-test/src/PersistentTest.hs | 1 - persistent/Database/Persist/Class/PersistEntity.hs | 14 ++++++-------- 2 files changed, 6 insertions(+), 9 deletions(-) diff --git a/persistent-test/src/PersistentTest.hs b/persistent-test/src/PersistentTest.hs index 5e2a406f6..5f04ad69a 100644 --- a/persistent-test/src/PersistentTest.hs +++ b/persistent-test/src/PersistentTest.hs @@ -35,7 +35,6 @@ import Database.Persist.MongoDB (toInsertDoc, docToEntityThrow, collectionName, import Database.Persist.TH (mkDeleteCascade, mkSave) import qualified Data.Text as T -import qualified Control.Exception as E # ifdef WITH_POSTGRESQL import Data.List (sort) diff --git a/persistent/Database/Persist/Class/PersistEntity.hs b/persistent/Database/Persist/Class/PersistEntity.hs index 9c077ec03..cabe6ccae 100644 --- a/persistent/Database/Persist/Class/PersistEntity.hs +++ b/persistent/Database/Persist/Class/PersistEntity.hs @@ -175,15 +175,13 @@ data Filter record = forall typ. PersistField typ => Filter data Entity record = Entity { entityKey :: Key record , entityVal :: record } + deriving Typeable -deriving instance (PersistEntity record, Generic (Key record), Generic record) => Generic (Entity record) -deriving instance (PersistEntity record, Eq (Key record), Eq record) => Eq (Entity record) -deriving instance (PersistEntity record, Ord (Key record), Ord record) => Ord (Entity record) -deriving instance (PersistEntity record, Show (Key record), Show record) => Show (Entity record) -deriving instance (PersistEntity record, Read (Key record), Read record) => Read (Entity record) -#if MIN_VERSION_base(4,7,0) -deriving instance Typeable Entity -#endif +deriving instance (Generic (Key record), Generic record) => Generic (Entity record) +deriving instance (Eq (Key record), Eq record) => Eq (Entity record) +deriving instance (Ord (Key record), Ord record) => Ord (Entity record) +deriving instance (Show (Key record), Show record) => Show (Entity record) +deriving instance (Read (Key record), Read record) => Read (Entity record) -- | Get list of values corresponding to given entity. entityValues :: PersistEntity record => Entity record -> [PersistValue] From 49cd4459aa20c099579c35f2d6cc60fa90fd0231 Mon Sep 17 00:00:00 2001 From: Michael Snoyman Date: Thu, 1 Feb 2018 21:45:42 +0200 Subject: [PATCH 20/20] Turn off --pedantic for lts-7 --- travis/run.sh | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/travis/run.sh b/travis/run.sh index 1ca949158..3de12da9e 100755 --- a/travis/run.sh +++ b/travis/run.sh @@ -5,7 +5,16 @@ set -euxo pipefail if [ "$BACKEND" = "none" ] then PACKAGES=$(stack --install-ghc query locals | grep '^ *path' | sed 's@^ *path:@@' | grep -v 'persistent-test' ) - exec stack $ARGS --no-terminal test --pedantic $PACKAGES + + PEDANTIC="--pedantic" + # Turn off pedantic for lts-7, due to the sometimes invalid + # redundant constraint warnings. + if [ "$ARGS" = "--resolver lts-7" ] + then + PEDANTIC="" + fi + + exec stack $ARGS --no-terminal test $PEDANTIC $PACKAGES else if [ "$BACKEND" = "postgresql" ] then