Compare commits
1 Commits
f3cf6a9438
...
impl-usera
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
3aae3e1d83 |
@@ -1,24 +1,26 @@
|
||||
use alloy::{consensus::TxEip1559, primitives::Address, signers::Signature};
|
||||
use diesel::{
|
||||
ExpressionMethods, OptionalExtension as _, QueryDsl, SelectableHelper as _, dsl::insert_into,
|
||||
BoolExpressionMethods as _, ExpressionMethods, OptionalExtension as _, QueryDsl,
|
||||
SelectableHelper as _, dsl::insert_into,
|
||||
};
|
||||
use diesel_async::RunQueryDsl;
|
||||
use diesel_async::{AsyncConnection as _, RunQueryDsl};
|
||||
use kameo::{Actor, actor::ActorRef, messages};
|
||||
use rand::{SeedableRng, rng, rngs::StdRng};
|
||||
|
||||
use crate::{
|
||||
actors::keyholder::{CreateNew, Decrypt, GetState, KeyHolder, KeyHolderState},
|
||||
actors::keyholder::{CreateNew, Decrypt, KeyHolder},
|
||||
crypto::integrity,
|
||||
db::{
|
||||
DatabaseError, DatabasePool,
|
||||
models::{self, SqliteTimestamp},
|
||||
models::{self},
|
||||
schema,
|
||||
},
|
||||
evm::{
|
||||
self, ListError, RunKind, policies::{
|
||||
self, ListError, RunKind,
|
||||
policies::{
|
||||
CombinedSettings, Grant, SharedGrantSettings, SpecificGrant, SpecificMeaning,
|
||||
ether_transfer::EtherTransfer, token_transfers::TokenTransfer,
|
||||
}
|
||||
},
|
||||
},
|
||||
safe_cell::{SafeCell, SafeCellHandle as _},
|
||||
};
|
||||
@@ -158,27 +160,114 @@ impl EvmActor {
|
||||
|
||||
#[message]
|
||||
pub async fn useragent_delete_grant(&mut self, grant_id: i32) -> Result<(), Error> {
|
||||
// let mut conn = self.db.get().await.map_err(DatabaseError::from)?;
|
||||
// let keyholder = self.keyholder.clone();
|
||||
let mut conn = self.db.get().await.map_err(DatabaseError::from)?;
|
||||
|
||||
// diesel_async::AsyncConnection::transaction(&mut conn, |conn| {
|
||||
// Box::pin(async move {
|
||||
// diesel::update(schema::evm_basic_grant::table)
|
||||
// .filter(schema::evm_basic_grant::id.eq(grant_id))
|
||||
// .set(schema::evm_basic_grant::revoked_at.eq(SqliteTimestamp::now()))
|
||||
// .execute(conn)
|
||||
// .await?;
|
||||
// We intentionally perform a hard delete here to avoid leaving revoked grants and their
|
||||
// related rows as long-lived DB garbage. We also don't rely on SQLite FK cascades because
|
||||
// they can be disabled per-connection.
|
||||
conn.transaction(|conn| {
|
||||
Box::pin(async move {
|
||||
// First, resolve policy-specific rows by basic grant id.
|
||||
let token_grant_id: Option<i32> = schema::evm_token_transfer_grant::table
|
||||
.select(schema::evm_token_transfer_grant::id)
|
||||
.filter(schema::evm_token_transfer_grant::basic_grant_id.eq(grant_id))
|
||||
.first::<i32>(conn)
|
||||
.await
|
||||
.optional()?;
|
||||
|
||||
// let signed = integrity::evm::load_signed_grant_by_basic_id(conn, grant_id).await?;
|
||||
let ether_grant: Option<(i32, i32)> = schema::evm_ether_transfer_grant::table
|
||||
.select((
|
||||
schema::evm_ether_transfer_grant::id,
|
||||
schema::evm_ether_transfer_grant::limit_id,
|
||||
))
|
||||
.filter(schema::evm_ether_transfer_grant::basic_grant_id.eq(grant_id))
|
||||
.first::<(i32, i32)>(conn)
|
||||
.await
|
||||
.optional()?;
|
||||
|
||||
// diesel::result::QueryResult::Ok(())
|
||||
// })
|
||||
// })
|
||||
// .await
|
||||
// .map_err(DatabaseError::from)?;
|
||||
// Token-transfer: logs must be deleted before transaction logs (FK restrict).
|
||||
if let Some(token_grant_id) = token_grant_id {
|
||||
diesel::delete(
|
||||
schema::evm_token_transfer_log::table
|
||||
.filter(schema::evm_token_transfer_log::grant_id.eq(token_grant_id)),
|
||||
)
|
||||
.execute(conn)
|
||||
.await?;
|
||||
|
||||
// Ok(())
|
||||
todo!()
|
||||
diesel::delete(schema::evm_token_transfer_volume_limit::table.filter(
|
||||
schema::evm_token_transfer_volume_limit::grant_id.eq(token_grant_id),
|
||||
))
|
||||
.execute(conn)
|
||||
.await?;
|
||||
|
||||
diesel::delete(
|
||||
schema::evm_token_transfer_grant::table
|
||||
.filter(schema::evm_token_transfer_grant::id.eq(token_grant_id)),
|
||||
)
|
||||
.execute(conn)
|
||||
.await?;
|
||||
}
|
||||
|
||||
// Shared transaction logs for any grant kind.
|
||||
diesel::delete(
|
||||
schema::evm_transaction_log::table
|
||||
.filter(schema::evm_transaction_log::grant_id.eq(grant_id)),
|
||||
)
|
||||
.execute(conn)
|
||||
.await?;
|
||||
|
||||
// Ether-transfer: delete targets, grant row, then its limit row.
|
||||
if let Some((ether_grant_id, limit_id)) = ether_grant {
|
||||
diesel::delete(schema::evm_ether_transfer_grant_target::table.filter(
|
||||
schema::evm_ether_transfer_grant_target::grant_id.eq(ether_grant_id),
|
||||
))
|
||||
.execute(conn)
|
||||
.await?;
|
||||
|
||||
diesel::delete(
|
||||
schema::evm_ether_transfer_grant::table
|
||||
.filter(schema::evm_ether_transfer_grant::id.eq(ether_grant_id)),
|
||||
)
|
||||
.execute(conn)
|
||||
.await?;
|
||||
|
||||
diesel::delete(
|
||||
schema::evm_ether_transfer_limit::table
|
||||
.filter(schema::evm_ether_transfer_limit::id.eq(limit_id)),
|
||||
)
|
||||
.execute(conn)
|
||||
.await?;
|
||||
}
|
||||
|
||||
// Integrity envelopes are not FK-constrained; delete only grant-related kinds to
|
||||
// avoid accidentally deleting other entities that share the same integer ID.
|
||||
let entity_id = grant_id.to_be_bytes().to_vec();
|
||||
diesel::delete(
|
||||
schema::integrity_envelope::table
|
||||
.filter(schema::integrity_envelope::entity_id.eq(entity_id))
|
||||
.filter(
|
||||
schema::integrity_envelope::entity_kind
|
||||
.eq("EtherTransfer")
|
||||
.or(schema::integrity_envelope::entity_kind.eq("TokenTransfer")),
|
||||
),
|
||||
)
|
||||
.execute(conn)
|
||||
.await?;
|
||||
|
||||
// Finally remove the basic grant row itself (idempotent if it doesn't exist).
|
||||
diesel::delete(
|
||||
schema::evm_basic_grant::table.filter(schema::evm_basic_grant::id.eq(grant_id)),
|
||||
)
|
||||
.execute(conn)
|
||||
.await?;
|
||||
|
||||
diesel::result::QueryResult::Ok(())
|
||||
})
|
||||
})
|
||||
.await
|
||||
.map_err(DatabaseError::from)?;
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
#[message]
|
||||
@@ -270,3 +359,6 @@ impl EvmActor {
|
||||
Ok(signer.sign_transaction_sync(&mut transaction)?)
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests;
|
||||
|
||||
283
server/crates/arbiter-server/src/actors/evm/tests.rs
Normal file
283
server/crates/arbiter-server/src/actors/evm/tests.rs
Normal file
@@ -0,0 +1,283 @@
|
||||
use diesel::{ExpressionMethods as _, QueryDsl as _, dsl::insert_into};
|
||||
use diesel_async::RunQueryDsl;
|
||||
use kameo::actor::Spawn as _;
|
||||
|
||||
use crate::{
|
||||
actors::{evm::EvmActor, keyholder::KeyHolder},
|
||||
db::{self, models, schema},
|
||||
};
|
||||
|
||||
#[tokio::test]
|
||||
async fn delete_ether_grant_cleans_related_tables() {
|
||||
let db = db::create_test_pool().await;
|
||||
let keyholder = KeyHolder::spawn(KeyHolder::new(db.clone()).await.unwrap());
|
||||
let mut actor = EvmActor::new(keyholder, db.clone());
|
||||
|
||||
let mut conn = db.get().await.unwrap();
|
||||
|
||||
let basic_id: i32 = insert_into(schema::evm_basic_grant::table)
|
||||
.values(&models::NewEvmBasicGrant {
|
||||
wallet_access_id: 1,
|
||||
chain_id: 1,
|
||||
valid_from: None,
|
||||
valid_until: None,
|
||||
max_gas_fee_per_gas: None,
|
||||
max_priority_fee_per_gas: None,
|
||||
rate_limit_count: None,
|
||||
rate_limit_window_secs: None,
|
||||
revoked_at: None,
|
||||
})
|
||||
.returning(schema::evm_basic_grant::id)
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
let limit_id: i32 = insert_into(schema::evm_ether_transfer_limit::table)
|
||||
.values(&models::NewEvmEtherTransferLimit {
|
||||
window_secs: 60,
|
||||
max_volume: vec![1],
|
||||
})
|
||||
.returning(schema::evm_ether_transfer_limit::id)
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
let ether_grant_id: i32 = insert_into(schema::evm_ether_transfer_grant::table)
|
||||
.values(&models::NewEvmEtherTransferGrant {
|
||||
basic_grant_id: basic_id,
|
||||
limit_id,
|
||||
})
|
||||
.returning(schema::evm_ether_transfer_grant::id)
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
insert_into(schema::evm_ether_transfer_grant_target::table)
|
||||
.values(&models::NewEvmEtherTransferGrantTarget {
|
||||
grant_id: ether_grant_id,
|
||||
address: vec![0u8; 20],
|
||||
})
|
||||
.execute(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
insert_into(schema::evm_transaction_log::table)
|
||||
.values(&models::NewEvmTransactionLog {
|
||||
grant_id: basic_id,
|
||||
wallet_access_id: 1,
|
||||
chain_id: 1,
|
||||
eth_value: vec![0],
|
||||
signed_at: models::SqliteTimestamp::now(),
|
||||
})
|
||||
.execute(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
insert_into(schema::integrity_envelope::table)
|
||||
.values(&models::NewIntegrityEnvelope {
|
||||
entity_kind: "EtherTransfer".to_owned(),
|
||||
entity_id: basic_id.to_be_bytes().to_vec(),
|
||||
payload_version: 1,
|
||||
key_version: 1,
|
||||
mac: vec![0u8; 32],
|
||||
})
|
||||
.execute(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
drop(conn);
|
||||
|
||||
actor.useragent_delete_grant(basic_id).await.unwrap();
|
||||
|
||||
// Idempotency: second delete should be a no-op.
|
||||
actor.useragent_delete_grant(basic_id).await.unwrap();
|
||||
|
||||
let mut conn = db.get().await.unwrap();
|
||||
|
||||
let basic_count: i64 = schema::evm_basic_grant::table
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(basic_count, 0);
|
||||
|
||||
let ether_grant_count: i64 = schema::evm_ether_transfer_grant::table
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(ether_grant_count, 0);
|
||||
|
||||
let target_count: i64 = schema::evm_ether_transfer_grant_target::table
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(target_count, 0);
|
||||
|
||||
let limit_count: i64 = schema::evm_ether_transfer_limit::table
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(limit_count, 0);
|
||||
|
||||
let log_count: i64 = schema::evm_transaction_log::table
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(log_count, 0);
|
||||
|
||||
let envelope_count: i64 = schema::integrity_envelope::table
|
||||
.filter(schema::integrity_envelope::entity_kind.eq("EtherTransfer"))
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(envelope_count, 0);
|
||||
}
|
||||
|
||||
#[tokio::test]
|
||||
async fn delete_token_grant_cleans_related_tables() {
|
||||
let db = db::create_test_pool().await;
|
||||
let keyholder = KeyHolder::spawn(KeyHolder::new(db.clone()).await.unwrap());
|
||||
let mut actor = EvmActor::new(keyholder, db.clone());
|
||||
|
||||
let mut conn = db.get().await.unwrap();
|
||||
|
||||
let basic_id: i32 = insert_into(schema::evm_basic_grant::table)
|
||||
.values(&models::NewEvmBasicGrant {
|
||||
wallet_access_id: 1,
|
||||
chain_id: 1,
|
||||
valid_from: None,
|
||||
valid_until: None,
|
||||
max_gas_fee_per_gas: None,
|
||||
max_priority_fee_per_gas: None,
|
||||
rate_limit_count: None,
|
||||
rate_limit_window_secs: None,
|
||||
revoked_at: None,
|
||||
})
|
||||
.returning(schema::evm_basic_grant::id)
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
let token_grant_id: i32 = insert_into(schema::evm_token_transfer_grant::table)
|
||||
.values(&models::NewEvmTokenTransferGrant {
|
||||
basic_grant_id: basic_id,
|
||||
token_contract: vec![1u8; 20],
|
||||
receiver: None,
|
||||
})
|
||||
.returning(schema::evm_token_transfer_grant::id)
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
insert_into(schema::evm_token_transfer_volume_limit::table)
|
||||
.values(&models::NewEvmTokenTransferVolumeLimit {
|
||||
grant_id: token_grant_id,
|
||||
window_secs: 60,
|
||||
max_volume: vec![1],
|
||||
})
|
||||
.execute(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
insert_into(schema::evm_token_transfer_volume_limit::table)
|
||||
.values(&models::NewEvmTokenTransferVolumeLimit {
|
||||
grant_id: token_grant_id,
|
||||
window_secs: 3600,
|
||||
max_volume: vec![2],
|
||||
})
|
||||
.execute(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
let tx_log_id: i32 = insert_into(schema::evm_transaction_log::table)
|
||||
.values(&models::NewEvmTransactionLog {
|
||||
grant_id: basic_id,
|
||||
wallet_access_id: 1,
|
||||
chain_id: 1,
|
||||
eth_value: vec![0],
|
||||
signed_at: models::SqliteTimestamp::now(),
|
||||
})
|
||||
.returning(schema::evm_transaction_log::id)
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
insert_into(schema::evm_token_transfer_log::table)
|
||||
.values(&models::NewEvmTokenTransferLog {
|
||||
grant_id: token_grant_id,
|
||||
log_id: tx_log_id,
|
||||
chain_id: 1,
|
||||
token_contract: vec![1u8; 20],
|
||||
recipient_address: vec![2u8; 20],
|
||||
value: vec![3],
|
||||
})
|
||||
.execute(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
insert_into(schema::integrity_envelope::table)
|
||||
.values(&models::NewIntegrityEnvelope {
|
||||
entity_kind: "TokenTransfer".to_owned(),
|
||||
entity_id: basic_id.to_be_bytes().to_vec(),
|
||||
payload_version: 1,
|
||||
key_version: 1,
|
||||
mac: vec![0u8; 32],
|
||||
})
|
||||
.execute(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
drop(conn);
|
||||
|
||||
actor.useragent_delete_grant(basic_id).await.unwrap();
|
||||
|
||||
let mut conn = db.get().await.unwrap();
|
||||
|
||||
let basic_count: i64 = schema::evm_basic_grant::table
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(basic_count, 0);
|
||||
|
||||
let token_grant_count: i64 = schema::evm_token_transfer_grant::table
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(token_grant_count, 0);
|
||||
|
||||
let token_limits_count: i64 = schema::evm_token_transfer_volume_limit::table
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(token_limits_count, 0);
|
||||
|
||||
let token_logs_count: i64 = schema::evm_token_transfer_log::table
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(token_logs_count, 0);
|
||||
|
||||
let tx_logs_count: i64 = schema::evm_transaction_log::table
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(tx_logs_count, 0);
|
||||
|
||||
let envelope_count: i64 = schema::integrity_envelope::table
|
||||
.filter(schema::integrity_envelope::entity_kind.eq("TokenTransfer"))
|
||||
.count()
|
||||
.get_result(&mut conn)
|
||||
.await
|
||||
.unwrap();
|
||||
assert_eq!(envelope_count, 0);
|
||||
}
|
||||
Reference in New Issue
Block a user