use alloc::borrow::Cow;
use alloc::string::ToString;
use alloc::vec::Vec;
use serde::{Deserialize, Serialize};
use serde_repr::{Deserialize_repr, Serialize_repr};
use serde_with::skip_serializing_none;
use strum_macros::{AsRefStr, Display, EnumIter};
use crate::models::amount::XRPAmount;
use crate::models::{
Currency, FlagCollection, Model, ValidateCurrencies, XRPLModelException, XRPLModelResult,
};
use super::vault_common::{validate_hex_blob, validate_nonnegative_number, validate_vault_id};
use super::{CommonFields, CommonTransactionBuilder, Memo, Signer, Transaction, TransactionType};
const MAX_VAULT_DATA_HEX_LEN: usize = 512;
const MAX_VAULT_MPTOKEN_METADATA_HEX_LEN: usize = 2048;
const MAX_VAULT_SCALE: u8 = 18;
const FIRST_COME_FIRST_SERVE_POLICY: u8 = 1;
#[derive(
Debug, Eq, PartialEq, Copy, Clone, Serialize_repr, Deserialize_repr, Display, AsRefStr, EnumIter,
)]
#[repr(u32)]
pub enum VaultCreateFlag {
TfVaultPrivate = 0x00010000,
TfVaultShareNonTransferable = 0x00020000,
}
#[skip_serializing_none]
#[derive(
Debug,
Default,
Serialize,
Deserialize,
PartialEq,
Eq,
Clone,
xrpl_rust_macros::ValidateCurrencies,
)]
#[serde(rename_all = "PascalCase")]
pub struct VaultCreate<'a> {
#[serde(flatten)]
pub common_fields: CommonFields<'a, VaultCreateFlag>,
pub asset: Currency<'a>,
pub data: Option<Cow<'a, str>>,
pub assets_maximum: Option<Cow<'a, str>>,
#[serde(rename = "MPTokenMetadata")]
pub mptoken_metadata: Option<Cow<'a, str>>,
#[serde(rename = "DomainID")]
pub domain_id: Option<Cow<'a, str>>,
pub withdrawal_policy: Option<u8>,
pub scale: Option<u8>,
}
impl Model for VaultCreate<'_> {
fn get_errors(&self) -> XRPLModelResult<()> {
self.validate_currencies()?;
if let Some(data) = self.data.as_deref() {
validate_hex_blob("data", data, MAX_VAULT_DATA_HEX_LEN)?;
}
if let Some(maximum) = self.assets_maximum.as_deref() {
validate_nonnegative_number("assets_maximum", maximum)?;
}
if let Some(metadata) = self.mptoken_metadata.as_deref() {
validate_hex_blob(
"mptoken_metadata",
metadata,
MAX_VAULT_MPTOKEN_METADATA_HEX_LEN,
)?;
}
if let Some(domain_id) = self.domain_id.as_deref() {
if !self.has_flag(&VaultCreateFlag::TfVaultPrivate) {
return Err(XRPLModelException::InvalidValue {
field: "domain_id".into(),
expected: "only present when tfVaultPrivate is set".into(),
found: domain_id.into(),
});
}
validate_vault_id(domain_id)?;
}
if let Some(policy) = self.withdrawal_policy {
if policy != FIRST_COME_FIRST_SERVE_POLICY {
return Err(XRPLModelException::InvalidValue {
field: "withdrawal_policy".into(),
expected: "1 (first-come-first-serve)".into(),
found: policy.to_string(),
});
}
}
if let Some(scale) = self.scale {
if matches!(self.asset, Currency::XRP(_) | Currency::MPTCurrency(_)) {
return Err(XRPLModelException::InvalidValue {
field: "scale".into(),
expected: "omitted for XRP or MPT vault assets".into(),
found: scale.to_string(),
});
}
if scale > MAX_VAULT_SCALE {
return Err(XRPLModelException::ValueTooHigh {
field: "scale".into(),
max: MAX_VAULT_SCALE as u32,
found: scale as u32,
});
}
}
Ok(())
}
}
impl<'a> Transaction<'a, VaultCreateFlag> for VaultCreate<'a> {
fn has_flag(&self, flag: &VaultCreateFlag) -> bool {
self.common_fields.has_flag(flag)
}
fn get_common_fields(&self) -> &CommonFields<'_, VaultCreateFlag> {
&self.common_fields
}
fn get_mut_common_fields(&mut self) -> &mut CommonFields<'a, VaultCreateFlag> {
&mut self.common_fields
}
fn get_transaction_type(&self) -> &TransactionType {
self.common_fields.get_transaction_type()
}
}
impl<'a> CommonTransactionBuilder<'a, VaultCreateFlag> for VaultCreate<'a> {
fn get_mut_common_fields(&mut self) -> &mut CommonFields<'a, VaultCreateFlag> {
&mut self.common_fields
}
fn into_self(self) -> Self {
self
}
}
impl<'a> VaultCreate<'a> {
#[allow(clippy::too_many_arguments)]
pub fn new(
account: Cow<'a, str>,
account_txn_id: Option<Cow<'a, str>>,
fee: Option<XRPAmount<'a>>,
flags: Option<FlagCollection<VaultCreateFlag>>,
last_ledger_sequence: Option<u32>,
memos: Option<Vec<Memo>>,
sequence: Option<u32>,
signers: Option<Vec<Signer>>,
source_tag: Option<u32>,
ticket_sequence: Option<u32>,
asset: Currency<'a>,
data: Option<Cow<'a, str>>,
assets_maximum: Option<Cow<'a, str>>,
mptoken_metadata: Option<Cow<'a, str>>,
domain_id: Option<Cow<'a, str>>,
withdrawal_policy: Option<u8>,
scale: Option<u8>,
) -> VaultCreate<'a> {
VaultCreate {
common_fields: CommonFields::new(
account,
TransactionType::VaultCreate,
account_txn_id,
fee,
Some(flags.unwrap_or_default()),
last_ledger_sequence,
memos,
None,
sequence,
signers,
None,
source_tag,
ticket_sequence,
None,
),
asset,
data,
assets_maximum,
mptoken_metadata,
domain_id,
withdrawal_policy,
scale,
}
}
pub fn with_data(mut self, data: Cow<'a, str>) -> Self {
self.data = Some(data);
self
}
pub fn with_assets_maximum(mut self, assets_maximum: Cow<'a, str>) -> Self {
self.assets_maximum = Some(assets_maximum);
self
}
pub fn with_mptoken_metadata(mut self, mptoken_metadata: Cow<'a, str>) -> Self {
self.mptoken_metadata = Some(mptoken_metadata);
self
}
pub fn with_domain_id(mut self, domain_id: Cow<'a, str>) -> Self {
self.domain_id = Some(domain_id);
self
}
pub fn with_withdrawal_policy(mut self, withdrawal_policy: u8) -> Self {
self.withdrawal_policy = Some(withdrawal_policy);
self
}
pub fn with_scale(mut self, scale: u8) -> Self {
self.scale = Some(scale);
self
}
pub fn with_flag(mut self, flag: VaultCreateFlag) -> Self {
self.common_fields.flags.0.push(flag);
self
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::models::currency::{IssuedCurrency, XRP};
use alloc::string::String;
#[test]
fn test_serde() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rVaultCreator123".into(),
transaction_type: TransactionType::VaultCreate,
signing_pub_key: Some("".into()),
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new("USD".into(), "rIssuer456".into())),
data: None,
assets_maximum: None,
mptoken_metadata: None,
domain_id: None,
withdrawal_policy: None,
scale: None,
};
let json_str = r#"{"Account":"rVaultCreator123","TransactionType":"VaultCreate","Flags":0,"SigningPubKey":"","Asset":{"currency":"USD","issuer":"rIssuer456"}}"#;
let serialized = serde_json::to_string(&vault_create).unwrap();
assert_eq!(
serde_json::to_value(&serialized).unwrap(),
serde_json::to_value(json_str).unwrap()
);
let deserialized: VaultCreate = serde_json::from_str(json_str).unwrap();
assert_eq!(vault_create, deserialized);
}
#[test]
fn test_serde_with_all_fields() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rVaultCreator789".into(),
transaction_type: TransactionType::VaultCreate,
signing_pub_key: Some("".into()),
..Default::default()
},
asset: Currency::XRP(XRP::new()),
data: Some("48656C6C6F".into()),
assets_maximum: Some("1000000000".into()),
mptoken_metadata: Some("ABCDEF".into()),
domain_id: Some(
"D0000000000000000000000000000000000000000000000000000000DEADBEEF".into(),
),
withdrawal_policy: Some(1),
scale: Some(6),
};
let serialized = serde_json::to_string(&vault_create).unwrap();
let deserialized: VaultCreate = serde_json::from_str(&serialized).unwrap();
assert_eq!(vault_create, deserialized);
}
#[test]
fn test_builder_pattern() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rVaultCreator123".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new("USD".into(), "rIssuer456".into())),
..Default::default()
}
.with_fee("12".into())
.with_sequence(100)
.with_last_ledger_sequence(7108682)
.with_source_tag(12345)
.with_data("48656C6C6F".into())
.with_assets_maximum("1000000000".into())
.with_mptoken_metadata("ABCDEF".into())
.with_domain_id("D0000000000000000000000000000000000000000000000000000000DEADBEEF".into())
.with_withdrawal_policy(1)
.with_scale(6)
.with_memo(Memo {
memo_data: Some("creating vault".into()),
memo_format: None,
memo_type: Some("text".into()),
});
assert_eq!(vault_create.common_fields.fee.as_ref().unwrap().0, "12");
assert_eq!(vault_create.common_fields.sequence, Some(100));
assert_eq!(
vault_create.common_fields.last_ledger_sequence,
Some(7108682)
);
assert_eq!(vault_create.common_fields.source_tag, Some(12345));
assert_eq!(vault_create.data, Some("48656C6C6F".into()));
assert_eq!(vault_create.assets_maximum, Some("1000000000".into()));
assert_eq!(vault_create.mptoken_metadata, Some("ABCDEF".into()));
assert_eq!(vault_create.withdrawal_policy, Some(1));
assert_eq!(vault_create.scale, Some(6));
assert_eq!(vault_create.common_fields.memos.as_ref().unwrap().len(), 1);
}
#[test]
fn test_default() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rVaultCreator123".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new("USD".into(), "rIssuer456".into())),
..Default::default()
};
assert_eq!(vault_create.common_fields.account, "rVaultCreator123");
assert_eq!(
vault_create.common_fields.transaction_type,
TransactionType::VaultCreate
);
assert!(vault_create.data.is_none());
assert!(vault_create.assets_maximum.is_none());
assert!(vault_create.mptoken_metadata.is_none());
assert!(vault_create.domain_id.is_none());
assert!(vault_create.withdrawal_policy.is_none());
assert!(vault_create.scale.is_none());
assert!(vault_create.common_fields.fee.is_none());
assert!(vault_create.common_fields.sequence.is_none());
}
#[test]
fn test_xrp_vault() {
let xrp_vault = VaultCreate {
common_fields: CommonFields {
account: "rXRPVaultCreator789".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::XRP(XRP::new()),
..Default::default()
}
.with_fee("12".into())
.with_sequence(100)
.with_assets_maximum("50000000000".into());
assert!(matches!(xrp_vault.asset, Currency::XRP(_)));
assert_eq!(xrp_vault.assets_maximum, Some("50000000000".into()));
assert_eq!(xrp_vault.common_fields.sequence, Some(100));
assert!(xrp_vault.validate().is_ok());
}
#[test]
fn test_issued_currency_vault() {
let token_vault = VaultCreate {
common_fields: CommonFields {
account: "rTokenVaultCreator111".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new(
"USD".into(),
"rUSDIssuer222".into(),
)),
..Default::default()
}
.with_fee("15".into())
.with_sequence(200)
.with_withdrawal_policy(1);
assert!(matches!(token_vault.asset, Currency::IssuedCurrency(_)));
assert_eq!(token_vault.withdrawal_policy, Some(1));
assert!(token_vault.validate().is_ok());
}
#[test]
fn test_ticket_sequence() {
let ticket_vault = VaultCreate {
common_fields: CommonFields {
account: "rTicketVault333".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::XRP(XRP::new()),
..Default::default()
}
.with_ticket_sequence(12345)
.with_fee("12".into());
assert_eq!(ticket_vault.common_fields.ticket_sequence, Some(12345));
assert!(ticket_vault.common_fields.sequence.is_none());
}
#[test]
fn test_multiple_memos() {
let multi_memo_vault = VaultCreate {
common_fields: CommonFields {
account: "rMultiMemoVault444".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new(
"EUR".into(),
"rEURIssuer555".into(),
)),
..Default::default()
}
.with_memo(Memo {
memo_data: Some("first memo".into()),
memo_format: None,
memo_type: Some("text".into()),
})
.with_memo(Memo {
memo_data: Some("second memo".into()),
memo_format: None,
memo_type: Some("text".into()),
})
.with_fee("18".into())
.with_sequence(400);
assert_eq!(
multi_memo_vault.common_fields.memos.as_ref().unwrap().len(),
2
);
assert_eq!(multi_memo_vault.common_fields.sequence, Some(400));
}
#[test]
fn test_new_constructor() {
let vault = VaultCreate::new(
"rNewVaultAccount".into(),
None,
Some("12".into()),
None,
Some(7108682),
None,
Some(100),
None,
None,
None,
Currency::IssuedCurrency(IssuedCurrency::new("USD".into(), "rIssuer789".into())),
Some("48656C6C6F".into()),
Some("1000000000".into()),
Some("ABCDEF".into()),
Some("D0000000000000000000000000000000000000000000000000000000DEADBEEF".into()),
Some(1),
Some(6),
);
assert_eq!(vault.common_fields.account, "rNewVaultAccount");
assert_eq!(
vault.common_fields.transaction_type,
TransactionType::VaultCreate
);
assert_eq!(vault.common_fields.fee.as_ref().unwrap().0, "12");
assert_eq!(vault.common_fields.last_ledger_sequence, Some(7108682));
assert_eq!(vault.common_fields.sequence, Some(100));
assert_eq!(vault.data, Some("48656C6C6F".into()));
assert_eq!(vault.assets_maximum, Some("1000000000".into()));
assert_eq!(vault.mptoken_metadata, Some("ABCDEF".into()));
assert_eq!(vault.withdrawal_policy, Some(1));
}
#[test]
fn test_get_transaction_type() {
use crate::models::transactions::Transaction;
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rTxTypeTest".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::XRP(XRP::new()),
..Default::default()
};
assert_eq!(
*vault_create.get_transaction_type(),
TransactionType::VaultCreate
);
}
#[test]
fn test_stranded_withdrawal_policy() {
let stranded_vault = VaultCreate {
common_fields: CommonFields {
account: "rStrandedVault666".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new(
"BTC".into(),
"rBTCIssuer777".into(),
)),
withdrawal_policy: Some(1),
..Default::default()
}
.with_fee("12".into())
.with_sequence(500);
assert_eq!(stranded_vault.withdrawal_policy, Some(1));
assert!(stranded_vault.validate().is_ok());
}
#[test]
fn test_vault_create_flag_values() {
assert_eq!(VaultCreateFlag::TfVaultPrivate as u32, 0x00010000);
assert_eq!(
VaultCreateFlag::TfVaultShareNonTransferable as u32,
0x00020000
);
}
#[test]
fn test_vault_create_flags_serialize() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rFlaggedVault".into(),
transaction_type: TransactionType::VaultCreate,
signing_pub_key: Some("".into()),
..Default::default()
},
asset: Currency::XRP(XRP::new()),
..Default::default()
}
.with_flag(VaultCreateFlag::TfVaultPrivate)
.with_flag(VaultCreateFlag::TfVaultShareNonTransferable);
let serialized = serde_json::to_string(&vault_create).unwrap();
assert!(
serialized.contains("\"Flags\":196608"),
"expected combined flag bits 0x30000 in serialized output, got: {serialized}"
);
let deserialized: VaultCreate = serde_json::from_str(&serialized).unwrap();
assert!(deserialized
.common_fields
.flags
.0
.contains(&VaultCreateFlag::TfVaultPrivate));
assert!(deserialized
.common_fields
.flags
.0
.contains(&VaultCreateFlag::TfVaultShareNonTransferable));
}
#[test]
fn test_data_too_long_rejected() {
let oversize: String = "A".repeat(513);
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rDataTooLong".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::XRP(XRP::new()),
data: Some(oversize.into()),
..Default::default()
};
assert!(vault_create.validate().is_err());
}
#[test]
fn test_data_non_hex_rejected() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rDataBadHex".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::XRP(XRP::new()),
data: Some("not-hex!".into()),
..Default::default()
};
assert!(vault_create.validate().is_err());
}
#[test]
fn test_mptoken_metadata_too_long_rejected() {
let oversize: String = "B".repeat(2049);
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rMetaTooLong".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::XRP(XRP::new()),
mptoken_metadata: Some(oversize.into()),
..Default::default()
};
assert!(vault_create.validate().is_err());
}
#[test]
fn test_mptoken_metadata_non_hex_rejected() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rMetaBadHex".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::XRP(XRP::new()),
mptoken_metadata: Some("ZZZZ".into()),
..Default::default()
};
assert!(vault_create.validate().is_err());
}
#[test]
fn test_domain_id_without_private_flag_rejected() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rVaultNoDomain".into(),
transaction_type: TransactionType::VaultCreate,
fee: Some("12".into()),
sequence: Some(1),
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new("USD".into(), "rIssuer".into())),
domain_id: Some(
"D0000000000000000000000000000000000000000000000000000000DEADBEEF".into(),
),
..Default::default()
};
assert!(vault_create.validate().is_err());
let err = vault_create.validate().unwrap_err().to_string();
assert!(
err.contains("domain_id"),
"expected domain_id in error, got: {err}"
);
}
#[test]
fn test_domain_id_with_private_flag_accepted() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rPrivateVault".into(),
transaction_type: TransactionType::VaultCreate,
fee: Some("12".into()),
sequence: Some(1),
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new("USD".into(), "rIssuer".into())),
domain_id: Some(
"D0000000000000000000000000000000000000000000000000000000DEADBEEF".into(),
),
..Default::default()
}
.with_flag(VaultCreateFlag::TfVaultPrivate)
.with_withdrawal_policy(1);
assert!(vault_create.validate().is_ok());
}
#[test]
fn test_assets_maximum_xrpl_number_accepted() {
for assets_maximum in ["100.5", "1e6", "9999999999999999e80"] {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rVaultNumber".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::XRP(XRP::new()),
assets_maximum: Some(assets_maximum.into()),
..Default::default()
};
assert!(
vault_create.validate().is_ok(),
"{assets_maximum} should validate"
);
}
}
#[test]
fn test_assets_maximum_integer_accepted() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rVaultMax".into(),
transaction_type: TransactionType::VaultCreate,
fee: Some("12".into()),
sequence: Some(1),
..Default::default()
},
asset: Currency::XRP(XRP::new()),
assets_maximum: Some("1000000000".into()),
..Default::default()
};
assert!(vault_create.validate().is_ok());
}
#[test]
fn test_builder_pattern_validates_correctly() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rVaultCreator123".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new("USD".into(), "rIssuer456".into())),
..Default::default()
}
.with_fee("12".into())
.with_sequence(100)
.with_domain_id("D0000000000000000000000000000000000000000000000000000000DEADBEEF".into())
.with_withdrawal_policy(1);
assert!(vault_create.validate().is_err());
}
#[test]
fn test_scale_on_mpt_asset_rejected() {
use crate::models::currency::MPTCurrency;
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rMPTScaleVault".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::MPTCurrency(MPTCurrency::new(
"000000016B4E90A4B36D74F6E16A5BED41EBD7AA37B19B89".into(),
)),
scale: Some(6),
..Default::default()
};
assert!(vault_create.validate().is_err());
}
#[test]
fn test_scale_on_xrp_asset_rejected() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rXRPScaleVault".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::XRP(XRP::new()),
scale: Some(0),
..Default::default()
};
assert!(vault_create.validate().is_err());
}
#[test]
fn test_scale_too_high_rejected() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rScaleTooHigh".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new("USD".into(), "rIssuer".into())),
scale: Some(19), ..Default::default()
};
assert!(vault_create.validate().is_err());
}
#[test]
fn test_scale_zero_on_iou_accepted() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rScaleZero".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new("USD".into(), "rIssuer".into())),
scale: Some(0),
..Default::default()
};
assert!(vault_create.validate().is_ok());
}
#[test]
fn test_scale_max_on_iou_accepted() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rScaleMax".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new("USD".into(), "rIssuer".into())),
scale: Some(18),
..Default::default()
};
assert!(vault_create.validate().is_ok());
}
#[test]
fn test_invalid_withdrawal_policy_rejected() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rBadPolicy".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::IssuedCurrency(IssuedCurrency::new("USD".into(), "rIssuer".into())),
withdrawal_policy: Some(2), ..Default::default()
};
assert!(vault_create.validate().is_err());
}
#[test]
fn test_assets_maximum_negative_rejected() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rNegMaxVault".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::XRP(XRP::new()),
assets_maximum: Some("-1000".into()),
..Default::default()
};
assert!(vault_create.validate().is_err());
}
#[test]
fn test_assets_maximum_non_numeric_rejected() {
let vault_create = VaultCreate {
common_fields: CommonFields {
account: "rBadMaxVault".into(),
transaction_type: TransactionType::VaultCreate,
..Default::default()
},
asset: Currency::XRP(XRP::new()),
assets_maximum: Some("not-a-number".into()),
..Default::default()
};
assert!(vault_create.validate().is_err());
}
}