migrations.rs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334
  1. use core::str;
  2. use std::collections::HashMap;
  3. use std::str::FromStr;
  4. use std::sync::Arc;
  5. use cdk_common::mint::MintQuote;
  6. use cdk_common::mint_url::MintUrl;
  7. use cdk_common::{Amount, CurrencyUnit, MintQuoteState, Proof, State};
  8. use lightning_invoice::Bolt11Invoice;
  9. use redb::{
  10. Database, MultimapTableDefinition, ReadableMultimapTable, ReadableTable, TableDefinition,
  11. };
  12. use serde::{Deserialize, Serialize};
  13. use uuid::Uuid;
  14. use super::{Error, PROOFS_STATE_TABLE, PROOFS_TABLE, QUOTE_SIGNATURES_TABLE};
  15. const ID_STR_MINT_QUOTES_TABLE: TableDefinition<&str, &str> = TableDefinition::new("mint_quotes");
  16. const PENDING_PROOFS_TABLE: TableDefinition<[u8; 33], &str> =
  17. TableDefinition::new("pending_proofs");
  18. const SPENT_PROOFS_TABLE: TableDefinition<[u8; 33], &str> = TableDefinition::new("spent_proofs");
  19. const QUOTE_PROOFS_TABLE: MultimapTableDefinition<&str, [u8; 33]> =
  20. MultimapTableDefinition::new("quote_proofs");
  21. pub fn migrate_01_to_02(db: Arc<Database>) -> Result<u32, Error> {
  22. migrate_mint_quotes_01_to_02(db)?;
  23. Ok(2)
  24. }
  25. pub fn migrate_02_to_03(db: Arc<Database>) -> Result<u32, Error> {
  26. migrate_mint_proofs_02_to_03(db)?;
  27. Ok(3)
  28. }
  29. pub fn migrate_03_to_04(db: Arc<Database>) -> Result<u32, Error> {
  30. let write_txn = db.begin_write()?;
  31. let _ = write_txn.open_multimap_table(QUOTE_PROOFS_TABLE)?;
  32. let _ = write_txn.open_multimap_table(QUOTE_SIGNATURES_TABLE)?;
  33. Ok(4)
  34. }
  35. pub fn migrate_04_to_05(db: Arc<Database>) -> Result<u32, Error> {
  36. let write_txn = db.begin_write()?;
  37. // Mint quotes
  38. {
  39. const MINT_QUOTE_TABLE_NAME: &str = "mint_quotes";
  40. const OLD_TABLE: TableDefinition<&str, &str> = TableDefinition::new(MINT_QUOTE_TABLE_NAME);
  41. const NEW_TABLE: TableDefinition<[u8; 16], &str> =
  42. TableDefinition::new(MINT_QUOTE_TABLE_NAME);
  43. let old_table = write_txn.open_table(OLD_TABLE)?;
  44. let mut tmp_hashmap = HashMap::new();
  45. for (k, v) in old_table.iter().map_err(Error::from)?.flatten() {
  46. let quote_id = Uuid::try_parse(k.value()).unwrap();
  47. tmp_hashmap.insert(quote_id, v.value().to_string());
  48. }
  49. write_txn.delete_table(old_table).map_err(Error::from)?;
  50. let mut new_table = write_txn.open_table(NEW_TABLE)?;
  51. for (k, v) in tmp_hashmap.into_iter() {
  52. new_table
  53. .insert(k.as_bytes(), v.as_str())
  54. .map_err(Error::from)?;
  55. }
  56. }
  57. // Melt quotes
  58. {
  59. const MELT_QUOTE_TABLE_NAME: &str = "melt_quotes";
  60. const OLD_TABLE: TableDefinition<&str, &str> = TableDefinition::new(MELT_QUOTE_TABLE_NAME);
  61. const NEW_TABLE: TableDefinition<[u8; 16], &str> =
  62. TableDefinition::new(MELT_QUOTE_TABLE_NAME);
  63. let old_table = write_txn.open_table(OLD_TABLE)?;
  64. let mut tmp_hashmap = HashMap::new();
  65. for (k, v) in old_table.iter().map_err(Error::from)?.flatten() {
  66. let quote_id = Uuid::try_parse(k.value()).unwrap();
  67. tmp_hashmap.insert(quote_id, v.value().to_string());
  68. }
  69. write_txn.delete_table(old_table).map_err(Error::from)?;
  70. let mut new_table = write_txn.open_table(NEW_TABLE)?;
  71. for (k, v) in tmp_hashmap.into_iter() {
  72. new_table
  73. .insert(k.as_bytes(), v.as_str())
  74. .map_err(Error::from)?;
  75. }
  76. }
  77. // Quote proofs
  78. {
  79. const QUOTE_PROOFS_TABLE_NAME: &str = "quote_proofs";
  80. const OLD_TABLE: MultimapTableDefinition<&str, [u8; 33]> =
  81. MultimapTableDefinition::new(QUOTE_PROOFS_TABLE_NAME);
  82. const NEW_TABLE: MultimapTableDefinition<[u8; 16], [u8; 33]> =
  83. MultimapTableDefinition::new(QUOTE_PROOFS_TABLE_NAME);
  84. let old_table = write_txn.open_multimap_table(OLD_TABLE)?;
  85. let mut tmp_hashmap = HashMap::new();
  86. for (k, v) in old_table.iter().map_err(Error::from)?.flatten() {
  87. let quote_id = Uuid::try_parse(k.value()).unwrap();
  88. let ys: Vec<[u8; 33]> = v.into_iter().flatten().map(|v| v.value()).collect();
  89. tmp_hashmap.insert(quote_id, ys);
  90. }
  91. write_txn
  92. .delete_multimap_table(old_table)
  93. .map_err(Error::from)?;
  94. let mut new_table = write_txn.open_multimap_table(NEW_TABLE)?;
  95. for (quote_id, blind_messages) in tmp_hashmap.into_iter() {
  96. for blind_message in blind_messages {
  97. new_table
  98. .insert(quote_id.as_bytes(), blind_message)
  99. .map_err(Error::from)?;
  100. }
  101. }
  102. }
  103. // Quote signatures
  104. {
  105. const QUOTE_SIGNATURES_TABLE_NAME: &str = "quote_signatures";
  106. const OLD_TABLE: MultimapTableDefinition<&str, [u8; 33]> =
  107. MultimapTableDefinition::new(QUOTE_SIGNATURES_TABLE_NAME);
  108. const NEW_TABLE: MultimapTableDefinition<[u8; 16], [u8; 33]> =
  109. MultimapTableDefinition::new(QUOTE_SIGNATURES_TABLE_NAME);
  110. let old_table = write_txn.open_multimap_table(OLD_TABLE)?;
  111. let mut tmp_hashmap = HashMap::new();
  112. for (k, v) in old_table.iter().map_err(Error::from)?.flatten() {
  113. let quote_id = Uuid::try_parse(k.value()).unwrap();
  114. let ys: Vec<[u8; 33]> = v.into_iter().flatten().map(|v| v.value()).collect();
  115. tmp_hashmap.insert(quote_id, ys);
  116. }
  117. write_txn
  118. .delete_multimap_table(old_table)
  119. .map_err(Error::from)?;
  120. let mut new_table = write_txn.open_multimap_table(NEW_TABLE)?;
  121. for (quote_id, signatures) in tmp_hashmap.into_iter() {
  122. for signature in signatures {
  123. new_table
  124. .insert(quote_id.as_bytes(), signature)
  125. .map_err(Error::from)?;
  126. }
  127. }
  128. }
  129. // Melt requests
  130. {
  131. const MELT_REQUESTS_TABLE_NAME: &str = "melt_requests";
  132. const OLD_TABLE: TableDefinition<&str, (&str, &str)> =
  133. TableDefinition::new(MELT_REQUESTS_TABLE_NAME);
  134. const NEW_TABLE: TableDefinition<[u8; 16], (&str, &str)> =
  135. TableDefinition::new(MELT_REQUESTS_TABLE_NAME);
  136. let old_table = write_txn.open_table(OLD_TABLE)?;
  137. let mut tmp_hashmap = HashMap::new();
  138. for (k, v) in old_table.iter().map_err(Error::from)?.flatten() {
  139. let quote_id = Uuid::try_parse(k.value()).unwrap();
  140. let value = v.value();
  141. tmp_hashmap.insert(quote_id, (value.0.to_string(), value.1.to_string()));
  142. }
  143. write_txn.delete_table(old_table).map_err(Error::from)?;
  144. let mut new_table = write_txn.open_table(NEW_TABLE)?;
  145. for (k, v) in tmp_hashmap.into_iter() {
  146. new_table
  147. .insert(k.as_bytes(), (v.0.as_str(), v.1.as_str()))
  148. .map_err(Error::from)?;
  149. }
  150. }
  151. write_txn.commit().map_err(Error::from)?;
  152. Ok(5)
  153. }
  154. /// Mint Quote Info
  155. #[derive(Debug, Clone, Hash, PartialEq, Eq, Serialize, Deserialize)]
  156. struct V1MintQuote {
  157. pub id: Uuid,
  158. pub mint_url: MintUrl,
  159. pub amount: Amount,
  160. pub unit: CurrencyUnit,
  161. pub request: String,
  162. pub state: MintQuoteState,
  163. pub expiry: u64,
  164. }
  165. impl From<V1MintQuote> for MintQuote {
  166. fn from(quote: V1MintQuote) -> MintQuote {
  167. MintQuote {
  168. id: quote.id,
  169. amount: quote.amount,
  170. unit: quote.unit,
  171. request: quote.request.clone(),
  172. state: quote.state,
  173. expiry: quote.expiry,
  174. request_lookup_id: Bolt11Invoice::from_str(&quote.request).unwrap().to_string(),
  175. pubkey: None,
  176. }
  177. }
  178. }
  179. fn migrate_mint_quotes_01_to_02(db: Arc<Database>) -> Result<(), Error> {
  180. let read_txn = db.begin_read().map_err(Error::from)?;
  181. let table = read_txn
  182. .open_table(ID_STR_MINT_QUOTES_TABLE)
  183. .map_err(Error::from)?;
  184. let mint_quotes: HashMap<String, Option<V1MintQuote>>;
  185. {
  186. mint_quotes = table
  187. .iter()
  188. .map_err(Error::from)?
  189. .flatten()
  190. .map(|(quote_id, mint_quote)| {
  191. (
  192. quote_id.value().to_string(),
  193. serde_json::from_str(mint_quote.value()).ok(),
  194. )
  195. })
  196. .collect();
  197. }
  198. let migrated_mint_quotes: HashMap<String, Option<MintQuote>> = mint_quotes
  199. .into_iter()
  200. .map(|(quote_id, quote)| (quote_id, quote.map(|q| q.into())))
  201. .collect();
  202. {
  203. let write_txn = db.begin_write()?;
  204. {
  205. let mut table = write_txn
  206. .open_table(ID_STR_MINT_QUOTES_TABLE)
  207. .map_err(Error::from)?;
  208. for (quote_id, quote) in migrated_mint_quotes {
  209. match quote {
  210. Some(quote) => {
  211. let quote_str = serde_json::to_string(&quote)?;
  212. table.insert(quote_id.as_str(), quote_str.as_str())?;
  213. }
  214. None => {
  215. table.remove(quote_id.as_str())?;
  216. }
  217. }
  218. }
  219. }
  220. write_txn.commit()?;
  221. }
  222. Ok(())
  223. }
  224. fn migrate_mint_proofs_02_to_03(db: Arc<Database>) -> Result<(), Error> {
  225. let pending_proofs: Vec<([u8; 33], Option<Proof>)>;
  226. let spent_proofs: Vec<([u8; 33], Option<Proof>)>;
  227. {
  228. let read_txn = db.begin_read().map_err(Error::from)?;
  229. let table = read_txn
  230. .open_table(PENDING_PROOFS_TABLE)
  231. .map_err(Error::from)?;
  232. pending_proofs = table
  233. .iter()
  234. .map_err(Error::from)?
  235. .flatten()
  236. .map(|(quote_id, mint_quote)| {
  237. (
  238. quote_id.value(),
  239. serde_json::from_str(mint_quote.value()).ok(),
  240. )
  241. })
  242. .collect();
  243. }
  244. {
  245. let read_txn = db.begin_read().map_err(Error::from)?;
  246. let table = read_txn
  247. .open_table(SPENT_PROOFS_TABLE)
  248. .map_err(Error::from)?;
  249. spent_proofs = table
  250. .iter()
  251. .map_err(Error::from)?
  252. .flatten()
  253. .map(|(quote_id, mint_quote)| {
  254. (
  255. quote_id.value(),
  256. serde_json::from_str(mint_quote.value()).ok(),
  257. )
  258. })
  259. .collect();
  260. }
  261. let write_txn = db.begin_write().map_err(Error::from)?;
  262. {
  263. let mut proofs_table = write_txn.open_table(PROOFS_TABLE).map_err(Error::from)?;
  264. let mut state_table = write_txn
  265. .open_table(PROOFS_STATE_TABLE)
  266. .map_err(Error::from)?;
  267. for (y, proof) in pending_proofs {
  268. if let Some(proof) = proof {
  269. proofs_table.insert(y, serde_json::to_string(&proof)?.as_str())?;
  270. state_table.insert(y, State::Pending.to_string().as_str())?;
  271. }
  272. }
  273. for (y, proof) in spent_proofs {
  274. if let Some(proof) = proof {
  275. proofs_table.insert(y, serde_json::to_string(&proof)?.as_str())?;
  276. state_table.insert(y, State::Spent.to_string().as_str())?;
  277. }
  278. }
  279. }
  280. write_txn.commit()?;
  281. Ok(())
  282. }