core: single function to initialize the chat controller only if encryption key is correct (#1107)
This commit is contained in:
parent
e1a7b02e59
commit
a977a0dd17
@ -24,6 +24,8 @@ Java_chat_simplex_app_SimplexAppKt_initHS(__unused JNIEnv *env, __unused jclass
|
|||||||
// from simplex-chat
|
// from simplex-chat
|
||||||
typedef void* chat_ctrl;
|
typedef void* chat_ctrl;
|
||||||
|
|
||||||
|
extern char *chat_migrate_init(const char *path, const char *key, chat_ctrl *ctrl);
|
||||||
|
|
||||||
extern char *chat_migrate_db(const char *path, const char *key);
|
extern char *chat_migrate_db(const char *path, const char *key);
|
||||||
extern chat_ctrl chat_init_key(const char *path, const char *key);
|
extern chat_ctrl chat_init_key(const char *path, const char *key);
|
||||||
extern chat_ctrl chat_init(const char *path); // deprecated
|
extern chat_ctrl chat_init(const char *path); // deprecated
|
||||||
|
@ -7,7 +7,7 @@ constraints: zip +disable-bzip2 +disable-zstd
|
|||||||
source-repository-package
|
source-repository-package
|
||||||
type: git
|
type: git
|
||||||
location: https://github.com/simplex-chat/simplexmq.git
|
location: https://github.com/simplex-chat/simplexmq.git
|
||||||
tag: f8505d4add64b2b2ce11a94d878857d404f30fe6
|
tag: 413aad5139acee28033404aed2e5516fc71c337c
|
||||||
|
|
||||||
source-repository-package
|
source-repository-package
|
||||||
type: git
|
type: git
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"https://github.com/simplex-chat/simplexmq.git"."f8505d4add64b2b2ce11a94d878857d404f30fe6" = "1h1dyk2m9kgv63nnnylm68371mg1vxpl6fggqdykg0dknmid3mpy";
|
"https://github.com/simplex-chat/simplexmq.git"."413aad5139acee28033404aed2e5516fc71c337c" = "0vzmglhnbr2x9frs597sg6v8if1hfydbmnza5532sc486qms0vmg";
|
||||||
"https://github.com/simplex-chat/direct-sqlcipher.git"."34309410eb2069b029b8fc1872deb1e0db123294" = "0kwkmhyfsn2lixdlgl15smgr1h5gjk7fky6abzh8rng2h5ymnffd";
|
"https://github.com/simplex-chat/direct-sqlcipher.git"."34309410eb2069b029b8fc1872deb1e0db123294" = "0kwkmhyfsn2lixdlgl15smgr1h5gjk7fky6abzh8rng2h5ymnffd";
|
||||||
"https://github.com/simplex-chat/sqlcipher-simple.git"."5e154a2aeccc33ead6c243ec07195ab673137221" = "1d1gc5wax4vqg0801ajsmx1sbwvd9y7p7b8mmskvqsmpbwgbh0m0";
|
"https://github.com/simplex-chat/sqlcipher-simple.git"."5e154a2aeccc33ead6c243ec07195ab673137221" = "1d1gc5wax4vqg0801ajsmx1sbwvd9y7p7b8mmskvqsmpbwgbh0m0";
|
||||||
"https://github.com/simplex-chat/aeson.git"."3eb66f9a68f103b5f1489382aad89f5712a64db7" = "0kilkx59fl6c3qy3kjczqvm8c3f4n3p0bdk9biyflf51ljnzp4yp";
|
"https://github.com/simplex-chat/aeson.git"."3eb66f9a68f103b5f1489382aad89f5712a64db7" = "0kilkx59fl6c3qy3kjczqvm8c3f4n3p0bdk9biyflf51ljnzp4yp";
|
||||||
|
@ -56,9 +56,9 @@ import Simplex.Chat.Store
|
|||||||
import Simplex.Chat.Types
|
import Simplex.Chat.Types
|
||||||
import Simplex.Chat.Util (lastMaybe, safeDecodeUtf8, uncurry3)
|
import Simplex.Chat.Util (lastMaybe, safeDecodeUtf8, uncurry3)
|
||||||
import Simplex.Messaging.Agent as Agent
|
import Simplex.Messaging.Agent as Agent
|
||||||
import Simplex.Messaging.Agent.Env.SQLite (AgentConfig (..), InitialAgentServers (..), defaultAgentConfig)
|
import Simplex.Messaging.Agent.Env.SQLite (AgentConfig (..), AgentDatabase (..), InitialAgentServers (..), createAgentStore, defaultAgentConfig)
|
||||||
import Simplex.Messaging.Agent.Protocol
|
import Simplex.Messaging.Agent.Protocol
|
||||||
import Simplex.Messaging.Agent.Store.SQLite (exexSQL)
|
import Simplex.Messaging.Agent.Store.SQLite (SQLiteStore (dbNew), exexSQL)
|
||||||
import Simplex.Messaging.Client (defaultNetworkConfig)
|
import Simplex.Messaging.Client (defaultNetworkConfig)
|
||||||
import qualified Simplex.Messaging.Crypto as C
|
import qualified Simplex.Messaging.Crypto as C
|
||||||
import Simplex.Messaging.Encoding
|
import Simplex.Messaging.Encoding
|
||||||
@ -86,8 +86,7 @@ defaultChatConfig =
|
|||||||
{ agentConfig =
|
{ agentConfig =
|
||||||
defaultAgentConfig
|
defaultAgentConfig
|
||||||
{ tcpPort = undefined, -- agent does not listen to TCP
|
{ tcpPort = undefined, -- agent does not listen to TCP
|
||||||
dbFile = "simplex_v1",
|
database = AgentDBFile {dbFile = "simplex_v1_agent", dbKey = ""},
|
||||||
dbKey = "",
|
|
||||||
yesToMigrations = False
|
yesToMigrations = False
|
||||||
},
|
},
|
||||||
yesToMigrations = False,
|
yesToMigrations = False,
|
||||||
@ -125,16 +124,21 @@ fixedImagePreview = ImageData "
|
|||||||
logCfg :: LogConfig
|
logCfg :: LogConfig
|
||||||
logCfg = LogConfig {lc_file = Nothing, lc_stderr = True}
|
logCfg = LogConfig {lc_file = Nothing, lc_stderr = True}
|
||||||
|
|
||||||
newChatController :: SQLiteStore -> Maybe User -> ChatConfig -> ChatOpts -> Maybe (Notification -> IO ()) -> IO ChatController
|
createChatDatabase :: FilePath -> String -> Bool -> IO ChatDatabase
|
||||||
newChatController chatStore user cfg@ChatConfig {agentConfig = aCfg, tbqSize, defaultServers} ChatOpts {dbFilePrefix, dbKey, smpServers, networkConfig, logConnections, logServerHosts} sendToast = do
|
createChatDatabase filePrefix key yesToMigrations = do
|
||||||
let f = chatStoreFile dbFilePrefix
|
chatStore <- createChatStore (chatStoreFile filePrefix) key yesToMigrations
|
||||||
config = cfg {subscriptionEvents = logConnections, hostEvents = logServerHosts}
|
agentStore <- createAgentStore (agentStoreFile filePrefix) key yesToMigrations
|
||||||
|
pure ChatDatabase {chatStore, agentStore}
|
||||||
|
|
||||||
|
newChatController :: ChatDatabase -> Maybe User -> ChatConfig -> ChatOpts -> Maybe (Notification -> IO ()) -> IO ChatController
|
||||||
|
newChatController ChatDatabase {chatStore, agentStore} user cfg@ChatConfig {agentConfig = aCfg, tbqSize, defaultServers} ChatOpts {smpServers, networkConfig, logConnections, logServerHosts} sendToast = do
|
||||||
|
let config = cfg {subscriptionEvents = logConnections, hostEvents = logServerHosts}
|
||||||
sendNotification = fromMaybe (const $ pure ()) sendToast
|
sendNotification = fromMaybe (const $ pure ()) sendToast
|
||||||
|
firstTime = dbNew chatStore
|
||||||
activeTo <- newTVarIO ActiveNone
|
activeTo <- newTVarIO ActiveNone
|
||||||
firstTime <- not <$> doesFileExist f
|
|
||||||
currentUser <- newTVarIO user
|
currentUser <- newTVarIO user
|
||||||
servers <- resolveServers defaultServers
|
servers <- resolveServers defaultServers
|
||||||
smpAgent <- getSMPAgentClient aCfg {dbFile = agentStoreFile dbFilePrefix, dbKey} servers {netCfg = networkConfig}
|
smpAgent <- getSMPAgentClient aCfg {database = AgentDB agentStore} servers {netCfg = networkConfig}
|
||||||
agentAsync <- newTVarIO Nothing
|
agentAsync <- newTVarIO Nothing
|
||||||
idsDrg <- newTVarIO =<< drgNew
|
idsDrg <- newTVarIO =<< drgNew
|
||||||
inputQ <- newTBQueueIO tbqSize
|
inputQ <- newTBQueueIO tbqSize
|
||||||
@ -2653,9 +2657,9 @@ withStore ::
|
|||||||
(DB.Connection -> ExceptT StoreError IO a) ->
|
(DB.Connection -> ExceptT StoreError IO a) ->
|
||||||
m a
|
m a
|
||||||
withStore action = do
|
withStore action = do
|
||||||
st <- asks chatStore
|
ChatController {chatStore} <- ask
|
||||||
liftEitherError ChatErrorStore $
|
liftEitherError ChatErrorStore $
|
||||||
withTransaction st (runExceptT . action) `E.catch` handleInternal
|
withTransaction chatStore (runExceptT . action) `E.catch` handleInternal
|
||||||
where
|
where
|
||||||
handleInternal :: E.SomeException -> IO (Either StoreError a)
|
handleInternal :: E.SomeException -> IO (Either StoreError a)
|
||||||
handleInternal = pure . Left . SEInternalError . show
|
handleInternal = pure . Left . SEInternalError . show
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
{-# LANGUAGE DuplicateRecordFields #-}
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE LambdaCase #-}
|
{-# LANGUAGE LambdaCase #-}
|
||||||
{-# LANGUAGE NamedFieldPuns #-}
|
{-# LANGUAGE NamedFieldPuns #-}
|
||||||
@ -19,7 +20,7 @@ import Data.Functor (($>))
|
|||||||
import qualified Data.Text as T
|
import qualified Data.Text as T
|
||||||
import qualified Database.SQLite3 as SQL
|
import qualified Database.SQLite3 as SQL
|
||||||
import Simplex.Chat.Controller
|
import Simplex.Chat.Controller
|
||||||
import Simplex.Messaging.Agent.Client (agentStore)
|
import Simplex.Messaging.Agent.Client (agentClientStore)
|
||||||
import Simplex.Messaging.Agent.Store.SQLite (SQLiteStore (..), sqlString)
|
import Simplex.Messaging.Agent.Store.SQLite (SQLiteStore (..), sqlString)
|
||||||
import Simplex.Messaging.Util (unlessM, whenM)
|
import Simplex.Messaging.Util (unlessM, whenM)
|
||||||
import System.FilePath
|
import System.FilePath
|
||||||
@ -97,7 +98,7 @@ storageFiles :: ChatMonad m => m StorageFiles
|
|||||||
storageFiles = do
|
storageFiles = do
|
||||||
ChatController {chatStore, filesFolder, smpAgent} <- ask
|
ChatController {chatStore, filesFolder, smpAgent} <- ask
|
||||||
let SQLiteStore {dbFilePath = chatDb, dbEncrypted = chatEncrypted} = chatStore
|
let SQLiteStore {dbFilePath = chatDb, dbEncrypted = chatEncrypted} = chatStore
|
||||||
SQLiteStore {dbFilePath = agentDb, dbEncrypted = agentEncrypted} = agentStore smpAgent
|
SQLiteStore {dbFilePath = agentDb, dbEncrypted = agentEncrypted} = agentClientStore smpAgent
|
||||||
filesPath <- readTVarIO filesFolder
|
filesPath <- readTVarIO filesFolder
|
||||||
pure StorageFiles {chatDb, chatEncrypted, agentDb, agentEncrypted, filesPath}
|
pure StorageFiles {chatDb, chatEncrypted, agentDb, agentEncrypted, filesPath}
|
||||||
|
|
||||||
|
@ -75,6 +75,8 @@ data ChatConfig = ChatConfig
|
|||||||
data ActiveTo = ActiveNone | ActiveC ContactName | ActiveG GroupName
|
data ActiveTo = ActiveNone | ActiveC ContactName | ActiveG GroupName
|
||||||
deriving (Eq)
|
deriving (Eq)
|
||||||
|
|
||||||
|
data ChatDatabase = ChatDatabase {chatStore :: SQLiteStore, agentStore :: SQLiteStore}
|
||||||
|
|
||||||
data ChatController = ChatController
|
data ChatController = ChatController
|
||||||
{ currentUser :: TVar (Maybe User),
|
{ currentUser :: TVar (Maybe User),
|
||||||
activeTo :: TVar ActiveTo,
|
activeTo :: TVar ActiveTo,
|
||||||
|
@ -10,22 +10,20 @@ import Data.Text.Encoding (encodeUtf8)
|
|||||||
import Simplex.Chat
|
import Simplex.Chat
|
||||||
import Simplex.Chat.Controller
|
import Simplex.Chat.Controller
|
||||||
import Simplex.Chat.Options (ChatOpts (..))
|
import Simplex.Chat.Options (ChatOpts (..))
|
||||||
import Simplex.Chat.Store
|
|
||||||
import Simplex.Chat.Types
|
import Simplex.Chat.Types
|
||||||
import UnliftIO.Async
|
import UnliftIO.Async
|
||||||
|
|
||||||
simplexChatCore :: ChatConfig -> ChatOpts -> Maybe (Notification -> IO ()) -> (User -> ChatController -> IO ()) -> IO ()
|
simplexChatCore :: ChatConfig -> ChatOpts -> Maybe (Notification -> IO ()) -> (User -> ChatController -> IO ()) -> IO ()
|
||||||
simplexChatCore cfg@ChatConfig {yesToMigrations} opts sendToast chat
|
simplexChatCore cfg@ChatConfig {yesToMigrations} opts@ChatOpts {dbFilePrefix, dbKey} sendToast chat
|
||||||
| logAgent opts = do
|
| logAgent opts = do
|
||||||
setLogLevel LogInfo -- LogError
|
setLogLevel LogInfo -- LogError
|
||||||
withGlobalLogging logCfg initRun
|
withGlobalLogging logCfg initRun
|
||||||
| otherwise = initRun
|
| otherwise = initRun
|
||||||
where
|
where
|
||||||
initRun = do
|
initRun = do
|
||||||
let f = chatStoreFile $ dbFilePrefix opts
|
db@ChatDatabase {chatStore} <- createChatDatabase dbFilePrefix dbKey yesToMigrations
|
||||||
st <- createChatStore f (dbKey opts) yesToMigrations
|
u <- getCreateActiveUser chatStore
|
||||||
u <- getCreateActiveUser st
|
cc <- newChatController db (Just u) cfg opts sendToast
|
||||||
cc <- newChatController st (Just u) cfg opts sendToast
|
|
||||||
runSimplexChat opts u cc chat
|
runSimplexChat opts u cc chat
|
||||||
|
|
||||||
runSimplexChat :: ChatOpts -> User -> ChatController -> (User -> ChatController -> IO ()) -> IO ()
|
runSimplexChat :: ChatOpts -> User -> ChatController -> (User -> ChatController -> IO ()) -> IO ()
|
||||||
|
@ -8,6 +8,7 @@ module Simplex.Chat.Mobile where
|
|||||||
|
|
||||||
import Control.Concurrent.STM
|
import Control.Concurrent.STM
|
||||||
import Control.Exception (catch)
|
import Control.Exception (catch)
|
||||||
|
import Control.Monad.Except
|
||||||
import Control.Monad.Reader
|
import Control.Monad.Reader
|
||||||
import Data.Aeson (ToJSON (..))
|
import Data.Aeson (ToJSON (..))
|
||||||
import qualified Data.Aeson as J
|
import qualified Data.Aeson as J
|
||||||
@ -20,7 +21,9 @@ import Database.SQLite.Simple (SQLError (..))
|
|||||||
import qualified Database.SQLite.Simple as DB
|
import qualified Database.SQLite.Simple as DB
|
||||||
import Foreign.C.String
|
import Foreign.C.String
|
||||||
import Foreign.C.Types (CInt (..))
|
import Foreign.C.Types (CInt (..))
|
||||||
|
import Foreign.Ptr
|
||||||
import Foreign.StablePtr
|
import Foreign.StablePtr
|
||||||
|
import Foreign.Storable (poke)
|
||||||
import GHC.Generics (Generic)
|
import GHC.Generics (Generic)
|
||||||
import Simplex.Chat
|
import Simplex.Chat
|
||||||
import Simplex.Chat.Controller
|
import Simplex.Chat.Controller
|
||||||
@ -37,11 +40,15 @@ import Simplex.Messaging.Protocol (CorrId (..))
|
|||||||
import Simplex.Messaging.Util (catchAll)
|
import Simplex.Messaging.Util (catchAll)
|
||||||
import System.Timeout (timeout)
|
import System.Timeout (timeout)
|
||||||
|
|
||||||
|
foreign export ccall "chat_migrate_init" cChatMigrateInit :: CString -> CString -> Ptr (StablePtr ChatController) -> IO CJSONString
|
||||||
|
|
||||||
|
-- TODO remove
|
||||||
foreign export ccall "chat_migrate_db" cChatMigrateDB :: CString -> CString -> IO CJSONString
|
foreign export ccall "chat_migrate_db" cChatMigrateDB :: CString -> CString -> IO CJSONString
|
||||||
|
|
||||||
-- chat_init is deprecated
|
-- chat_init is deprecated
|
||||||
foreign export ccall "chat_init" cChatInit :: CString -> IO (StablePtr ChatController)
|
foreign export ccall "chat_init" cChatInit :: CString -> IO (StablePtr ChatController)
|
||||||
|
|
||||||
|
-- TODO remove
|
||||||
foreign export ccall "chat_init_key" cChatInitKey :: CString -> CString -> IO (StablePtr ChatController)
|
foreign export ccall "chat_init_key" cChatInitKey :: CString -> CString -> IO (StablePtr ChatController)
|
||||||
|
|
||||||
foreign export ccall "chat_send_cmd" cChatSendCmd :: StablePtr ChatController -> CString -> IO CJSONString
|
foreign export ccall "chat_send_cmd" cChatSendCmd :: StablePtr ChatController -> CString -> IO CJSONString
|
||||||
@ -52,8 +59,20 @@ foreign export ccall "chat_recv_msg_wait" cChatRecvMsgWait :: StablePtr ChatCont
|
|||||||
|
|
||||||
foreign export ccall "chat_parse_markdown" cChatParseMarkdown :: CString -> IO CJSONString
|
foreign export ccall "chat_parse_markdown" cChatParseMarkdown :: CString -> IO CJSONString
|
||||||
|
|
||||||
|
-- | check / migrate database and initialize chat controller on success
|
||||||
|
cChatMigrateInit :: CString -> CString -> Ptr (StablePtr ChatController) -> IO CJSONString
|
||||||
|
cChatMigrateInit fp key ctrl = do
|
||||||
|
dbPath <- peekCAString fp
|
||||||
|
dbKey <- peekCAString key
|
||||||
|
r <-
|
||||||
|
chatMigrateInit dbPath dbKey >>= \case
|
||||||
|
Right cc -> (newStablePtr cc >>= poke ctrl) $> DBMOk
|
||||||
|
Left e -> pure e
|
||||||
|
newCAString . LB.unpack $ J.encode r
|
||||||
|
|
||||||
-- | check and migrate the database
|
-- | check and migrate the database
|
||||||
-- This function validates that the encryption is correct and runs migrations - it should be called before cChatInitKey
|
-- This function validates that the encryption is correct and runs migrations - it should be called before cChatInitKey
|
||||||
|
-- TODO remove
|
||||||
cChatMigrateDB :: CString -> CString -> IO CJSONString
|
cChatMigrateDB :: CString -> CString -> IO CJSONString
|
||||||
cChatMigrateDB fp key =
|
cChatMigrateDB fp key =
|
||||||
((,) <$> peekCAString fp <*> peekCAString key) >>= uncurry chatMigrateDB >>= newCAString . LB.unpack . J.encode
|
((,) <$> peekCAString fp <*> peekCAString key) >>= uncurry chatMigrateDB >>= newCAString . LB.unpack . J.encode
|
||||||
@ -65,6 +84,7 @@ cChatInit fp = peekCAString fp >>= chatInit >>= newStablePtr
|
|||||||
|
|
||||||
-- | initialize chat controller with encrypted database
|
-- | initialize chat controller with encrypted database
|
||||||
-- The active user has to be created and the chat has to be started before most commands can be used.
|
-- The active user has to be created and the chat has to be started before most commands can be used.
|
||||||
|
-- TODO remove
|
||||||
cChatInitKey :: CString -> CString -> IO (StablePtr ChatController)
|
cChatInitKey :: CString -> CString -> IO (StablePtr ChatController)
|
||||||
cChatInitKey fp key =
|
cChatInitKey fp key =
|
||||||
((,) <$> peekCAString fp <*> peekCAString key) >>= uncurry chatInitKey >>= newStablePtr
|
((,) <$> peekCAString fp <*> peekCAString key) >>= uncurry chatInitKey >>= newStablePtr
|
||||||
@ -126,6 +146,27 @@ instance ToJSON DBMigrationResult where
|
|||||||
toJSON = J.genericToJSON . sumTypeJSON $ dropPrefix "DBM"
|
toJSON = J.genericToJSON . sumTypeJSON $ dropPrefix "DBM"
|
||||||
toEncoding = J.genericToEncoding . sumTypeJSON $ dropPrefix "DBM"
|
toEncoding = J.genericToEncoding . sumTypeJSON $ dropPrefix "DBM"
|
||||||
|
|
||||||
|
chatMigrateInit :: String -> String -> IO (Either DBMigrationResult ChatController)
|
||||||
|
chatMigrateInit dbFilePrefix dbKey = runExceptT $ do
|
||||||
|
chatStore <- migrate createChatStore $ chatStoreFile dbFilePrefix
|
||||||
|
agentStore <- migrate createAgentStore $ agentStoreFile dbFilePrefix
|
||||||
|
liftIO $ initialize chatStore ChatDatabase {chatStore, agentStore}
|
||||||
|
where
|
||||||
|
initialize st db = do
|
||||||
|
user_ <- getActiveUser_ st
|
||||||
|
newChatController db user_ defaultMobileConfig mobileChatOpts {dbFilePrefix, dbKey} Nothing
|
||||||
|
migrate createStore dbFile =
|
||||||
|
ExceptT $
|
||||||
|
(Right <$> createStore dbFile dbKey True)
|
||||||
|
`catch` (pure . checkDBError)
|
||||||
|
`catchAll` (pure . dbError)
|
||||||
|
where
|
||||||
|
checkDBError e = case sqlError e of
|
||||||
|
DB.ErrorNotADatabase -> Left $ DBMErrorNotADatabase dbFile
|
||||||
|
_ -> dbError e
|
||||||
|
dbError e = Left . DBMError dbFile $ show e
|
||||||
|
|
||||||
|
-- TODO remove
|
||||||
chatMigrateDB :: String -> String -> IO DBMigrationResult
|
chatMigrateDB :: String -> String -> IO DBMigrationResult
|
||||||
chatMigrateDB dbFilePrefix dbKey =
|
chatMigrateDB dbFilePrefix dbKey =
|
||||||
migrate createChatStore (chatStoreFile dbFilePrefix) >>= \case
|
migrate createChatStore (chatStoreFile dbFilePrefix) >>= \case
|
||||||
@ -145,12 +186,12 @@ chatMigrateDB dbFilePrefix dbKey =
|
|||||||
chatInit :: String -> IO ChatController
|
chatInit :: String -> IO ChatController
|
||||||
chatInit = (`chatInitKey` "")
|
chatInit = (`chatInitKey` "")
|
||||||
|
|
||||||
|
-- TODO remove
|
||||||
chatInitKey :: String -> String -> IO ChatController
|
chatInitKey :: String -> String -> IO ChatController
|
||||||
chatInitKey dbFilePrefix dbKey = do
|
chatInitKey dbFilePrefix dbKey = do
|
||||||
let f = chatStoreFile dbFilePrefix
|
db@ChatDatabase {chatStore} <- createChatDatabase dbFilePrefix dbKey True
|
||||||
chatStore <- createChatStore f dbKey (yesToMigrations (defaultMobileConfig :: ChatConfig))
|
|
||||||
user_ <- getActiveUser_ chatStore
|
user_ <- getActiveUser_ chatStore
|
||||||
newChatController chatStore user_ defaultMobileConfig mobileChatOpts {dbFilePrefix, dbKey} Nothing
|
newChatController db user_ defaultMobileConfig mobileChatOpts {dbFilePrefix, dbKey} Nothing
|
||||||
|
|
||||||
chatSendCmd :: ChatController -> String -> IO JSONString
|
chatSendCmd :: ChatController -> String -> IO JSONString
|
||||||
chatSendCmd cc s = LB.unpack . J.encode . APIResponse Nothing <$> runReaderT (execChatCommand $ B.pack s) cc
|
chatSendCmd cc s = LB.unpack . J.encode . APIResponse Nothing <$> runReaderT (execChatCommand $ B.pack s) cc
|
||||||
|
@ -49,7 +49,7 @@ extra-deps:
|
|||||||
# - simplexmq-1.0.0@sha256:34b2004728ae396e3ae449cd090ba7410781e2b3cefc59259915f4ca5daa9ea8,8561
|
# - simplexmq-1.0.0@sha256:34b2004728ae396e3ae449cd090ba7410781e2b3cefc59259915f4ca5daa9ea8,8561
|
||||||
# - ../simplexmq
|
# - ../simplexmq
|
||||||
- github: simplex-chat/simplexmq
|
- github: simplex-chat/simplexmq
|
||||||
commit: f8505d4add64b2b2ce11a94d878857d404f30fe6
|
commit: 413aad5139acee28033404aed2e5516fc71c337c
|
||||||
# - ../direct-sqlcipher
|
# - ../direct-sqlcipher
|
||||||
- github: simplex-chat/direct-sqlcipher
|
- github: simplex-chat/direct-sqlcipher
|
||||||
commit: 34309410eb2069b029b8fc1872deb1e0db123294
|
commit: 34309410eb2069b029b8fc1872deb1e0db123294
|
||||||
|
@ -18,7 +18,7 @@ import Data.Maybe (fromJust, isNothing)
|
|||||||
import qualified Data.Text as T
|
import qualified Data.Text as T
|
||||||
import Network.Socket
|
import Network.Socket
|
||||||
import Simplex.Chat
|
import Simplex.Chat
|
||||||
import Simplex.Chat.Controller (ChatConfig (..), ChatController (..))
|
import Simplex.Chat.Controller (ChatConfig (..), ChatController (..), ChatDatabase (..))
|
||||||
import Simplex.Chat.Core
|
import Simplex.Chat.Core
|
||||||
import Simplex.Chat.Options
|
import Simplex.Chat.Options
|
||||||
import Simplex.Chat.Store
|
import Simplex.Chat.Store
|
||||||
@ -104,23 +104,21 @@ testCfgV1 = testCfg {agentConfig = testAgentCfgV1}
|
|||||||
|
|
||||||
createTestChat :: ChatConfig -> ChatOpts -> String -> Profile -> IO TestCC
|
createTestChat :: ChatConfig -> ChatOpts -> String -> Profile -> IO TestCC
|
||||||
createTestChat cfg opts@ChatOpts {dbKey} dbPrefix profile = do
|
createTestChat cfg opts@ChatOpts {dbKey} dbPrefix profile = do
|
||||||
let dbFilePrefix = testDBPrefix <> dbPrefix
|
db@ChatDatabase {chatStore} <- createChatDatabase (testDBPrefix <> dbPrefix) dbKey False
|
||||||
st <- createChatStore (dbFilePrefix <> "_chat.db") dbKey False
|
Right user <- withTransaction chatStore $ \db' -> runExceptT $ createUser db' profile True
|
||||||
Right user <- withTransaction st $ \db -> runExceptT $ createUser db profile True
|
startTestChat_ db cfg opts user
|
||||||
startTestChat_ st cfg opts dbFilePrefix user
|
|
||||||
|
|
||||||
startTestChat :: ChatConfig -> ChatOpts -> String -> IO TestCC
|
startTestChat :: ChatConfig -> ChatOpts -> String -> IO TestCC
|
||||||
startTestChat cfg opts@ChatOpts {dbKey} dbPrefix = do
|
startTestChat cfg opts@ChatOpts {dbKey} dbPrefix = do
|
||||||
let dbFilePrefix = testDBPrefix <> dbPrefix
|
db@ChatDatabase {chatStore} <- createChatDatabase (testDBPrefix <> dbPrefix) dbKey False
|
||||||
st <- createChatStore (dbFilePrefix <> "_chat.db") dbKey False
|
Just user <- find activeUser <$> withTransaction chatStore getUsers
|
||||||
Just user <- find activeUser <$> withTransaction st getUsers
|
startTestChat_ db cfg opts user
|
||||||
startTestChat_ st cfg opts dbFilePrefix user
|
|
||||||
|
|
||||||
startTestChat_ :: SQLiteStore -> ChatConfig -> ChatOpts -> FilePath -> User -> IO TestCC
|
startTestChat_ :: ChatDatabase -> ChatConfig -> ChatOpts -> User -> IO TestCC
|
||||||
startTestChat_ st cfg opts dbFilePrefix user = do
|
startTestChat_ db cfg opts user = do
|
||||||
t <- withVirtualTerminal termSettings pure
|
t <- withVirtualTerminal termSettings pure
|
||||||
ct <- newChatTerminal t
|
ct <- newChatTerminal t
|
||||||
cc <- newChatController st (Just user) cfg opts {dbFilePrefix} Nothing -- no notifications
|
cc <- newChatController db (Just user) cfg opts Nothing -- no notifications
|
||||||
chatAsync <- async . runSimplexChat opts user cc . const $ runChatTerminal ct
|
chatAsync <- async . runSimplexChat opts user cc . const $ runChatTerminal ct
|
||||||
atomically . unless (maintenance opts) $ readTVar (agentAsync cc) >>= \a -> when (isNothing a) retry
|
atomically . unless (maintenance opts) $ readTVar (agentAsync cc) >>= \a -> when (isNothing a) retry
|
||||||
termQ <- newTQueueIO
|
termQ <- newTQueueIO
|
||||||
|
@ -73,9 +73,8 @@ parsedMarkdown = "{\"formattedText\":[{\"format\":{\"type\":\"bold\"},\"text\":\
|
|||||||
|
|
||||||
testChatApiNoUser :: IO ()
|
testChatApiNoUser :: IO ()
|
||||||
testChatApiNoUser = withTmpFiles $ do
|
testChatApiNoUser = withTmpFiles $ do
|
||||||
DBMOk <- chatMigrateDB testDBPrefix ""
|
Right cc <- chatMigrateInit testDBPrefix ""
|
||||||
cc <- chatInit testDBPrefix
|
Left (DBMErrorNotADatabase _) <- chatMigrateInit testDBPrefix "myKey"
|
||||||
DBMErrorNotADatabase _ <- chatMigrateDB testDBPrefix "myKey"
|
|
||||||
chatSendCmd cc "/u" `shouldReturn` noActiveUser
|
chatSendCmd cc "/u" `shouldReturn` noActiveUser
|
||||||
chatSendCmd cc "/_start" `shouldReturn` noActiveUser
|
chatSendCmd cc "/_start" `shouldReturn` noActiveUser
|
||||||
chatSendCmd cc "/u alice Alice" `shouldReturn` activeUser
|
chatSendCmd cc "/u alice Alice" `shouldReturn` activeUser
|
||||||
@ -87,10 +86,9 @@ testChatApi = withTmpFiles $ do
|
|||||||
f = chatStoreFile dbPrefix
|
f = chatStoreFile dbPrefix
|
||||||
st <- createChatStore f "myKey" True
|
st <- createChatStore f "myKey" True
|
||||||
Right _ <- withTransaction st $ \db -> runExceptT $ createUser db aliceProfile True
|
Right _ <- withTransaction st $ \db -> runExceptT $ createUser db aliceProfile True
|
||||||
DBMOk <- chatMigrateDB testDBPrefix "myKey"
|
Right cc <- chatMigrateInit dbPrefix "myKey"
|
||||||
cc <- chatInitKey dbPrefix "myKey"
|
Left (DBMErrorNotADatabase _) <- chatMigrateInit dbPrefix ""
|
||||||
DBMErrorNotADatabase _ <- chatMigrateDB testDBPrefix ""
|
Left (DBMErrorNotADatabase _) <- chatMigrateInit dbPrefix "anotherKey"
|
||||||
DBMErrorNotADatabase _ <- chatMigrateDB testDBPrefix "anotherKey"
|
|
||||||
chatSendCmd cc "/u" `shouldReturn` activeUser
|
chatSendCmd cc "/u" `shouldReturn` activeUser
|
||||||
chatSendCmd cc "/u alice Alice" `shouldReturn` activeUserExists
|
chatSendCmd cc "/u alice Alice" `shouldReturn` activeUserExists
|
||||||
chatSendCmd cc "/_start" `shouldReturn` chatStarted
|
chatSendCmd cc "/_start" `shouldReturn` chatStarted
|
||||||
|
Loading…
Reference in New Issue
Block a user