migrations.rs 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. use std::collections::HashMap;
  2. use std::sync::Arc;
  3. use cdk_common::mint_url::MintUrl;
  4. use cdk_common::{Amount, CurrencyUnit, MeltQuoteState, MintQuoteState};
  5. use redb::{Database, ReadableTable, TableDefinition};
  6. use serde::{Deserialize, Serialize};
  7. use super::error::Error;
  8. const MINT_QUOTES_TABLE: TableDefinition<&str, &str> = TableDefinition::new("mint_quotes");
  9. const MELT_QUOTES_TABLE: TableDefinition<&str, &str> = TableDefinition::new("melt_quotes");
  10. /// Mint Quote Info
  11. #[derive(Debug, Clone, Hash, PartialEq, Eq, Serialize, Deserialize)]
  12. pub struct V0MintQuote {
  13. pub id: String,
  14. pub mint_url: MintUrl,
  15. pub amount: Amount,
  16. pub unit: CurrencyUnit,
  17. pub request: String,
  18. pub paid: bool,
  19. pub expiry: u64,
  20. }
  21. /// Mint Quote Info
  22. #[derive(Debug, Clone, Hash, PartialEq, Eq, Serialize, Deserialize)]
  23. pub struct V1MintQuote {
  24. pub id: String,
  25. pub mint_url: MintUrl,
  26. pub amount: Amount,
  27. pub unit: CurrencyUnit,
  28. pub request: String,
  29. pub state: MintQuoteState,
  30. pub expiry: u64,
  31. }
  32. /// Melt Quote Info
  33. #[derive(Debug, Clone, Hash, PartialEq, Eq, Serialize, Deserialize)]
  34. pub struct V1MeltQuote {
  35. /// Quote id
  36. pub id: String,
  37. /// Quote unit
  38. pub unit: CurrencyUnit,
  39. /// Quote amount
  40. pub amount: Amount,
  41. /// Quote Payment request e.g. bolt11
  42. pub request: String,
  43. /// Quote fee reserve
  44. pub fee_reserve: Amount,
  45. /// Quote state
  46. pub state: MeltQuoteState,
  47. /// Expiration time of quote
  48. pub expiry: u64,
  49. /// Payment preimage
  50. pub payment_preimage: Option<String>,
  51. }
  52. impl From<V0MintQuote> for V1MintQuote {
  53. fn from(quote: V0MintQuote) -> V1MintQuote {
  54. let state = match quote.paid {
  55. true => MintQuoteState::Paid,
  56. false => MintQuoteState::Unpaid,
  57. };
  58. V1MintQuote {
  59. id: quote.id,
  60. mint_url: quote.mint_url,
  61. amount: quote.amount,
  62. unit: quote.unit,
  63. request: quote.request.clone(),
  64. state,
  65. expiry: quote.expiry,
  66. }
  67. }
  68. }
  69. /// Melt Quote Info
  70. #[derive(Debug, Clone, Hash, PartialEq, Eq, Serialize, Deserialize)]
  71. pub struct V0MeltQuote {
  72. pub id: String,
  73. pub unit: CurrencyUnit,
  74. pub amount: Amount,
  75. pub request: String,
  76. pub fee_reserve: Amount,
  77. pub paid: bool,
  78. pub expiry: u64,
  79. }
  80. impl From<V0MeltQuote> for V1MeltQuote {
  81. fn from(quote: V0MeltQuote) -> V1MeltQuote {
  82. let state = match quote.paid {
  83. true => MeltQuoteState::Paid,
  84. false => MeltQuoteState::Unpaid,
  85. };
  86. V1MeltQuote {
  87. id: quote.id,
  88. amount: quote.amount,
  89. unit: quote.unit,
  90. request: quote.request,
  91. state,
  92. expiry: quote.expiry,
  93. fee_reserve: quote.fee_reserve,
  94. payment_preimage: None,
  95. }
  96. }
  97. }
  98. fn migrate_mint_quotes_00_to_01(db: Arc<Database>) -> Result<(), Error> {
  99. let read_txn = db.begin_read().map_err(Error::from)?;
  100. let table = read_txn
  101. .open_table(MINT_QUOTES_TABLE)
  102. .map_err(Error::from)?;
  103. let mint_quotes: HashMap<String, Option<V0MintQuote>>;
  104. {
  105. mint_quotes = table
  106. .iter()
  107. .map_err(Error::from)?
  108. .flatten()
  109. .map(|(quote_id, mint_quote)| {
  110. (
  111. quote_id.value().to_string(),
  112. serde_json::from_str(mint_quote.value()).ok(),
  113. )
  114. })
  115. .collect();
  116. }
  117. let migrated_mint_quotes: HashMap<String, Option<V1MintQuote>> = mint_quotes
  118. .into_iter()
  119. .map(|(quote_id, quote)| (quote_id, quote.map(|q| q.into())))
  120. .collect();
  121. {
  122. let write_txn = db.begin_write()?;
  123. {
  124. let mut table = write_txn
  125. .open_table(MINT_QUOTES_TABLE)
  126. .map_err(Error::from)?;
  127. for (quote_id, quote) in migrated_mint_quotes {
  128. match quote {
  129. Some(quote) => {
  130. let quote_str = serde_json::to_string(&quote)?;
  131. table.insert(quote_id.as_str(), quote_str.as_str())?;
  132. }
  133. None => {
  134. table.remove(quote_id.as_str())?;
  135. }
  136. }
  137. }
  138. }
  139. write_txn.commit()?;
  140. }
  141. Ok(())
  142. }
  143. fn migrate_melt_quotes_00_to_01(db: Arc<Database>) -> Result<(), Error> {
  144. let read_txn = db.begin_read().map_err(Error::from)?;
  145. let table = read_txn
  146. .open_table(MELT_QUOTES_TABLE)
  147. .map_err(Error::from)?;
  148. let melt_quotes: HashMap<String, Option<V0MeltQuote>>;
  149. {
  150. melt_quotes = table
  151. .iter()
  152. .map_err(Error::from)?
  153. .flatten()
  154. .map(|(quote_id, melt_quote)| {
  155. (
  156. quote_id.value().to_string(),
  157. serde_json::from_str(melt_quote.value()).ok(),
  158. )
  159. })
  160. .collect();
  161. }
  162. let migrated_melt_quotes: HashMap<String, Option<V1MeltQuote>> = melt_quotes
  163. .into_iter()
  164. .map(|(quote_id, quote)| (quote_id, quote.map(|q| q.into())))
  165. .collect();
  166. {
  167. let write_txn = db.begin_write()?;
  168. {
  169. let mut table = write_txn
  170. .open_table(MELT_QUOTES_TABLE)
  171. .map_err(Error::from)?;
  172. for (quote_id, quote) in migrated_melt_quotes {
  173. match quote {
  174. Some(quote) => {
  175. let quote_str = serde_json::to_string(&quote)?;
  176. table.insert(quote_id.as_str(), quote_str.as_str())?;
  177. }
  178. None => {
  179. table.remove(quote_id.as_str())?;
  180. }
  181. }
  182. }
  183. }
  184. write_txn.commit()?;
  185. }
  186. Ok(())
  187. }
  188. pub(crate) fn migrate_00_to_01(db: Arc<Database>) -> Result<u32, Error> {
  189. tracing::info!("Starting Migrations of mint quotes from 00 to 01");
  190. migrate_mint_quotes_00_to_01(Arc::clone(&db))?;
  191. tracing::info!("Finished Migrations of mint quotes from 00 to 01");
  192. tracing::info!("Starting Migrations of melt quotes from 00 to 01");
  193. migrate_melt_quotes_00_to_01(Arc::clone(&db))?;
  194. tracing::info!("Finished Migrations of melt quotes from 00 to 01");
  195. Ok(1)
  196. }