Compare commits
1 Commits
security-h
...
impl-usera
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
3aae3e1d83 |
@@ -1,24 +1,26 @@
|
|||||||
use alloy::{consensus::TxEip1559, primitives::Address, signers::Signature};
|
use alloy::{consensus::TxEip1559, primitives::Address, signers::Signature};
|
||||||
use diesel::{
|
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 kameo::{Actor, actor::ActorRef, messages};
|
||||||
use rand::{SeedableRng, rng, rngs::StdRng};
|
use rand::{SeedableRng, rng, rngs::StdRng};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
actors::keyholder::{CreateNew, Decrypt, GetState, KeyHolder, KeyHolderState},
|
actors::keyholder::{CreateNew, Decrypt, KeyHolder},
|
||||||
crypto::integrity,
|
crypto::integrity,
|
||||||
db::{
|
db::{
|
||||||
DatabaseError, DatabasePool,
|
DatabaseError, DatabasePool,
|
||||||
models::{self, SqliteTimestamp},
|
models::{self},
|
||||||
schema,
|
schema,
|
||||||
},
|
},
|
||||||
evm::{
|
evm::{
|
||||||
self, ListError, RunKind, policies::{
|
self, ListError, RunKind,
|
||||||
|
policies::{
|
||||||
CombinedSettings, Grant, SharedGrantSettings, SpecificGrant, SpecificMeaning,
|
CombinedSettings, Grant, SharedGrantSettings, SpecificGrant, SpecificMeaning,
|
||||||
ether_transfer::EtherTransfer, token_transfers::TokenTransfer,
|
ether_transfer::EtherTransfer, token_transfers::TokenTransfer,
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
safe_cell::{SafeCell, SafeCellHandle as _},
|
safe_cell::{SafeCell, SafeCellHandle as _},
|
||||||
};
|
};
|
||||||
@@ -158,27 +160,114 @@ impl EvmActor {
|
|||||||
|
|
||||||
#[message]
|
#[message]
|
||||||
pub async fn useragent_delete_grant(&mut self, grant_id: i32) -> Result<(), Error> {
|
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 mut conn = self.db.get().await.map_err(DatabaseError::from)?;
|
||||||
// let keyholder = self.keyholder.clone();
|
|
||||||
|
|
||||||
// diesel_async::AsyncConnection::transaction(&mut conn, |conn| {
|
// We intentionally perform a hard delete here to avoid leaving revoked grants and their
|
||||||
// Box::pin(async move {
|
// related rows as long-lived DB garbage. We also don't rely on SQLite FK cascades because
|
||||||
// diesel::update(schema::evm_basic_grant::table)
|
// they can be disabled per-connection.
|
||||||
// .filter(schema::evm_basic_grant::id.eq(grant_id))
|
conn.transaction(|conn| {
|
||||||
// .set(schema::evm_basic_grant::revoked_at.eq(SqliteTimestamp::now()))
|
Box::pin(async move {
|
||||||
// .execute(conn)
|
// First, resolve policy-specific rows by basic grant id.
|
||||||
// .await?;
|
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(())
|
// Token-transfer: logs must be deleted before transaction logs (FK restrict).
|
||||||
// })
|
if let Some(token_grant_id) = token_grant_id {
|
||||||
// })
|
diesel::delete(
|
||||||
// .await
|
schema::evm_token_transfer_log::table
|
||||||
// .map_err(DatabaseError::from)?;
|
.filter(schema::evm_token_transfer_log::grant_id.eq(token_grant_id)),
|
||||||
|
)
|
||||||
|
.execute(conn)
|
||||||
|
.await?;
|
||||||
|
|
||||||
// Ok(())
|
diesel::delete(schema::evm_token_transfer_volume_limit::table.filter(
|
||||||
todo!()
|
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]
|
#[message]
|
||||||
@@ -270,3 +359,6 @@ impl EvmActor {
|
|||||||
Ok(signer.sign_transaction_sync(&mut transaction)?)
|
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