diff --git a/.vscode/launch.json b/.vscode/launch.json new file mode 100644 index 000000000..185f6f97a --- /dev/null +++ b/.vscode/launch.json @@ -0,0 +1,252 @@ +{ + // Use IntelliSense to learn about possible attributes. + // Hover to view descriptions of existing attributes. + // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 + "version": "0.2.0", + "configurations": [ + { + "type": "lldb", + "request": "launch", + "name": "Debug executable 'xmtp_cli'", + "cargo": { + "args": [ + "build", + "--bin=xmtp_cli", + "--package=xmtp_cli" + ], + "filter": { + "name": "xmtp_cli", + "kind": "bin" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug unit tests in executable 'xmtp_cli'", + "cargo": { + "args": [ + "test", + "--no-run", + "--bin=xmtp_cli", + "--package=xmtp_cli" + ], + "filter": { + "name": "xmtp_cli", + "kind": "bin" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug unit tests in library 'xmtp_api_grpc'", + "cargo": { + "args": [ + "test", + "--no-run", + "--lib", + "--package=xmtp_api_grpc" + ], + "filter": { + "name": "xmtp_api_grpc", + "kind": "lib" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug unit tests in library 'xmtp_proto'", + "cargo": { + "args": [ + "test", + "--no-run", + "--lib", + "--package=xmtp_proto" + ], + "filter": { + "name": "xmtp_proto", + "kind": "lib" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug unit tests in library 'xmtp_v2'", + "cargo": { + "args": [ + "test", + "--no-run", + "--lib", + "--package=xmtp_v2" + ], + "filter": { + "name": "xmtp_v2", + "kind": "lib" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug unit tests in library 'xmtp_cryptography'", + "cargo": { + "args": [ + "test", + "--no-run", + "--lib", + "--package=xmtp_cryptography" + ], + "filter": { + "name": "xmtp_cryptography", + "kind": "lib" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug unit tests in library 'xmtp_mls'", + "cargo": { + "args": [ + "test", + "--no-run", + "--lib", + "--package=xmtp_mls" + ], + "filter": { + "name": "xmtp_mls", + "kind": "lib" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug executable 'update-schema'", + "cargo": { + "args": [ + "build", + "--bin=update-schema", + "--package=xmtp_mls" + ], + "filter": { + "name": "update-schema", + "kind": "bin" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug unit tests in executable 'update-schema'", + "cargo": { + "args": [ + "test", + "--no-run", + "--bin=update-schema", + "--package=xmtp_mls" + ], + "filter": { + "name": "update-schema", + "kind": "bin" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug executable 'mls-validation-service'", + "cargo": { + "args": [ + "build", + "--bin=mls-validation-service", + "--package=mls_validation_service" + ], + "filter": { + "name": "mls-validation-service", + "kind": "bin" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug unit tests in executable 'mls-validation-service'", + "cargo": { + "args": [ + "test", + "--no-run", + "--bin=mls-validation-service", + "--package=mls_validation_service" + ], + "filter": { + "name": "mls-validation-service", + "kind": "bin" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug unit tests in library 'xmtp_user_preferences'", + "cargo": { + "args": [ + "test", + "--no-run", + "--lib", + "--package=xmtp_user_preferences" + ], + "filter": { + "name": "xmtp_user_preferences", + "kind": "lib" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug unit tests in library 'xmtp_id'", + "cargo": { + "args": [ + "test", + "--no-run", + "--lib", + "--package=xmtp_id" + ], + "filter": { + "name": "xmtp_id", + "kind": "lib" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + } + ] +} \ No newline at end of file diff --git a/xmtp_id/src/associations/association_log.rs b/xmtp_id/src/associations/association_log.rs new file mode 100644 index 000000000..0e3565058 --- /dev/null +++ b/xmtp_id/src/associations/association_log.rs @@ -0,0 +1,392 @@ +use super::hashes::generate_inbox_id; +use super::member::{Member, MemberIdentifier, MemberKind}; +use super::signature::{Signature, SignatureError, SignatureKind}; +use super::state::AssociationState; + +use thiserror::Error; + +#[derive(Debug, Error, PartialEq)] +pub enum AssociationError { + #[error("Error creating association {0}")] + Generic(String), + #[error("Multiple create operations detected")] + MultipleCreate, + #[error("XID not yet created")] + NotCreated, + #[error("Signature validation failed {0}")] + Signature(#[from] SignatureError), + #[error("Member of kind {0} not allowed to add {1}")] + MemberNotAllowed(String, String), + #[error("Missing existing member")] + MissingExistingMember, + #[error("Legacy key is only allowed to be associated using a legacy signature with nonce 0")] + LegacySignatureReuse, + #[error("The new member identifier does not match the signer")] + NewMemberIdSignatureMismatch, + #[error("Signature not allowed for role {0:?} {1:?}")] + SignatureNotAllowed(String, String), + #[error("Replay detected")] + Replay, +} + +pub trait IdentityAction { + fn update_state( + &self, + existing_state: Option, + ) -> Result; + fn signatures(&self) -> Vec>; + fn replay_check(&self, state: &AssociationState) -> Result<(), AssociationError> { + let signatures = self.signatures(); + for signature in signatures { + if state.has_seen(&signature) { + return Err(AssociationError::Replay); + } + } + + Ok(()) + } +} + +/// CreateInbox Action +pub struct CreateInbox { + pub nonce: u64, + pub account_address: String, + pub initial_address_signature: Box, +} + +impl IdentityAction for CreateInbox { + fn update_state( + &self, + existing_state: Option, + ) -> Result { + if existing_state.is_some() { + return Err(AssociationError::MultipleCreate); + } + + let account_address = self.account_address.clone(); + let recovered_signer = self.initial_address_signature.recover_signer()?; + if recovered_signer.ne(&MemberIdentifier::Address(account_address.clone())) { + return Err(AssociationError::MissingExistingMember); + } + + allowed_signature_for_kind( + &MemberKind::Address, + &self.initial_address_signature.signature_kind(), + )?; + + if self.initial_address_signature.signature_kind() == SignatureKind::LegacyDelegated + && self.nonce != 0 + { + return Err(AssociationError::LegacySignatureReuse); + } + + Ok(AssociationState::new(account_address, self.nonce)) + } + + fn signatures(&self) -> Vec> { + vec![self.initial_address_signature.bytes()] + } +} + +/// AddAssociation Action +pub struct AddAssociation { + pub client_timestamp_ns: u64, + pub new_member_signature: Box, + pub new_member_identifier: MemberIdentifier, + pub existing_member_signature: Box, +} + +impl IdentityAction for AddAssociation { + fn update_state( + &self, + maybe_existing_state: Option, + ) -> Result { + let existing_state = maybe_existing_state.ok_or(AssociationError::NotCreated)?; + self.replay_check(&existing_state)?; + + // Validate the new member signature and get the recovered signer + let new_member_address = self.new_member_signature.recover_signer()?; + // Validate the existing member signature and get the recovedred signer + let existing_member_identifier = self.existing_member_signature.recover_signer()?; + + if new_member_address.ne(&self.new_member_identifier) { + return Err(AssociationError::NewMemberIdSignatureMismatch); + } + + // You cannot add yourself + if new_member_address == existing_member_identifier { + return Err(AssociationError::Generic("tried to add self".to_string())); + } + + // Only allow LegacyDelegated signatures on XIDs with a nonce of 0 + // Otherwise the client should use the regular wallet signature to create + if (is_legacy_signature(&self.new_member_signature) + || is_legacy_signature(&self.existing_member_signature)) + && existing_state.inbox_id().ne(&generate_inbox_id( + &existing_member_identifier.to_string(), + &0, + )) + { + return Err(AssociationError::LegacySignatureReuse); + } + + allowed_signature_for_kind( + &self.new_member_identifier.kind(), + &self.new_member_signature.signature_kind(), + )?; + + let existing_member = existing_state.get(&existing_member_identifier); + + let existing_entity_id = match existing_member { + // If there is an existing member of the XID, use that member's ID + Some(member) => member.identifier, + None => { + // Get the recovery address from the state as a MemberIdentifier + let recovery_identifier: MemberIdentifier = + existing_state.recovery_address().clone().into(); + + // Check if it is a signature from the recovery address, which is allowed to add members + if existing_member_identifier.ne(&recovery_identifier) { + return Err(AssociationError::MissingExistingMember); + } + // BUT, the recovery address has to be used with a real wallet signature, can't be delegated + if is_legacy_signature(&self.existing_member_signature) { + return Err(AssociationError::LegacySignatureReuse); + } + // If it is a real wallet signature, then it is allowed to add members + recovery_identifier + } + }; + + // Ensure that the existing member signature is correct for the existing member type + allowed_signature_for_kind( + &existing_entity_id.kind(), + &self.existing_member_signature.signature_kind(), + )?; + + // Ensure that the new member signature is correct for the new member type + allowed_association( + &existing_member_identifier.kind(), + &self.new_member_identifier.kind(), + )?; + + let new_member = Member::new(new_member_address, Some(existing_entity_id)); + + println!("Adding new entity to state {:?}", &new_member); + + Ok(existing_state.add(new_member)) + } + + fn signatures(&self) -> Vec> { + vec![ + self.existing_member_signature.bytes(), + self.new_member_signature.bytes(), + ] + } +} + +/// RevokeAssociation Action +pub struct RevokeAssociation { + pub client_timestamp_ns: u64, + pub recovery_address_signature: Box, + pub revoked_member: MemberIdentifier, +} + +impl IdentityAction for RevokeAssociation { + fn update_state( + &self, + maybe_existing_state: Option, + ) -> Result { + let existing_state = maybe_existing_state.ok_or(AssociationError::NotCreated)?; + self.replay_check(&existing_state)?; + + if is_legacy_signature(&self.recovery_address_signature) { + return Err(AssociationError::SignatureNotAllowed( + MemberKind::Address.to_string(), + SignatureKind::LegacyDelegated.to_string(), + )); + } + // Don't need to check for replay here since revocation is idempotent + let recovery_signer = self.recovery_address_signature.recover_signer()?; + // Make sure there is a recovery address set on the state + let state_recovery_address = existing_state.recovery_address(); + + // Ensure this message is signed by the recovery address + if recovery_signer.ne(&MemberIdentifier::Address(state_recovery_address.clone())) { + return Err(AssociationError::MissingExistingMember); + } + + let installations_to_remove: Vec = existing_state + .members_by_parent(&self.revoked_member) + .into_iter() + // Only remove children if they are installations + .filter(|child| child.kind() == MemberKind::Installation) + .collect(); + + // Actually apply the revocation to the parent + let new_state = existing_state.remove(&self.revoked_member); + + Ok(installations_to_remove + .iter() + .fold(new_state, |state, installation| { + state.remove(&installation.identifier) + })) + } + + fn signatures(&self) -> Vec> { + vec![self.recovery_address_signature.bytes()] + } +} + +/// ChangeRecoveryAddress Action +pub struct ChangeRecoveryAddress { + pub client_timestamp_ns: u64, + pub recovery_address_signature: Box, + pub new_recovery_address: String, +} + +impl IdentityAction for ChangeRecoveryAddress { + fn update_state( + &self, + existing_state: Option, + ) -> Result { + let existing_state = existing_state.ok_or(AssociationError::NotCreated)?; + self.replay_check(&existing_state)?; + + if is_legacy_signature(&self.recovery_address_signature) { + return Err(AssociationError::SignatureNotAllowed( + MemberKind::Address.to_string(), + SignatureKind::LegacyDelegated.to_string(), + )); + } + + let recovery_signer = self.recovery_address_signature.recover_signer()?; + if recovery_signer.ne(&existing_state.recovery_address().clone().into()) { + return Err(AssociationError::MissingExistingMember); + } + + Ok(existing_state.set_recovery_address(self.new_recovery_address.clone())) + } + + fn signatures(&self) -> Vec> { + vec![self.recovery_address_signature.bytes()] + } +} + +/// All possible Action types that can be used inside an `IdentityUpdate` +pub enum Action { + CreateInbox(CreateInbox), + AddAssociation(AddAssociation), + RevokeAssociation(RevokeAssociation), + ChangeRecoveryAddress(ChangeRecoveryAddress), +} + +impl IdentityAction for Action { + fn update_state( + &self, + existing_state: Option, + ) -> Result { + match self { + Action::CreateInbox(event) => event.update_state(existing_state), + Action::AddAssociation(event) => event.update_state(existing_state), + Action::RevokeAssociation(event) => event.update_state(existing_state), + Action::ChangeRecoveryAddress(event) => event.update_state(existing_state), + } + } + + fn signatures(&self) -> Vec> { + match self { + Action::CreateInbox(event) => event.signatures(), + Action::AddAssociation(event) => event.signatures(), + Action::RevokeAssociation(event) => event.signatures(), + Action::ChangeRecoveryAddress(event) => event.signatures(), + } + } +} + +/// An `IdentityUpdate` contains one or more Actions that can be applied to the AssociationState +pub struct IdentityUpdate { + pub actions: Vec, +} + +impl IdentityUpdate { + pub fn new(actions: Vec) -> Self { + Self { actions } + } +} + +impl IdentityAction for IdentityUpdate { + fn update_state( + &self, + existing_state: Option, + ) -> Result { + let mut state = existing_state.clone(); + for action in &self.actions { + state = Some(action.update_state(state)?); + } + + let new_state = state.ok_or(AssociationError::NotCreated)?; + + // After all the updates in the LogEntry have been processed, add the list of signatures to the state + // so that the signatures can not be re-used in subsequent updates + Ok(new_state.add_seen_signatures(self.signatures())) + } + + fn signatures(&self) -> Vec> { + self.actions + .iter() + .flat_map(|action| action.signatures()) + .collect() + } +} + +fn is_legacy_signature(signature: &Box) -> bool { + signature.signature_kind() == SignatureKind::LegacyDelegated +} + +fn allowed_association( + existing_member_kind: &MemberKind, + new_member_kind: &MemberKind, +) -> Result<(), AssociationError> { + // The only disallowed association is an installation adding an installation + if existing_member_kind.eq(&MemberKind::Installation) + && new_member_kind.eq(&MemberKind::Installation) + { + return Err(AssociationError::MemberNotAllowed( + existing_member_kind.to_string(), + new_member_kind.to_string(), + )); + } + + Ok(()) +} + +// Ensure that the type of signature matches the new entity's role. +fn allowed_signature_for_kind( + role: &MemberKind, + signature_kind: &SignatureKind, +) -> Result<(), AssociationError> { + let is_ok = match role { + MemberKind::Address => match signature_kind { + SignatureKind::Erc191 => true, + SignatureKind::Erc1271 => true, + SignatureKind::InstallationKey => false, + SignatureKind::LegacyDelegated => true, + }, + MemberKind::Installation => match signature_kind { + SignatureKind::Erc191 => false, + SignatureKind::Erc1271 => false, + SignatureKind::InstallationKey => true, + SignatureKind::LegacyDelegated => false, + }, + }; + + if !is_ok { + return Err(AssociationError::SignatureNotAllowed( + role.to_string(), + signature_kind.to_string(), + )); + } + + Ok(()) +} diff --git a/xmtp_id/src/associations/hashes.rs b/xmtp_id/src/associations/hashes.rs index 2434f67e8..fe000e037 100644 --- a/xmtp_id/src/associations/hashes.rs +++ b/xmtp_id/src/associations/hashes.rs @@ -7,6 +7,6 @@ pub fn sha256_string(input: String) -> String { format!("{:x}", result) } -pub fn generate_xid(account_address: &String, nonce: &u64) -> String { +pub fn generate_inbox_id(account_address: &String, nonce: &u64) -> String { sha256_string(format!("{}{}", account_address, nonce)) } diff --git a/xmtp_id/src/associations/member.rs b/xmtp_id/src/associations/member.rs index c61549fc5..8efc76049 100644 --- a/xmtp_id/src/associations/member.rs +++ b/xmtp_id/src/associations/member.rs @@ -13,6 +13,7 @@ impl std::fmt::Display for MemberKind { } } +/// A MemberIdentifier can be either an Address or an Installation Public Key #[derive(Clone, Debug, PartialEq, Eq, Hash)] pub enum MemberIdentifier { Address(String), @@ -51,6 +52,7 @@ impl From> for MemberIdentifier { } } +/// A Member of Inbox #[derive(Clone, Debug, PartialEq)] pub struct Member { pub identifier: MemberIdentifier, diff --git a/xmtp_id/src/associations/mod.rs b/xmtp_id/src/associations/mod.rs index 839409895..439e5e802 100644 --- a/xmtp_id/src/associations/mod.rs +++ b/xmtp_id/src/associations/mod.rs @@ -1,8 +1,608 @@ +mod association_log; mod hashes; mod member; +mod signature; mod state; #[cfg(test)] mod test_utils; +pub use self::association_log::*; pub use self::member::{Member, MemberIdentifier, MemberKind}; +pub use self::signature::{Signature, SignatureError, SignatureKind}; pub use self::state::AssociationState; + +// Apply a single IdentityUpdate to an existing AssociationState +pub fn apply_update( + initial_state: AssociationState, + update: IdentityUpdate, +) -> Result { + update.update_state(Some(initial_state)) +} + +// Get the current state from an array of `IdentityUpdate`s. Entire operation fails if any operation fails +pub fn get_state(updates: Vec) -> Result { + let new_state = updates.iter().try_fold( + None, + |state, update| -> Result, AssociationError> { + let updated_state = update.update_state(state)?; + Ok(Some(updated_state)) + }, + )?; + + new_state.ok_or(AssociationError::NotCreated) +} + +#[cfg(test)] +mod tests { + use self::test_utils::{rand_string, rand_u64, rand_vec}; + + use super::*; + + #[derive(Clone)] + struct MockSignature { + is_valid: bool, + signer_identity: MemberIdentifier, + signature_kind: SignatureKind, + signature_nonce: u64, + } + + impl MockSignature { + pub fn new_boxed( + is_valid: bool, + signer_identity: MemberIdentifier, + signature_kind: SignatureKind, + // Signature nonce is used to control what the signature bytes are + // Defaults to random + signature_nonce: Option, + ) -> Box { + let nonce = signature_nonce.unwrap_or(rand_u64()); + Box::new(Self { + is_valid, + signer_identity, + signature_kind, + signature_nonce: nonce, + }) + } + } + + impl Signature for MockSignature { + fn signature_kind(&self) -> SignatureKind { + self.signature_kind.clone() + } + + fn recover_signer(&self) -> Result { + match self.is_valid { + true => Ok(self.signer_identity.clone()), + false => Err(SignatureError::Invalid), + } + } + + fn bytes(&self) -> Vec { + let sig = format!("{}{}", self.signer_identity, self.signature_nonce); + sig.as_bytes().to_vec() + } + } + + impl Default for AddAssociation { + fn default() -> Self { + let existing_member = rand_string(); + let new_member = rand_vec(); + return Self { + client_timestamp_ns: rand_u64(), + existing_member_signature: MockSignature::new_boxed( + true, + existing_member.into(), + SignatureKind::Erc191, + None, + ), + new_member_signature: MockSignature::new_boxed( + true, + new_member.clone().into(), + SignatureKind::InstallationKey, + None, + ), + new_member_identifier: new_member.into(), + }; + } + } + + // Default will create an inbox with a ERC-191 signature + impl Default for CreateInbox { + fn default() -> Self { + let signer = rand_string(); + return Self { + nonce: rand_u64(), + account_address: signer.clone(), + initial_address_signature: MockSignature::new_boxed( + true, + signer.into(), + SignatureKind::Erc191, + None, + ), + }; + } + } + + impl Default for RevokeAssociation { + fn default() -> Self { + let signer = rand_string(); + return Self { + client_timestamp_ns: rand_u64(), + recovery_address_signature: MockSignature::new_boxed( + true, + signer.into(), + SignatureKind::Erc191, + None, + ), + revoked_member: rand_string().into(), + }; + } + } + + fn new_test_inbox() -> AssociationState { + let create_request = CreateInbox::default(); + let identity_update = IdentityUpdate::new(vec![Action::CreateInbox(create_request)]); + + get_state(vec![identity_update]).unwrap() + } + + fn new_test_inbox_with_installation() -> AssociationState { + let initial_state = new_test_inbox(); + let initial_wallet_address: MemberIdentifier = + initial_state.recovery_address().clone().into(); + + let update = Action::AddAssociation(AddAssociation { + existing_member_signature: MockSignature::new_boxed( + true, + initial_wallet_address.clone(), + SignatureKind::Erc191, + None, + ), + ..Default::default() + }); + + apply_update(initial_state, IdentityUpdate::new(vec![update])).unwrap() + } + + #[test] + fn test_create_inbox() { + let create_request = CreateInbox::default(); + let account_address = create_request.account_address.clone(); + let identity_update = IdentityUpdate::new(vec![Action::CreateInbox(create_request)]); + let state = get_state(vec![identity_update]).unwrap(); + assert_eq!(state.members().len(), 1); + + let existing_entity = state.get(&account_address.clone().into()).unwrap(); + assert!(existing_entity.identifier.eq(&account_address.into())); + } + + #[test] + fn create_and_add_separately() { + let initial_state = new_test_inbox(); + let new_installation_identifier: MemberIdentifier = rand_vec().into(); + let first_member: MemberIdentifier = initial_state.recovery_address().clone().into(); + + let update = Action::AddAssociation(AddAssociation { + new_member_identifier: new_installation_identifier.clone(), + new_member_signature: MockSignature::new_boxed( + true, + new_installation_identifier.clone(), + SignatureKind::InstallationKey, + None, + ), + existing_member_signature: MockSignature::new_boxed( + true, + first_member.clone(), + SignatureKind::Erc191, + None, + ), + ..Default::default() + }); + + let new_state = apply_update(initial_state, IdentityUpdate::new(vec![update])).unwrap(); + assert_eq!(new_state.members().len(), 2); + + let new_member = new_state.get(&new_installation_identifier).unwrap(); + assert_eq!(new_member.added_by_entity, Some(first_member)); + } + + #[test] + fn create_and_add_together() { + let create_action = CreateInbox::default(); + let account_address = create_action.account_address.clone(); + let new_member_identifier: MemberIdentifier = rand_vec().into(); + let add_action = AddAssociation { + existing_member_signature: MockSignature::new_boxed( + true, + account_address.clone().into(), + SignatureKind::Erc191, + None, + ), + // Add an installation ID + new_member_signature: MockSignature::new_boxed( + true, + new_member_identifier.clone(), + SignatureKind::InstallationKey, + None, + ), + new_member_identifier: new_member_identifier.clone(), + ..Default::default() + }; + let identity_update = IdentityUpdate::new(vec![ + Action::CreateInbox(create_action), + Action::AddAssociation(add_action), + ]); + let state = get_state(vec![identity_update]).unwrap(); + assert_eq!(state.members().len(), 2); + assert_eq!( + state.get(&new_member_identifier).unwrap().added_by_entity, + Some(account_address.into()) + ); + } + + #[test] + fn create_from_legacy_key() { + let member_identifier: MemberIdentifier = rand_string().into(); + let create_action = CreateInbox { + nonce: 0, + account_address: member_identifier.to_string(), + initial_address_signature: MockSignature::new_boxed( + true, + member_identifier.clone(), + SignatureKind::LegacyDelegated, + Some(0), + ), + }; + let state = get_state(vec![IdentityUpdate::new(vec![Action::CreateInbox( + create_action, + )])]) + .unwrap(); + assert_eq!(state.members().len(), 1); + + // The legacy key can only be used once. After this, subsequent updates should fail + let update = Action::AddAssociation(AddAssociation { + existing_member_signature: MockSignature::new_boxed( + true, + member_identifier, + SignatureKind::LegacyDelegated, + // All requests from the same legacy key will have the same signature nonce + Some(0), + ), + ..Default::default() + }); + let update_result = apply_update(state, IdentityUpdate::new(vec![update])); + assert!(update_result.is_err()); + assert_eq!(update_result.err().unwrap(), AssociationError::Replay); + } + + #[test] + fn add_wallet_from_installation_key() { + let initial_state = new_test_inbox_with_installation(); + let installation_id = initial_state + .members_by_kind(MemberKind::Installation) + .first() + .cloned() + .unwrap() + .identifier; + + let new_wallet_address: MemberIdentifier = rand_string().into(); + let add_association = Action::AddAssociation(AddAssociation { + new_member_identifier: new_wallet_address.clone(), + new_member_signature: MockSignature::new_boxed( + true, + new_wallet_address.clone(), + SignatureKind::Erc191, + None, + ), + existing_member_signature: MockSignature::new_boxed( + true, + installation_id.clone(), + SignatureKind::InstallationKey, + None, + ), + ..Default::default() + }); + + let new_state = apply_update(initial_state, IdentityUpdate::new(vec![add_association])) + .expect("expected update to succeed"); + assert_eq!(new_state.members().len(), 3); + } + + #[test] + fn reject_invalid_signature_on_create() { + let bad_signature = + MockSignature::new_boxed(false, rand_string().into(), SignatureKind::Erc191, None); + let action = CreateInbox { + initial_address_signature: bad_signature.clone(), + ..Default::default() + }; + + let state_result = get_state(vec![IdentityUpdate::new(vec![Action::CreateInbox(action)])]); + assert!(state_result.is_err()); + assert_eq!( + state_result.err().unwrap(), + AssociationError::Signature(SignatureError::Invalid) + ); + } + + #[test] + fn reject_invalid_signature_on_update() { + let initial_state = new_test_inbox(); + let bad_signature = + MockSignature::new_boxed(false, rand_string().into(), SignatureKind::Erc191, None); + + let update_with_bad_existing_member = Action::AddAssociation(AddAssociation { + existing_member_signature: bad_signature.clone(), + ..Default::default() + }); + + let update_result = apply_update( + initial_state.clone(), + IdentityUpdate::new(vec![update_with_bad_existing_member]), + ); + assert!(update_result.is_err()); + assert_eq!( + update_result.err().unwrap(), + AssociationError::Signature(SignatureError::Invalid) + ); + + let update_with_bad_new_member = Action::AddAssociation(AddAssociation { + new_member_signature: bad_signature.clone(), + existing_member_signature: MockSignature::new_boxed( + true, + initial_state.recovery_address().clone().into(), + SignatureKind::Erc191, + None, + ), + ..Default::default() + }); + + let update_result_2 = apply_update( + initial_state, + IdentityUpdate::new(vec![update_with_bad_new_member]), + ); + assert!(update_result_2.is_err()); + assert_eq!( + update_result_2.err().unwrap(), + AssociationError::Signature(SignatureError::Invalid) + ); + } + + #[test] + fn reject_if_signer_not_existing_member() { + let create_request = Action::CreateInbox(CreateInbox::default()); + // The default here will create an AddAssociation from a random wallet + let update = Action::AddAssociation(AddAssociation { + // Existing member signature is coming from a random wallet + existing_member_signature: MockSignature::new_boxed( + true, + rand_string().into(), + SignatureKind::Erc191, + None, + ), + ..Default::default() + }); + + let state_result = get_state(vec![IdentityUpdate::new(vec![create_request, update])]); + assert!(state_result.is_err()); + assert_eq!( + state_result.err().unwrap(), + AssociationError::MissingExistingMember + ); + } + + #[test] + fn reject_if_installation_adding_installation() { + let existing_state = new_test_inbox_with_installation(); + let existing_installations = existing_state.members_by_kind(MemberKind::Installation); + let existing_installation = existing_installations.first().unwrap(); + let new_installation_id: MemberIdentifier = rand_vec().into(); + + let update = Action::AddAssociation(AddAssociation { + existing_member_signature: MockSignature::new_boxed( + true, + existing_installation.identifier.clone(), + SignatureKind::InstallationKey, + None, + ), + new_member_identifier: new_installation_id.clone(), + new_member_signature: MockSignature::new_boxed( + true, + new_installation_id.clone(), + SignatureKind::InstallationKey, + None, + ), + ..Default::default() + }); + + let update_result = apply_update(existing_state, IdentityUpdate::new(vec![update])); + assert!(update_result.is_err()); + assert_eq!( + update_result.err().unwrap(), + AssociationError::MemberNotAllowed( + MemberKind::Installation.to_string(), + MemberKind::Installation.to_string() + ) + ); + } + + #[test] + fn revoke() { + let initial_state = new_test_inbox_with_installation(); + let installation_id = initial_state + .members_by_kind(MemberKind::Installation) + .first() + .cloned() + .unwrap() + .identifier; + let update = Action::RevokeAssociation(RevokeAssociation { + recovery_address_signature: MockSignature::new_boxed( + true, + initial_state.recovery_address().clone().into(), + SignatureKind::Erc191, + None, + ), + revoked_member: installation_id.clone(), + ..Default::default() + }); + + let new_state = apply_update(initial_state, IdentityUpdate::new(vec![update])) + .expect("expected update to succeed"); + assert!(new_state.get(&installation_id).is_none()); + } + + #[test] + fn revoke_children() { + let initial_state = new_test_inbox_with_installation(); + let wallet_address = initial_state + .members_by_kind(MemberKind::Address) + .first() + .cloned() + .unwrap() + .identifier; + + let add_second_installation = Action::AddAssociation(AddAssociation { + existing_member_signature: MockSignature::new_boxed( + true, + wallet_address.clone(), + SignatureKind::Erc191, + None, + ), + ..Default::default() + }); + + let new_state = apply_update( + initial_state, + IdentityUpdate::new(vec![add_second_installation]), + ) + .expect("expected update to succeed"); + assert_eq!(new_state.members().len(), 3); + + let revocation = Action::RevokeAssociation(RevokeAssociation { + recovery_address_signature: MockSignature::new_boxed( + true, + wallet_address.clone(), + SignatureKind::Erc191, + None, + ), + revoked_member: wallet_address.clone(), + ..Default::default() + }); + + // With this revocation the original wallet + both installations should be gone + let new_state = apply_update(new_state, IdentityUpdate::new(vec![revocation])) + .expect("expected update to succeed"); + assert_eq!(new_state.members().len(), 0); + } + + #[test] + fn revoke_and_re_add() { + let initial_state = new_test_inbox(); + let wallet_address = initial_state + .members_by_kind(MemberKind::Address) + .first() + .cloned() + .unwrap() + .identifier; + + let second_wallet_address: MemberIdentifier = rand_string().into(); + let add_second_wallet = Action::AddAssociation(AddAssociation { + new_member_identifier: second_wallet_address.clone(), + new_member_signature: MockSignature::new_boxed( + true, + second_wallet_address.clone(), + SignatureKind::Erc191, + None, + ), + existing_member_signature: MockSignature::new_boxed( + true, + wallet_address.clone(), + SignatureKind::Erc191, + None, + ), + ..Default::default() + }); + + let revoke_second_wallet = Action::RevokeAssociation(RevokeAssociation { + recovery_address_signature: MockSignature::new_boxed( + true, + wallet_address.clone(), + SignatureKind::Erc191, + None, + ), + revoked_member: second_wallet_address.clone(), + ..Default::default() + }); + + let state_after_remove = apply_update( + initial_state, + IdentityUpdate::new(vec![add_second_wallet, revoke_second_wallet]), + ) + .expect("expected update to succeed"); + assert_eq!(state_after_remove.members().len(), 1); + + let add_second_wallet_again = Action::AddAssociation(AddAssociation { + new_member_identifier: second_wallet_address.clone(), + new_member_signature: MockSignature::new_boxed( + true, + second_wallet_address.clone(), + SignatureKind::Erc191, + None, + ), + existing_member_signature: MockSignature::new_boxed( + true, + wallet_address, + SignatureKind::Erc191, + None, + ), + ..Default::default() + }); + + let state_after_re_add = apply_update( + state_after_remove, + IdentityUpdate::new(vec![add_second_wallet_again]), + ) + .expect("expected update to succeed"); + assert_eq!(state_after_re_add.members().len(), 2); + } + + #[test] + fn change_recovery_address() { + let initial_state = new_test_inbox_with_installation(); + let initial_recovery_address: MemberIdentifier = + initial_state.recovery_address().clone().into(); + let new_recovery_address = rand_string(); + let update_recovery = Action::ChangeRecoveryAddress(ChangeRecoveryAddress { + client_timestamp_ns: rand_u64(), + new_recovery_address: new_recovery_address.clone(), + recovery_address_signature: MockSignature::new_boxed( + true, + initial_state.recovery_address().clone().into(), + SignatureKind::Erc191, + None, + ), + }); + + let new_state = apply_update(initial_state, IdentityUpdate::new(vec![update_recovery])) + .expect("expected update to succeed"); + assert_eq!(new_state.recovery_address(), &new_recovery_address); + + let attempted_revoke = Action::RevokeAssociation(RevokeAssociation { + recovery_address_signature: MockSignature::new_boxed( + true, + initial_recovery_address.clone(), + SignatureKind::Erc191, + None, + ), + revoked_member: initial_recovery_address.clone(), + ..Default::default() + }); + + let revoke_result = apply_update(new_state, IdentityUpdate::new(vec![attempted_revoke])); + assert!(revoke_result.is_err()); + assert_eq!( + revoke_result.err().unwrap(), + AssociationError::MissingExistingMember + ); + } +} diff --git a/xmtp_id/src/associations/signature.rs b/xmtp_id/src/associations/signature.rs new file mode 100644 index 000000000..b435dee5c --- /dev/null +++ b/xmtp_id/src/associations/signature.rs @@ -0,0 +1,35 @@ +use thiserror::Error; + +use super::MemberIdentifier; + +#[derive(Debug, Error, PartialEq)] +pub enum SignatureError { + #[error("Signature validation failed")] + Invalid, +} + +#[derive(Clone, Debug, PartialEq)] +pub enum SignatureKind { + // We might want to have some sort of LegacyErc191 Signature Kind for the `CreateIdentity` signatures only + Erc191, + Erc1271, + InstallationKey, + LegacyDelegated, +} + +impl std::fmt::Display for SignatureKind { + fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { + match self { + SignatureKind::Erc191 => write!(f, "erc-191"), + SignatureKind::Erc1271 => write!(f, "erc-1271"), + SignatureKind::InstallationKey => write!(f, "installation-key"), + SignatureKind::LegacyDelegated => write!(f, "legacy-delegated"), + } + } +} + +pub trait Signature { + fn recover_signer(&self) -> Result; + fn signature_kind(&self) -> SignatureKind; + fn bytes(&self) -> Vec; +} diff --git a/xmtp_id/src/associations/state.rs b/xmtp_id/src/associations/state.rs index 3e687fd19..0e1246fa1 100644 --- a/xmtp_id/src/associations/state.rs +++ b/xmtp_id/src/associations/state.rs @@ -1,10 +1,10 @@ use std::collections::{HashMap, HashSet}; -use super::{hashes::generate_xid, member::Member, MemberIdentifier, MemberKind}; +use super::{hashes::generate_inbox_id, member::Member, MemberIdentifier, MemberKind}; #[derive(Clone, Debug)] pub struct AssociationState { - xid: String, + inbox_id: String, members: HashMap, recovery_address: String, seen_signatures: HashSet>, @@ -51,8 +51,8 @@ impl AssociationState { self.members.values().cloned().collect() } - pub fn xid(&self) -> &String { - &self.xid + pub fn inbox_id(&self) -> &String { + &self.inbox_id } pub fn recovery_address(&self) -> &String { @@ -76,7 +76,7 @@ impl AssociationState { } pub fn new(account_address: String, nonce: u64) -> Self { - let xid = generate_xid(&account_address, &nonce); + let inbox_id = generate_inbox_id(&account_address, &nonce); let identifier = MemberIdentifier::Address(account_address.clone()); let new_member = Member::new(identifier.clone(), None); Self { @@ -87,7 +87,7 @@ impl AssociationState { }, seen_signatures: HashSet::new(), recovery_address: account_address, - xid, + inbox_id, } } }