status.rs 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. use crate::MaxLengthString;
  2. use serde::{Deserialize, Serialize};
  3. use std::collections::HashMap;
  4. /// Status type
  5. pub type Status = MaxLengthString<10>;
  6. #[derive(Debug, Serialize, Deserialize)]
  7. pub struct StatusManager {
  8. statuses: Vec<Status>,
  9. spendable: Vec<Status>,
  10. default_spendable: Status,
  11. reverted: Vec<Status>,
  12. withdrawable: Vec<Status>,
  13. transition: HashMap<Status, Vec<Status>>,
  14. }
  15. /// Status error object
  16. #[derive(Debug, Serialize, thiserror::Error)]
  17. pub enum Error {
  18. #[error("Unknown status: {0}")]
  19. UnknownStatus(String),
  20. #[error("Invalid transition from {0} to {1}")]
  21. InvalidTransition(Status, Status),
  22. }
  23. impl StatusManager {
  24. pub fn new_status(&self, status_name: &str) -> Result<Status, Error> {
  25. self.statuses
  26. .iter()
  27. .find(|status| *status == status_name)
  28. .cloned()
  29. .ok_or(Error::UnknownStatus(status_name.to_owned()))
  30. }
  31. pub fn spendables(&self) -> &[Status] {
  32. &self.spendable
  33. }
  34. pub fn default_spendable(&self) -> Status {
  35. self.default_spendable.clone()
  36. }
  37. /// Spendable statuses are the final state of a transaction that has finished as successfully
  38. /// therefore it is spendable.
  39. pub fn is_spendable(&self, status: &Status) -> bool {
  40. self.spendable.contains(status)
  41. }
  42. /// Reverted transactions are the final state of a transaction that has failed or cancelled and
  43. /// they inputs payments are reverted and spendedable.
  44. pub fn is_reverted(&self, status: &Status) -> bool {
  45. self.reverted.contains(status)
  46. }
  47. /// The transaction is final and cannot longer be updated
  48. pub fn is_final(&self, status: &Status) -> bool {
  49. self.is_spendable(status) || self.is_reverted(status)
  50. }
  51. /// Checks if the status transition is allowed
  52. pub fn is_valid_transition(&self, from: &Status, to: &Status) -> Result<(), Error> {
  53. if self.transition.get(from).map_or(false, |v| v.contains(to)) {
  54. Ok(())
  55. } else {
  56. Err(Error::InvalidTransition(from.clone(), to.clone()))
  57. }
  58. }
  59. }
  60. impl Default for StatusManager {
  61. fn default() -> Self {
  62. let pending: Status = "pending".into();
  63. let processing: Status = "processing".into();
  64. let cancelled: Status = "cancelled".into();
  65. let settled: Status = "settled".into();
  66. let failed: Status = "failed".into();
  67. Self {
  68. statuses: vec![
  69. pending.clone(),
  70. processing.clone(),
  71. cancelled.clone(),
  72. settled.clone(),
  73. failed.clone(),
  74. ],
  75. default_spendable: settled.clone(),
  76. spendable: vec![settled.clone()],
  77. reverted: vec![cancelled.clone(), failed.clone()],
  78. withdrawable: vec![settled.clone()],
  79. transition: {
  80. let mut map = HashMap::new();
  81. map.insert(
  82. pending.clone(),
  83. vec![processing.clone(), settled.clone(), cancelled.clone()],
  84. );
  85. map.insert(processing.clone(), vec![settled, failed]);
  86. map
  87. },
  88. }
  89. }
  90. }