From ff8c49854e6816928f70942cac9d3a03c5d4534f Mon Sep 17 00:00:00 2001 From: Simon Laux Date: Fri, 1 Nov 2024 15:21:23 +0100 Subject: [PATCH 1/2] feat: add support for locked/encrypted accounts to jsonrpc api api!: jsonrpc new variant of `Account` enum: `Account::Locked`. --- deltachat-jsonrpc/src/api/types/account.rs | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/deltachat-jsonrpc/src/api/types/account.rs b/deltachat-jsonrpc/src/api/types/account.rs index a5f14240ca..5fded3f484 100644 --- a/deltachat-jsonrpc/src/api/types/account.rs +++ b/deltachat-jsonrpc/src/api/types/account.rs @@ -23,10 +23,15 @@ pub enum Account { }, #[serde(rename_all = "camelCase")] Unconfigured { id: u32 }, + #[serde(rename_all = "camelCase")] + Locked { id: u32 }, } impl Account { pub async fn from_context(ctx: &deltachat::context::Context, id: u32) -> Result { + if !ctx.is_open().await { + return Ok(Account::Locked { id }); + } if ctx.is_configured().await? { let display_name = ctx.get_config(Config::Displayname).await?; let addr = ctx.get_config(Config::Addr).await?; From 659c2a071b7e798df4f063ef33b4d2e0b3e669e4 Mon Sep 17 00:00:00 2001 From: Simon Laux Date: Fri, 1 Nov 2024 15:41:34 +0100 Subject: [PATCH 2/2] api: jsonrpc: add `is_open(account_id: u32)`, `add_closed_account()`, `change_passphrase(account_id: u32, passphrase: String)` and `open(account_id: u32, passphrase: String)` --- deltachat-jsonrpc/src/api.rs | 40 ++++++++++++++++++++++++++++++++++++ 1 file changed, 40 insertions(+) diff --git a/deltachat-jsonrpc/src/api.rs b/deltachat-jsonrpc/src/api.rs index 92d4804e51..6e0cfbb999 100644 --- a/deltachat-jsonrpc/src/api.rs +++ b/deltachat-jsonrpc/src/api.rs @@ -184,10 +184,20 @@ impl CommandApi { // Account Management // --------------------------------------------- + /// Adds a new account. async fn add_account(&self) -> Result { self.accounts.write().await.add_account().await } + /// Adds a new closed/encrypted account. + /// + /// To use it it must be opened first with `open`. + /// The first call to `open` after the creation will set the passphrase. + /// It can later be changed with `change_passphrase`. + async fn add_closed_account(&self) -> Result { + self.accounts.write().await.add_closed_account().await + } + /// Imports/migrated an existing account from a database path into this account manager. /// Returns the ID of new account. async fn migrate_account(&self, path_to_db: String) -> Result { @@ -337,6 +347,36 @@ impl CommandApi { ctx.is_configured().await } + /// Checks if the database of the context is open (encrypted accounts are closed unless opened with their passphrase). + /// + /// Closed accounts need to be opened with `open(account_id: u32, passphrase: String)` + async fn is_open(&self, account_id: u32) -> Result { + let ctx = self.get_context(account_id).await?; + Ok(ctx.is_open().await) + } + + /// Changes the passphrase on the open database. + /// + /// Existing database must already be encrypted and the passphrase cannot be empty. + /// + /// It is impossible to encrypt unencrypted database with this method and vice versa. + async fn change_passphrase(&self, account_id: u32, passphrase: String) -> Result<()> { + let ctx = self.get_context(account_id).await?; + ctx.change_passphrase(passphrase).await + } + + /// Opens the database with the given passphrase. + /// + /// This can only be used on closed context, such as created by `add_closed_account`. + /// If the database is new, this operation sets the database passphrase. + /// For existing databases the passphrase should be the one used to encrypt the database the first time. + /// + /// Returns true if passphrase is correct, false is passphrase is not correct. Fails on other errors. + async fn open(&self, account_id: u32, passphrase: String) -> Result { + let ctx = self.get_context(account_id).await?; + ctx.open(passphrase).await + } + /// Get system info for an account. async fn get_info(&self, account_id: u32) -> Result> { let ctx = self.get_context(account_id).await?;