Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(walletconnect): walletconnect integration #2223

Open
wants to merge 172 commits into
base: dev
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
172 commits
Select commit Hold shift + click to select a range
9100780
init, start walletconnect client impl
borngraced Sep 9, 2024
ca3f558
save dev state - walletconnect client impl
borngraced Sep 9, 2024
c3dc6c6
save dev state - WalletConnect client impl
borngraced Sep 9, 2024
0eea639
rough implementations
borngraced Sep 11, 2024
588316f
commit lock file
borngraced Sep 11, 2024
60a2fc5
save dev state - handle connection close, session improvements, etc
borngraced Sep 12, 2024
fd8797c
minor changes
borngraced Sep 12, 2024
437767b
fix wasm compilation
borngraced Sep 13, 2024
6374da3
minor changes - create metadata mod
borngraced Sep 13, 2024
1c9cfe0
minor changes and code organization
borngraced Sep 14, 2024
f7987dd
minor changes + merge session and session_key mod
borngraced Sep 14, 2024
1f2adb2
minor changes to namespaces
borngraced Sep 15, 2024
658357f
some improvements
borngraced Sep 16, 2024
98026ac
add some session fn doc comments
borngraced Sep 16, 2024
9dddb91
big improvements to session handling
borngraced Sep 16, 2024
52a0f72
implement session delete and cleaup
borngraced Sep 17, 2024
9f5944c
improve code and locking
borngraced Sep 17, 2024
dff1b42
remove unused dep
borngraced Sep 17, 2024
591fbdb
refactorings and cleanupds
borngraced Sep 17, 2024
f6e7864
minor changes
borngraced Sep 18, 2024
f535f0b
make session single
borngraced Sep 19, 2024
3eb8969
minor changes
borngraced Sep 19, 2024
2d68488
save dev state
borngraced Sep 19, 2024
42dd5ba
implement coin activation for tendermint and other minor changes
borngraced Sep 21, 2024
edcb5a7
improve code and minor needed changes
borngraced Sep 22, 2024
e606ca4
improve relayer disconnection, remove unneeded changes, track topic s…
borngraced Sep 23, 2024
d7b5a17
fix tendermint pubkey derivation from walletconnect
borngraced Sep 23, 2024
3a85a17
start persistent session storage implementation
borngraced Sep 23, 2024
be98c40
implement sqlite session storage table and insert session method
borngraced Sep 24, 2024
6453a33
fix cyclic deps and continue storage implementations
borngraced Sep 25, 2024
9f87a9e
implement WASM persistent storage
borngraced Sep 26, 2024
6135665
implement persistent indexed_db session storage
borngraced Sep 26, 2024
5457d80
save dev state - implement wc rpc commands
borngraced Sep 26, 2024
9d12079
move walletconnect rpc to mm2_main rpc
borngraced Sep 26, 2024
03d9978
remove mm2_test_helpers lock file
borngraced Sep 26, 2024
32e46d9
format mm2_net cargo.toml
borngraced Sep 26, 2024
1e352b1
update storage session upon session settle response/request
borngraced Sep 26, 2024
a30511b
add tokio to wc dependency list
borngraced Sep 26, 2024
341b18e
improve tenderming with_pubkey activation params
borngraced Sep 26, 2024
4c8e299
minor changes
borngraced Sep 29, 2024
73d8315
fix proposer mod
borngraced Sep 29, 2024
c1accc9
merge with dev
borngraced Sep 29, 2024
7ab33d5
implement walletconnect sign tx for tendermint - wip
borngraced Oct 2, 2024
0bcd41c
remove chain params from get cosmos account method
borngraced Oct 2, 2024
b2ec309
tx impl - wip
borngraced Oct 2, 2024
2e25bde
save dev state
borngraced Oct 6, 2024
328bc76
handle client connnection/reconnection efficiently and minor renaming
borngraced Oct 6, 2024
9b20d7f
minor changes
borngraced Oct 6, 2024
ca2e19c
implement multi session management
borngraced Oct 8, 2024
481f1c3
remove mm2_test_helpers lock file
borngraced Oct 8, 2024
0c00710
refactoring and unit tests
borngraced Oct 8, 2024
e5e2150
fix session topic generation bug
borngraced Oct 8, 2024
8d749de
tendermint sign tx impl fixes
borngraced Oct 9, 2024
8b1b6c3
improve wc tx_signing codes
borngraced Oct 9, 2024
4202950
minor fix
borngraced Oct 9, 2024
d440ed9
add more session rpc endpoint
borngraced Oct 10, 2024
f3cf28b
update wc deps
borngraced Oct 10, 2024
e56f8d7
update wc_common deps
borngraced Oct 10, 2024
1ab21d7
merge with dev and fix conflicts
borngraced Oct 10, 2024
a498381
use rustls
borngraced Oct 10, 2024
c7b41a8
update bip39 deps
borngraced Oct 11, 2024
8fc7620
fix wasm clippy
borngraced Oct 11, 2024
2d52404
add chains/tendermint test
borngraced Oct 11, 2024
4cbd2fa
minor changes, improve api
borngraced Oct 14, 2024
a57495d
remove subscription on session delete
borngraced Oct 14, 2024
c7e7de6
improve ledger external wallet connection checks
borngraced Oct 14, 2024
cc3c568
wc tx handling for tendermint
borngraced Oct 14, 2024
7683090
merge with dev and fix conflicts
borngraced Oct 14, 2024
5a574a6
complete persistent session storage with unit test for native and wasm
borngraced Oct 15, 2024
81ebee7
minor changes
borngraced Oct 15, 2024
46e3802
fix is ledger connection fn
borngraced Oct 17, 2024
500cb87
module refactorings
borngraced Oct 19, 2024
b2dea85
Merge branch 'dev' of github.com:KomodoPlatform/komodo-defi-framework…
borngraced Oct 20, 2024
1840ca8
introduce WcRequstOps trait, untighten code and more
borngraced Oct 21, 2024
6501134
commit cargo.lock
borngraced Oct 21, 2024
38dadbe
implement cosmos ledger amino sign support
borngraced Oct 21, 2024
301a5b0
fix fmt
borngraced Oct 21, 2024
3cde05b
start eth pubkey mode impl
borngraced Oct 22, 2024
5eb3e4c
save progress - impl wc eth_personal sign
borngraced Oct 22, 2024
b35bd4a
implement eth coin activation using walletconnect
borngraced Oct 23, 2024
337154c
eth walletconnect coin activation and message handler code improvements
borngraced Oct 24, 2024
b358825
improve session message handling process
borngraced Oct 24, 2024
afc4058
further wc improve message handling
borngraced Oct 24, 2024
18eb62f
uncommit mm2_test_helpers lock file
borngraced Oct 24, 2024
9c083ac
eth wc transaction - save dev state
borngraced Oct 25, 2024
7f37fd3
impl eth withdraw for walletconnect policy
borngraced Oct 26, 2024
6e57330
don't support sign_raw_tx rpc in walletconnect mode
borngraced Oct 26, 2024
e45243b
cleanups
borngraced Oct 27, 2024
a8d937e
improvements to session, wc ops, and more
borngraced Oct 28, 2024
1569ede
create connection with custom namespaces
borngraced Oct 28, 2024
4fc5a43
impl eth sign and send tx for WalletConnect
borngraced Oct 31, 2024
3792027
swaps working, impl connection health manager
borngraced Nov 1, 2024
0b1da04
remove debug log
borngraced Nov 1, 2024
f0659a4
connection_handler improvements
borngraced Nov 1, 2024
187bf5e
improve tendermint wc code
borngraced Nov 2, 2024
2a2f167
use const_hex
borngraced Nov 2, 2024
cd368bf
major changes to connection handling and other minor fixes
borngraced Nov 3, 2024
0c61287
add logging to wc storage impl
borngraced Nov 3, 2024
fe0d096
Merge branch 'refs/heads/dev' into wc-integration
borngraced Nov 3, 2024
a0220d7
fix conflict and minor changes + merge with dev branch
borngraced Nov 3, 2024
3468502
minor changes
borngraced Nov 4, 2024
c4a07e4
monitor if relay client is connected
borngraced Nov 4, 2024
1e78257
changes to client impls
borngraced Nov 6, 2024
a43dc60
handle broadcast option for WalletConnect and some minor fixes
borngraced Nov 6, 2024
eea1532
remove unused
borngraced Nov 6, 2024
fb9f8c1
fix: handle unexpected end of file
borngraced Nov 6, 2024
d58b18f
re-enable use_watcher
borngraced Nov 6, 2024
045625b
cleanups
borngraced Nov 6, 2024
65c7f90
fix websocket connection drop
borngraced Nov 12, 2024
2326959
handle conn drop for session creation
borngraced Nov 12, 2024
42eaa29
nits
borngraced Nov 12, 2024
4ad423e
Merge branch 'refs/heads/dev' into wc-integration
borngraced Nov 12, 2024
e82083a
fix conflict after merge
borngraced Nov 12, 2024
ad38241
nits
borngraced Nov 12, 2024
59db165
move storage to session manager and minor improvements
borngraced Nov 13, 2024
f5caa19
fix review notes
borngraced Nov 13, 2024
bb186c2
fix: self review
borngraced Nov 15, 2024
77d7909
fix review notes
borngraced Nov 16, 2024
0d3fdfe
increase rpc timeout and return PublicKey from recover
borngraced Nov 18, 2024
35d0fa4
fix review notes
borngraced Nov 18, 2024
db1e687
fmt
borngraced Nov 18, 2024
11d114d
remove dup session delete rpc
borngraced Nov 19, 2024
af91d47
rename var and remove unnecessary sorting from storage sessions loading
borngraced Nov 19, 2024
10e9e67
refactor wait_for_tx_appears_on_rpc to use secs for time
borngraced Nov 19, 2024
b9075ce
batch subscribe to topics
borngraced Nov 26, 2024
1ff17cd
rename ctx.session to ctx.session_manager
borngraced Nov 26, 2024
5e89b9f
use weak spawner to spawn wc related task/event loops
borngraced Nov 27, 2024
1fec24f
validate table name only during initialization
borngraced Nov 27, 2024
bdf9636
make required_namespaces required for wc activation and remove defaults
borngraced Nov 27, 2024
31c9594
use sync mutex for active topic
borngraced Nov 27, 2024
37a36f7
fix review notes
borngraced Nov 28, 2024
0351d67
Merge branch 'dev' into wc-integration
borngraced Nov 28, 2024
21f42f0
nits
borngraced Dec 3, 2024
c743464
bump wc libs
borngraced Dec 3, 2024
cab1b36
remove todo comment, replace remove const-hex crate
borngraced Dec 5, 2024
4e82c4f
remove dashmap
borngraced Dec 6, 2024
b96881e
revert mm2-parity-eth lib ver
borngraced Dec 6, 2024
738f941
review deps, add todo for WC ctx initialization
borngraced Dec 8, 2024
19b1d8a
reverse lock_api deps
borngraced Dec 8, 2024
9f73688
remove session request timestamp and add doc comments to message hand…
borngraced Dec 9, 2024
b874221
fix fmt
borngraced Dec 9, 2024
26ea297
review deps
borngraced Dec 9, 2024
371af58
use exponential backoff in spawn_connection_initializations reconnection
borngraced Dec 9, 2024
fdc18d7
remove redundancy and fix nits
borngraced Dec 9, 2024
e3d9a9f
Merge branch 'dev' into wc-integration
borngraced Dec 9, 2024
5f47368
fix notes and graceful shutdown on init failure
borngraced Dec 10, 2024
1860776
nits
borngraced Dec 10, 2024
27cf3fb
track message via message_id
borngraced Dec 11, 2024
06e4dc7
impl correlated message handling for inbound messages
borngraced Dec 11, 2024
25275e9
remove println
borngraced Dec 12, 2024
89756db
revert session property deserializing
borngraced Dec 12, 2024
a2c7c71
nits
borngraced Dec 12, 2024
bbddec5
Merge branch 'dev' into wc-integration
borngraced Dec 13, 2024
9db99ac
fix merge conflict
borngraced Dec 13, 2024
4739ec3
self review fixes
borngraced Dec 24, 2024
414fda5
nits
borngraced Dec 25, 2024
f95e385
bump WalletConnectRust lib
borngraced Dec 26, 2024
dd1a842
save dev state - implement dynamic session management for activated c…
borngraced Dec 31, 2024
4dd315a
complete coin with session topic
borngraced Jan 1, 2025
3b07397
improve and remove redundancy session_properties call in tendermint w…
borngraced Jan 2, 2025
5b4838d
minor changes
borngraced Jan 3, 2025
7124b4f
marge with dev
borngraced Jan 3, 2025
2243c9d
revert metadata to fn
borngraced Jan 3, 2025
1a8f243
move u256_to_hex fn to common mod and minor changes
borngraced Jan 6, 2025
020ecb8
replace EthPrivKeyActivationPolicy name for WC des
borngraced Jan 8, 2025
432b056
remove active_session/topic and some other minor changes
borngraced Jan 9, 2025
bb73b1f
reduce lookup for session in session/settle.rs
borngraced Jan 12, 2025
ed7b4d2
fix review notes
borngraced Jan 17, 2025
e5fa7a4
Merge branch 'dev' into wc-integration
borngraced Jan 17, 2025
45a5961
fix review notes
borngraced Jan 20, 2025
c697e3c
refactor get_full_sessions to get_sessions_topic_and_controller
borngraced Jan 20, 2025
e39507b
use timed_map
borngraced Jan 23, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
major changes to connection handling and other minor fixes
  • Loading branch information
