1use super::{db::SqliteStorage, error::PersistResult};
2use crate::{FullReverseSwapInfo, ReverseSwapInfoCached, ReverseSwapStatus};
3use rusqlite::{named_params, OptionalExtension, Row, TransactionBehavior};
4
5impl SqliteStorage {
6 pub(crate) fn insert_reverse_swap(&self, rsi: &FullReverseSwapInfo) -> PersistResult<()> {
7 let mut con = self.get_connection()?;
8 let tx = con.transaction_with_behavior(TransactionBehavior::Immediate)?;
9
10 tx.execute(
11 "INSERT INTO sync.reverse_swaps (id, created_at_block_height, preimage, private_key, claim_pubkey, timeout_block_height, invoice, onchain_amount_sat, sat_per_vbyte, receive_amount_sat, redeem_script)\
12 VALUES (:id, :created_at_block_height, :preimage, :private_key, :claim_pubkey, :timeout_block_height, :invoice, :onchain_amount_sat, :sat_per_vbyte, :receive_amount_sat, :redeem_script)",
13 named_params! {
14 ":id": rsi.id,
15 ":created_at_block_height": rsi.created_at_block_height,
16 ":preimage": rsi.preimage,
17 ":private_key": rsi.private_key,
18 ":claim_pubkey": rsi.claim_pubkey,
19 ":timeout_block_height": rsi.timeout_block_height,
20 ":invoice": rsi.invoice,
21 ":onchain_amount_sat": rsi.onchain_amount_sat,
22 ":sat_per_vbyte": rsi.sat_per_vbyte,
23 ":receive_amount_sat": rsi.receive_amount_sat,
24 ":redeem_script": rsi.redeem_script
25 },
26 )?;
27
28 tx.execute(
29 "INSERT INTO reverse_swaps_info (id, status)\
30 VALUES (:id, :status)",
31 named_params! {
32 ":id": rsi.id,
33 ":status": serde_json::to_value(rsi.cache.status)?
34 },
35 )?;
36
37 tx.commit()?;
38 Ok(())
39 }
40
41 pub(crate) fn update_reverse_swap_status(
42 &self,
43 id: &str,
44 status: &ReverseSwapStatus,
45 ) -> PersistResult<()> {
46 debug!("Persisting new status for reverse swap {id} to be {status:?}");
47
48 self.get_connection()?.execute(
49 "INSERT OR REPLACE INTO reverse_swaps_info(id, status) VALUES(:id, :status)",
50 named_params! {
51 ":status": serde_json::to_value(status)?,
52 ":id": id,
53 },
54 )?;
55
56 Ok(())
57 }
58
59 pub(crate) fn update_reverse_swap_lockup_txid(
60 &self,
61 id: &str,
62 lockup_txid: Option<String>,
63 ) -> PersistResult<()> {
64 debug!("Updating lockup_txid for reverse swap {id} to be - lockup_txid: {lockup_txid:?}");
65
66 self.get_connection()?.execute(
67 "UPDATE reverse_swaps_info SET lockup_txid = :lockup_txid WHERE id = :id",
68 named_params! {
69 ":id": id,
70 ":lockup_txid": lockup_txid,
71 },
72 )?;
73
74 Ok(())
75 }
76
77 pub(crate) fn update_reverse_swap_claim_txid(
78 &self,
79 id: &str,
80 claim_txid: Option<String>,
81 ) -> PersistResult<()> {
82 debug!("Updating claim_txid for reverse swap {id} to be - claim_txid: {claim_txid:?}");
83
84 self.get_connection()?.execute(
85 "UPDATE reverse_swaps_info SET claim_txid = :claim_txid WHERE id = :id",
86 named_params! {
87 ":id": id,
88 ":claim_txid": claim_txid,
89 },
90 )?;
91
92 Ok(())
93 }
94
95 pub(crate) fn get_reverse_swap(&self, id: &str) -> PersistResult<Option<FullReverseSwapInfo>> {
96 Ok(self
97 .get_connection()?
98 .query_row(
99 &self.select_reverse_swap_query("reverse_swaps.id = ?1", ""),
100 [id],
101 |row| self.sql_row_to_reverse_swap(row, ""),
102 )
103 .optional()?)
104 }
105
106 pub(crate) fn list_reverse_swaps(&self) -> PersistResult<Vec<FullReverseSwapInfo>> {
107 let con = self.get_connection()?;
108 let mut stmt = con.prepare(&self.select_reverse_swap_query("true", ""))?;
109
110 let vec: Vec<FullReverseSwapInfo> = stmt
111 .query_map([], |row| self.sql_row_to_reverse_swap(row, ""))?
112 .map(|i| i.unwrap())
113 .collect();
114
115 Ok(vec)
116 }
117
118 pub(crate) fn select_reverse_swap_fields(&self, prefix: &str) -> String {
119 format!(
120 "
121 {prefix}id,
122 {prefix}created_at_block_height,
123 {prefix}preimage,
124 {prefix}private_key,
125 {prefix}timeout_block_height,
126 {prefix}claim_pubkey,
127 {prefix}invoice,
128 {prefix}onchain_amount_sat,
129 {prefix}sat_per_vbyte,
130 {prefix}receive_amount_sat,
131 {prefix}redeem_script,
132 {prefix}status,
133 {prefix}lockup_txid,
134 {prefix}claim_txid
135 "
136 )
137 }
138
139 pub(crate) fn sql_row_to_reverse_swap(
140 &self,
141 row: &Row,
142 prefix: &str,
143 ) -> PersistResult<FullReverseSwapInfo, rusqlite::Error> {
144 Ok(FullReverseSwapInfo {
145 id: row.get(format!("{prefix}id").as_str())?,
146 created_at_block_height: row
147 .get(format!("{prefix}created_at_block_height").as_str())?,
148 preimage: row.get(format!("{prefix}preimage").as_str())?,
149 private_key: row.get(format!("{prefix}private_key").as_str())?,
150 timeout_block_height: row.get(format!("{prefix}timeout_block_height").as_str())?,
151 claim_pubkey: row.get(format!("{prefix}claim_pubkey").as_str())?,
152 invoice: row.get(format!("{prefix}invoice").as_str())?,
153 onchain_amount_sat: row.get(format!("{prefix}onchain_amount_sat").as_str())?,
154 sat_per_vbyte: row.get(format!("{prefix}sat_per_vbyte").as_str())?,
155 receive_amount_sat: row.get(format!("{prefix}receive_amount_sat").as_str())?,
156 redeem_script: row.get(format!("{prefix}redeem_script").as_str())?,
157 cache: ReverseSwapInfoCached {
158 status: serde_json::from_value(row.get(format!("{prefix}status").as_str())?)
161 .unwrap_or(ReverseSwapStatus::Initial),
162 lockup_txid: row.get(format!("{prefix}lockup_txid").as_str())?,
163 claim_txid: row.get(format!("{prefix}claim_txid").as_str())?,
164 },
165 })
166 }
167
168 pub(crate) fn select_reverse_swap_query(&self, where_clause: &str, prefix: &str) -> String {
169 let fields = format!(
170 "
171 reverse_swaps.id as {prefix}id,
172 created_at_block_height as {prefix}created_at_block_height,
173 preimage as {prefix}preimage,
174 private_key as {prefix}private_key,
175 timeout_block_height as {prefix}timeout_block_height,
176 claim_pubkey as {prefix}claim_pubkey,
177 invoice as {prefix}invoice,
178 onchain_amount_sat as {prefix}onchain_amount_sat,
179 sat_per_vbyte as {prefix}sat_per_vbyte,
180 receive_amount_sat as {prefix}receive_amount_sat,
181 redeem_script as {prefix}redeem_script,
182 status as {prefix}status,
183 lockup_txid as {prefix}lockup_txid,
184 claim_txid as {prefix}claim_txid
185 "
186 );
187
188 format!(
189 "
190 SELECT
191 {fields}
192 FROM sync.reverse_swaps
193 LEFT JOIN reverse_swaps_info ON reverse_swaps.id = reverse_swaps_info.id
194 WHERE {}
195 ",
196 where_clause
197 )
198 }
199}