borngraced committed Nov 3, 2024
commit cd368bfb51dad9a10fc88c57c6d9b651008a8000
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

34 changes: 15 additions & 19 deletions mm2src/coins/eth/wallet_connect.rs
Original file line number Diff line number Diff line change
Expand Up @@ -102,24 +102,22 @@ impl WalletConnectOps for EthCoin {
type SendTxData = (SignedTransaction, BytesJson);
type Params<'a> = WcEthTxParams<'a>;

async fn wc_chain_id(&self, ctx: &WalletConnectCtx) -> Result<WcChainId, Self::Error> {
let chain = WcChainId::new_eip155(self.chain_id.to_string());
if ctx.is_chain_supported(&chain).await {
return MmError::err(WalletConnectError::ChainIdNotSupported(chain.to_string()).into());
};
async fn wc_chain_id(&self, wc: &WalletConnectCtx) -> Result<WcChainId, Self::Error> {
let chain_id = WcChainId::new_eip155(self.chain_id.to_string());
wc.validate_update_active_chain_id(&chain_id).await?;

Ok(chain)
Ok(chain_id)
}

async fn wc_sign_tx<'a>(
&self,
ctx: &WalletConnectCtx,
wc: &WalletConnectCtx,
params: Self::Params<'a>,
) -> Result<Self::SignTxData, Self::Error> {
let bytes = {
let chain_id = self.wc_chain_id(ctx).await?;
let chain_id = self.wc_chain_id(wc).await?;
let tx_json = params.prepare_wc_tx_format()?;
let tx_hex: String = ctx
let tx_hex: String = wc
.send_session_request_and_wait(&chain_id, WcRequestMethods::EthSignTransaction, tx_json, Ok)
.await?;
// First 4 bytes from WalletConnect represents Protoc info
Expand All @@ -135,13 +133,13 @@ impl WalletConnectOps for EthCoin {

async fn wc_send_tx<'a>(
&self,
ctx: &WalletConnectCtx,
wc: &WalletConnectCtx,
params: Self::Params<'a>,
) -> Result<Self::SignTxData, Self::Error> {
let tx_hash: String = {
let chain_id = self.wc_chain_id(ctx).await?;
let chain_id = self.wc_chain_id(wc).await?;
let tx_json = params.prepare_wc_tx_format()?;
ctx.send_session_request_and_wait(&chain_id, WcRequestMethods::EthSendTransaction, tx_json, Ok)
wc.send_session_request_and_wait(&chain_id, WcRequestMethods::EthSendTransaction, tx_json, Ok)
.await?
};
let tx_hash = tx_hash.strip_prefix("0x").unwrap_or(&tx_hash);
Expand Down Expand Up @@ -175,20 +173,18 @@ impl WalletConnectOps for EthCoin {
}

pub async fn eth_request_wc_personal_sign(
ctx: &WalletConnectCtx,
wc: &WalletConnectCtx,
chain_id: u64,
) -> MmResult<(H520, Address), EthWalletConnectError> {
let chain_id = WcChainId::new_eip155(chain_id.to_string());
if ctx.is_chain_supported(&chain_id).await {
return MmError::err(WalletConnectError::ChainIdNotSupported(chain_id.to_string()).into());
};
wc.validate_update_active_chain_id(&chain_id).await?;

let result = {
let account_str = ctx.get_account_for_chain_id(&chain_id).await?;
let account_str = wc.get_account_for_chain_id(&chain_id).await?;
let message = "Authenticate with Komodefi";
let message_hex = format!("0x{}", hex::encode(message));
let params = json!(&[&message_hex, &account_str]);
ctx.send_session_request_and_wait(&chain_id, WcRequestMethods::PersonalSign, params, |data: String| {
wc.send_session_request_and_wait(&chain_id, WcRequestMethods::PersonalSign, params, |data: String| {
Ok(extract_pubkey_from_signature(&data, message, &account_str)?)
})
.await?
Expand Down Expand Up @@ -255,7 +251,6 @@ pub(crate) async fn send_transaction_with_walletconnect(
coin.get_swap_pay_for_gas_option(coin.get_swap_transaction_fee_policy())
.await
);

let (nonce, _) = try_tx_s!(coin.clone().get_addr_nonce(address).compat().await);
let params = WcEthTxParams {
gas,
Expand All @@ -268,6 +263,7 @@ pub(crate) async fn send_transaction_with_walletconnect(
};
// Please note that this method may take a long time
// due to `eth_sendTransaction` requests.
info!(target: "sign-and-send", "wallet signing and sending tx…");
let (signed_tx, _) = try_tx_s!(coin.wc_send_tx(wc, params).await);

Ok(signed_tx)
Expand Down
10 changes: 4 additions & 6 deletions mm2src/coins/tendermint/wallet_connect.rs
Original file line number Diff line number Diff line change
Expand Up @@ -80,12 +80,10 @@ impl WalletConnectOps for TendermintCoin {
type SendTxData = CosmosTransaction;

async fn wc_chain_id(&self, wc: &WalletConnectCtx) -> Result<WcChainId, Self::Error> {
let chain = WcChainId::new_cosmos(self.chain_id.to_string());
if !wc.is_chain_supported(&chain).await {
return MmError::err(WalletConnectError::ChainIdNotSupported(chain.to_string()));
};
let chain_id = WcChainId::new_cosmos(self.chain_id.to_string());
wc.validate_update_active_chain_id(&chain_id).await?;

Ok(chain)
Ok(chain_id)
}

async fn wc_sign_tx<'a>(
Expand Down Expand Up @@ -127,7 +125,7 @@ pub async fn cosmos_get_accounts_impl(
chain_id: &str,
) -> MmResult<CosmosAccount, WalletConnectError> {
let chain_id = WcChainId::new_cosmos(chain_id.to_string());
wc.validate_or_update_active_chain_id(&chain_id).await?;
wc.validate_update_active_chain_id(&chain_id).await?;

let account = wc.get_account_for_chain_id(&chain_id).await?;
let session = wc
Expand Down
3 changes: 2 additions & 1 deletion mm2src/kdf_walletconnect/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -23,17 +23,18 @@ dashmap = "6.1.0"
mm2_core = { path = "../mm2_core" }
mm2_db = { path = "../mm2_db" }
mm2_err_handle = { path = "../mm2_err_handle" }
parking_lot = { version = "0.12.0", features = ["nightly"] }
pairing_api = { git = "https://github.com/komodoplatform/walletconnectrust", branch = "pairing-api" }
rand = "0.8"
relay_client = { git = "https://github.com/komodoplatform/walletconnectrust", branch = "pairing-api" }
relay_rpc = { git = "https://github.com/komodoplatform/walletconnectrust", branch = "pairing-api" }
sha2 = "0.10.7"
thiserror = "1.0.40"
tokio = { version = "1.20" }
wc_common = { git = "https://github.com/komodoplatform/walletconnectrust", branch = "pairing-api" }
secp256k1 = { version = "0.20" }
serde = { version = "1.0", features = ["derive"] }
serde_json = { version = "1", features = ["preserve_order", "raw_value"] }
sha2 = "0.10.7"
x25519-dalek = { version = "2.0", features = ["static_secrets"] }

[target.'cfg(target_arch = "wasm32")'.dependencies]
Expand Down
16 changes: 12 additions & 4 deletions mm2src/kdf_walletconnect/src/chain.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
use mm2_err_handle::prelude::{MmError, MmResult};
use relay_rpc::rpc::params::session::{ProposeNamespace, ProposeNamespaces};
use serde::{Deserialize, Serialize};
use std::{collections::{BTreeMap, BTreeSet},
str::FromStr};

Expand All @@ -14,7 +15,7 @@ pub(crate) const ETH_SUPPORTED_METHODS: &[&str] = &["eth_signTransaction", "pers
pub(crate) const ETH_SUPPORTED_CHAINS: &[&str] = &["eip155:1", "eip155:137"];
pub(crate) const ETH_SUPPORTED_EVENTS: &[&str] = &["accountsChanged", "chainChanged"];

#[derive(Debug)]
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
pub struct WcChainId {
pub chain: WcChain,
pub id: String,
Expand Down Expand Up @@ -52,11 +53,9 @@ impl WcChainId {
id: sp[1].to_owned(),
})
}

pub(crate) fn chain_id_from_id(&self, id: &str) -> String { format!("{}:{}", self.chain.as_ref(), id) }
}

#[derive(Debug)]
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
pub enum WcChain {
Eip155,
Cosmos,
Expand Down Expand Up @@ -84,6 +83,15 @@ impl AsRef<str> for WcChain {
}
}

impl WcChain {
pub(crate) fn derive_chain_id(&self, id: String) -> WcChainId {
WcChainId {
chain: self.clone(),
id,
}
}
}

#[derive(Debug, Clone)]
pub enum WcRequestMethods {
CosmosSignDirect,
Expand Down
124 changes: 17 additions & 107 deletions mm2src/kdf_walletconnect/src/connection_handler.rs
Original file line number Diff line number Diff line change
@@ -1,24 +1,20 @@
use std::sync::Arc;

use crate::storage::WalletConnectStorageOps;
use crate::{WalletConnectCtx, WalletConnectError};

use common::executor::Timer;
use common::log::{debug, error, info};
use common::log::{error, info};
use futures::channel::mpsc::UnboundedSender;
use futures::StreamExt;
use mm2_err_handle::prelude::*;
use relay_client::error::ClientError;
use relay_client::websocket::{CloseFrame, ConnectionHandler, PublishedMessage};
use relay_rpc::domain::Topic;
use relay_rpc::rpc::params::RequestParams;
use tokio::time::interval;
use tokio::time::Duration;

const INITIAL_RETRY_SECS: f64 = 5.0;
const RETRY_INCREMENT: f64 = 5.0;
const MAX_BACKOFF: u64 = 60;
const PING_INTERVAL: u64 = 140;
const PING_TIMEOUT: f64 = 60.;
const RETRY_INCREMENT: f64 = 5.0;

pub struct Handler {
name: &'static str,
Expand Down Expand Up @@ -76,7 +72,8 @@ impl ConnectionHandler for Handler {
/// Establishes initial connection to WalletConnect relay server with linear retry mechanism.
/// Uses increasing delay between retry attempts starting from INITIAL_RETRY_SECS.
/// After successful connection, attempts to restore previous session state from storage.
pub(crate) async fn initial_connection(this: &WalletConnectCtx) {
pub(crate) async fn initialize_connection(this: Arc<WalletConnectCtx>) {
info!("Initializing WalletConnect connection");
let mut retry_count = 0;
let mut retry_secs = INITIAL_RETRY_SECS;

Expand All @@ -90,12 +87,20 @@ pub(crate) async fn initial_connection(this: &WalletConnectCtx) {
retry_secs += RETRY_INCREMENT;
mariocynicys marked this conversation as resolved.
Show resolved Hide resolved
}

debug!("Successfully connected to client after {} attempt(s).", retry_count + 1);
info!("Successfully connected to client after {} attempt(s).", retry_count + 1);

// Initialize storage
if let Err(err) = this.storage.init().await {
info!("Unable to initialize WalletConnect persistent storage: {err:?}. Only inmemory storage will be utilized for this Session.");
};

// load session from storage
if let Err(err) = this.load_session_from_storage().await {
error!("Unable to load session from storage: {err:?}");
info!("Unable to load session from storage: {err:?}");
};

// Spawn session disconnection watcher.
handle_disconnections(&this).await;
}

/// Handles unexpected disconnections from WalletConnect relay server.
Expand All @@ -106,15 +111,15 @@ pub(crate) async fn handle_disconnections(this: &WalletConnectCtx) {
let mut backoff = 1;

while let Some(_msg) = recv.next().await {
debug!("Connection disconnected. Attempting to reconnect...");
info!("Connection disconnected. Attempting to reconnect...");

loop {
match this.connect_client().await {
Ok(_) => {
if let Err(e) = resubscribe_to_topics(this, None).await {
error!("Failed to resubscribe after reconnection: {:?}", e);
}
debug!("Reconnection process complete.");
info!("Reconnection process complete.");
backoff = 1;
break;
},
Expand All @@ -128,101 +133,6 @@ pub(crate) async fn handle_disconnections(this: &WalletConnectCtx) {
}
}

/// Maintains connection health with WalletConnect relay server through periodic pings.
/// Sends a ping every 4 minutes to proactively detect disconnections and automatically reconnects if needed.
/// This helps prevent connection drops as the relay server tends to disconnect inactive connections after 5 minutes.
/// The ping interval is designed to catch potential disconnections before the server timeout while minimizing
/// unnecessary network traffic.
pub(crate) async fn keep_session_alive_ping(ctx: Arc<WalletConnectCtx>) {
info!("keep_session_alive_ping running");
let mut interval = interval(Duration::from_secs(PING_INTERVAL));
let mut backoff = 1.;

// Skip the first tick which happens immediately
interval.tick().await;

loop {
let Ok(session_topic) = ctx.session.get_active_topic_or_err().await else {
info!("No active session to ping will check again shortly");
Timer::sleep(PING_INTERVAL as f64).await;
continue;
};
tokio::select! {
_ = interval.tick() => {
match try_ping(&ctx, &session_topic).await {
Ok(_) => {
backoff = 1.;
continue;
}
Err(e) => {
error!("WalletConnect ping failed: {:?}. Attempting reconnect...", e);
// Attempt reconnection with backoff
if let Err(reconnect_err) = handle_ping_failure(&ctx, &session_topic).await {
error!("{} Session reconnection failed: {:?}", session_topic, reconnect_err);
// Increase backoff for next attempt
backoff = std::cmp::min(backoff as u64 * 2, MAX_BACKOFF ) as f64;
Timer::sleep(backoff).await;
continue;
}
// Reset backoff after successful reconnection
backoff = 1.;
debug!("{}: Successfully reconnected after ping failure", session_topic);
}
}
}
}
}
}

/// Attempts to verify connection health with WalletConnect relay server by sending a ping message.
/// Waits for a pong response with a specified timeout period.
async fn try_ping(ctx: &WalletConnectCtx, session_topic: &Topic) -> MmResult<(), WalletConnectError> {
let param = RequestParams::SessionPing(());
ctx.publish_request(session_topic, param).await?;

let timeout = Timer::sleep(PING_TIMEOUT);
tokio::pin!(timeout);

let mut recv = ctx.message_rx.lock().await;
tokio::select! {
msg = recv.next() => {
match msg {
Some(Ok(_)) => {
debug!("WalletConnect {}: Session is alive", session_topic);
Ok(())
},
Some(Err(err)) => {
MmError::err(WalletConnectError::InternalError(
format!("WalletConnect {}: Ping Error: {err:?}", session_topic)
))
},
None => {
error!("WalletConnect Ping timeout");
MmError::err(WalletConnectError::InternalError(
"WalletConnect Ping timeout".into()
))
}
}
},
_ = timeout => {
error!("WalletConnect {}: Ping timeout", session_topic);
MmError::err(WalletConnectError::InternalError(
"WalletConnect Ping timeout".into()
))
}
}
}

// Handle reconnection after ping failure
async fn handle_ping_failure(ctx: &WalletConnectCtx, session_topic: &Topic) -> MmResult<(), WalletConnectError> {
// Attempt to reconnect
ctx.connect_client().await?;
// Resubscribe to topics if needed
resubscribe_to_topics(ctx, Some(session_topic)).await?;
// Verify connection with a new ping
try_ping(ctx, session_topic).await
}

/// Resubscribes to previously active topics after reconnection.
/// Called by handle_disconnections to restore subscription state.
async fn resubscribe_to_topics(this: &WalletConnectCtx, topic: Option<&Topic>) -> MmResult<(), WalletConnectError> {
Expand Down
Loading
Loading