]> Untitled Git - bitcoindevkit.org/blob
f52a1c8b9ae6824f20d65bea18688b69d02069f7
[bitcoindevkit.org] /
1 <!DOCTYPE html><html lang="en"><head><meta charset="utf-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><meta name="generator" content="rustdoc"><meta name="description" content="Source of the Rust file `src/wallet/coin_selection.rs`."><meta name="keywords" content="rust, rustlang, rust-lang"><title>coin_selection.rs - source</title><link rel="stylesheet" type="text/css" href="../../../normalize.css"><link rel="stylesheet" type="text/css" href="../../../rustdoc.css" id="mainThemeStyle"><link rel="stylesheet" type="text/css" href="../../../light.css" id="themeStyle"><link rel="stylesheet" type="text/css" href="../../../dark.css" disabled ><link rel="stylesheet" type="text/css" href="../../../ayu.css" disabled ><script id="default-settings"></script><script src="../../../storage.js"></script><noscript><link rel="stylesheet" href="../../../noscript.css"></noscript><link rel="icon" type="image/svg+xml" href="../../../favicon.svg">
2 <link rel="alternate icon" type="image/png" href="../../../favicon-16x16.png">
3 <link rel="alternate icon" type="image/png" href="../../../favicon-32x32.png"><style type="text/css">#crate-search{background-image:url("../../../down-arrow.svg");}</style></head><body class="rustdoc source"><!--[if lte IE 8]><div class="warning">This old browser is unsupported and will most likely display funky things.</div><![endif]--><nav class="sidebar"><div class="sidebar-menu">&#9776;</div><a href='../../../bdk/index.html'><div class='logo-container rust-logo'><img src='../../../rust-logo.png' alt='logo'></div></a></nav><div class="theme-picker"><button id="theme-picker" aria-label="Pick another theme!" aria-haspopup="menu"><img src="../../../brush.svg" width="18" alt="Pick another theme!"></button><div id="theme-choices" role="menu"></div></div><script src="../../../theme.js"></script><nav class="sub"><form class="search-form"><div class="search-container"><div><select id="crate-search"><option value="All crates">All crates</option></select><input class="search-input" name="search" disabled autocomplete="off" spellcheck="false" placeholder="Click or press ‘S’ to search, ‘?’ for more options…" type="search"></div><button type="button" class="help-button">?</button>
4 <a id="settings-menu" href="../../../settings.html"><img src="../../../wheel.svg" width="18" alt="Change settings"></a></div></form></nav><section id="main" class="content"><pre class="line-numbers"><span id="1"> 1</span>
5 <span id="2"> 2</span>
6 <span id="3"> 3</span>
7 <span id="4"> 4</span>
8 <span id="5"> 5</span>
9 <span id="6"> 6</span>
10 <span id="7"> 7</span>
11 <span id="8"> 8</span>
12 <span id="9"> 9</span>
13 <span id="10"> 10</span>
14 <span id="11"> 11</span>
15 <span id="12"> 12</span>
16 <span id="13"> 13</span>
17 <span id="14"> 14</span>
18 <span id="15"> 15</span>
19 <span id="16"> 16</span>
20 <span id="17"> 17</span>
21 <span id="18"> 18</span>
22 <span id="19"> 19</span>
23 <span id="20"> 20</span>
24 <span id="21"> 21</span>
25 <span id="22"> 22</span>
26 <span id="23"> 23</span>
27 <span id="24"> 24</span>
28 <span id="25"> 25</span>
29 <span id="26"> 26</span>
30 <span id="27"> 27</span>
31 <span id="28"> 28</span>
32 <span id="29"> 29</span>
33 <span id="30"> 30</span>
34 <span id="31"> 31</span>
35 <span id="32"> 32</span>
36 <span id="33"> 33</span>
37 <span id="34"> 34</span>
38 <span id="35"> 35</span>
39 <span id="36"> 36</span>
40 <span id="37"> 37</span>
41 <span id="38"> 38</span>
42 <span id="39"> 39</span>
43 <span id="40"> 40</span>
44 <span id="41"> 41</span>
45 <span id="42"> 42</span>
46 <span id="43"> 43</span>
47 <span id="44"> 44</span>
48 <span id="45"> 45</span>
49 <span id="46"> 46</span>
50 <span id="47"> 47</span>
51 <span id="48"> 48</span>
52 <span id="49"> 49</span>
53 <span id="50"> 50</span>
54 <span id="51"> 51</span>
55 <span id="52"> 52</span>
56 <span id="53"> 53</span>
57 <span id="54"> 54</span>
58 <span id="55"> 55</span>
59 <span id="56"> 56</span>
60 <span id="57"> 57</span>
61 <span id="58"> 58</span>
62 <span id="59"> 59</span>
63 <span id="60"> 60</span>
64 <span id="61"> 61</span>
65 <span id="62"> 62</span>
66 <span id="63"> 63</span>
67 <span id="64"> 64</span>
68 <span id="65"> 65</span>
69 <span id="66"> 66</span>
70 <span id="67"> 67</span>
71 <span id="68"> 68</span>
72 <span id="69"> 69</span>
73 <span id="70"> 70</span>
74 <span id="71"> 71</span>
75 <span id="72"> 72</span>
76 <span id="73"> 73</span>
77 <span id="74"> 74</span>
78 <span id="75"> 75</span>
79 <span id="76"> 76</span>
80 <span id="77"> 77</span>
81 <span id="78"> 78</span>
82 <span id="79"> 79</span>
83 <span id="80"> 80</span>
84 <span id="81"> 81</span>
85 <span id="82"> 82</span>
86 <span id="83"> 83</span>
87 <span id="84"> 84</span>
88 <span id="85"> 85</span>
89 <span id="86"> 86</span>
90 <span id="87"> 87</span>
91 <span id="88"> 88</span>
92 <span id="89"> 89</span>
93 <span id="90"> 90</span>
94 <span id="91"> 91</span>
95 <span id="92"> 92</span>
96 <span id="93"> 93</span>
97 <span id="94"> 94</span>
98 <span id="95"> 95</span>
99 <span id="96"> 96</span>
100 <span id="97"> 97</span>
101 <span id="98"> 98</span>
102 <span id="99"> 99</span>
103 <span id="100">100</span>
104 <span id="101">101</span>
105 <span id="102">102</span>
106 <span id="103">103</span>
107 <span id="104">104</span>
108 <span id="105">105</span>
109 <span id="106">106</span>
110 <span id="107">107</span>
111 <span id="108">108</span>
112 <span id="109">109</span>
113 <span id="110">110</span>
114 <span id="111">111</span>
115 <span id="112">112</span>
116 <span id="113">113</span>
117 <span id="114">114</span>
118 <span id="115">115</span>
119 <span id="116">116</span>
120 <span id="117">117</span>
121 <span id="118">118</span>
122 <span id="119">119</span>
123 <span id="120">120</span>
124 <span id="121">121</span>
125 <span id="122">122</span>
126 <span id="123">123</span>
127 <span id="124">124</span>
128 <span id="125">125</span>
129 <span id="126">126</span>
130 <span id="127">127</span>
131 <span id="128">128</span>
132 <span id="129">129</span>
133 <span id="130">130</span>
134 <span id="131">131</span>
135 <span id="132">132</span>
136 <span id="133">133</span>
137 <span id="134">134</span>
138 <span id="135">135</span>
139 <span id="136">136</span>
140 <span id="137">137</span>
141 <span id="138">138</span>
142 <span id="139">139</span>
143 <span id="140">140</span>
144 <span id="141">141</span>
145 <span id="142">142</span>
146 <span id="143">143</span>
147 <span id="144">144</span>
148 <span id="145">145</span>
149 <span id="146">146</span>
150 <span id="147">147</span>
151 <span id="148">148</span>
152 <span id="149">149</span>
153 <span id="150">150</span>
154 <span id="151">151</span>
155 <span id="152">152</span>
156 <span id="153">153</span>
157 <span id="154">154</span>
158 <span id="155">155</span>
159 <span id="156">156</span>
160 <span id="157">157</span>
161 <span id="158">158</span>
162 <span id="159">159</span>
163 <span id="160">160</span>
164 <span id="161">161</span>
165 <span id="162">162</span>
166 <span id="163">163</span>
167 <span id="164">164</span>
168 <span id="165">165</span>
169 <span id="166">166</span>
170 <span id="167">167</span>
171 <span id="168">168</span>
172 <span id="169">169</span>
173 <span id="170">170</span>
174 <span id="171">171</span>
175 <span id="172">172</span>
176 <span id="173">173</span>
177 <span id="174">174</span>
178 <span id="175">175</span>
179 <span id="176">176</span>
180 <span id="177">177</span>
181 <span id="178">178</span>
182 <span id="179">179</span>
183 <span id="180">180</span>
184 <span id="181">181</span>
185 <span id="182">182</span>
186 <span id="183">183</span>
187 <span id="184">184</span>
188 <span id="185">185</span>
189 <span id="186">186</span>
190 <span id="187">187</span>
191 <span id="188">188</span>
192 <span id="189">189</span>
193 <span id="190">190</span>
194 <span id="191">191</span>
195 <span id="192">192</span>
196 <span id="193">193</span>
197 <span id="194">194</span>
198 <span id="195">195</span>
199 <span id="196">196</span>
200 <span id="197">197</span>
201 <span id="198">198</span>
202 <span id="199">199</span>
203 <span id="200">200</span>
204 <span id="201">201</span>
205 <span id="202">202</span>
206 <span id="203">203</span>
207 <span id="204">204</span>
208 <span id="205">205</span>
209 <span id="206">206</span>
210 <span id="207">207</span>
211 <span id="208">208</span>
212 <span id="209">209</span>
213 <span id="210">210</span>
214 <span id="211">211</span>
215 <span id="212">212</span>
216 <span id="213">213</span>
217 <span id="214">214</span>
218 <span id="215">215</span>
219 <span id="216">216</span>
220 <span id="217">217</span>
221 <span id="218">218</span>
222 <span id="219">219</span>
223 <span id="220">220</span>
224 <span id="221">221</span>
225 <span id="222">222</span>
226 <span id="223">223</span>
227 <span id="224">224</span>
228 <span id="225">225</span>
229 <span id="226">226</span>
230 <span id="227">227</span>
231 <span id="228">228</span>
232 <span id="229">229</span>
233 <span id="230">230</span>
234 <span id="231">231</span>
235 <span id="232">232</span>
236 <span id="233">233</span>
237 <span id="234">234</span>
238 <span id="235">235</span>
239 <span id="236">236</span>
240 <span id="237">237</span>
241 <span id="238">238</span>
242 <span id="239">239</span>
243 <span id="240">240</span>
244 <span id="241">241</span>
245 <span id="242">242</span>
246 <span id="243">243</span>
247 <span id="244">244</span>
248 <span id="245">245</span>
249 <span id="246">246</span>
250 <span id="247">247</span>
251 <span id="248">248</span>
252 <span id="249">249</span>
253 <span id="250">250</span>
254 <span id="251">251</span>
255 <span id="252">252</span>
256 <span id="253">253</span>
257 <span id="254">254</span>
258 <span id="255">255</span>
259 <span id="256">256</span>
260 <span id="257">257</span>
261 <span id="258">258</span>
262 <span id="259">259</span>
263 <span id="260">260</span>
264 <span id="261">261</span>
265 <span id="262">262</span>
266 <span id="263">263</span>
267 <span id="264">264</span>
268 <span id="265">265</span>
269 <span id="266">266</span>
270 <span id="267">267</span>
271 <span id="268">268</span>
272 <span id="269">269</span>
273 <span id="270">270</span>
274 <span id="271">271</span>
275 <span id="272">272</span>
276 <span id="273">273</span>
277 <span id="274">274</span>
278 <span id="275">275</span>
279 <span id="276">276</span>
280 <span id="277">277</span>
281 <span id="278">278</span>
282 <span id="279">279</span>
283 <span id="280">280</span>
284 <span id="281">281</span>
285 <span id="282">282</span>
286 <span id="283">283</span>
287 <span id="284">284</span>
288 <span id="285">285</span>
289 <span id="286">286</span>
290 <span id="287">287</span>
291 <span id="288">288</span>
292 <span id="289">289</span>
293 <span id="290">290</span>
294 <span id="291">291</span>
295 <span id="292">292</span>
296 <span id="293">293</span>
297 <span id="294">294</span>
298 <span id="295">295</span>
299 <span id="296">296</span>
300 <span id="297">297</span>
301 <span id="298">298</span>
302 <span id="299">299</span>
303 <span id="300">300</span>
304 <span id="301">301</span>
305 <span id="302">302</span>
306 <span id="303">303</span>
307 <span id="304">304</span>
308 <span id="305">305</span>
309 <span id="306">306</span>
310 <span id="307">307</span>
311 <span id="308">308</span>
312 <span id="309">309</span>
313 <span id="310">310</span>
314 <span id="311">311</span>
315 <span id="312">312</span>
316 <span id="313">313</span>
317 <span id="314">314</span>
318 <span id="315">315</span>
319 <span id="316">316</span>
320 <span id="317">317</span>
321 <span id="318">318</span>
322 <span id="319">319</span>
323 <span id="320">320</span>
324 <span id="321">321</span>
325 <span id="322">322</span>
326 <span id="323">323</span>
327 <span id="324">324</span>
328 <span id="325">325</span>
329 <span id="326">326</span>
330 <span id="327">327</span>
331 <span id="328">328</span>
332 <span id="329">329</span>
333 <span id="330">330</span>
334 <span id="331">331</span>
335 <span id="332">332</span>
336 <span id="333">333</span>
337 <span id="334">334</span>
338 <span id="335">335</span>
339 <span id="336">336</span>
340 <span id="337">337</span>
341 <span id="338">338</span>
342 <span id="339">339</span>
343 <span id="340">340</span>
344 <span id="341">341</span>
345 <span id="342">342</span>
346 <span id="343">343</span>
347 <span id="344">344</span>
348 <span id="345">345</span>
349 <span id="346">346</span>
350 <span id="347">347</span>
351 <span id="348">348</span>
352 <span id="349">349</span>
353 <span id="350">350</span>
354 <span id="351">351</span>
355 <span id="352">352</span>
356 <span id="353">353</span>
357 <span id="354">354</span>
358 <span id="355">355</span>
359 <span id="356">356</span>
360 <span id="357">357</span>
361 <span id="358">358</span>
362 <span id="359">359</span>
363 <span id="360">360</span>
364 <span id="361">361</span>
365 <span id="362">362</span>
366 <span id="363">363</span>
367 <span id="364">364</span>
368 <span id="365">365</span>
369 <span id="366">366</span>
370 <span id="367">367</span>
371 <span id="368">368</span>
372 <span id="369">369</span>
373 <span id="370">370</span>
374 <span id="371">371</span>
375 <span id="372">372</span>
376 <span id="373">373</span>
377 <span id="374">374</span>
378 <span id="375">375</span>
379 <span id="376">376</span>
380 <span id="377">377</span>
381 <span id="378">378</span>
382 <span id="379">379</span>
383 <span id="380">380</span>
384 <span id="381">381</span>
385 <span id="382">382</span>
386 <span id="383">383</span>
387 <span id="384">384</span>
388 <span id="385">385</span>
389 <span id="386">386</span>
390 <span id="387">387</span>
391 <span id="388">388</span>
392 <span id="389">389</span>
393 <span id="390">390</span>
394 <span id="391">391</span>
395 <span id="392">392</span>
396 <span id="393">393</span>
397 <span id="394">394</span>
398 <span id="395">395</span>
399 <span id="396">396</span>
400 <span id="397">397</span>
401 <span id="398">398</span>
402 <span id="399">399</span>
403 <span id="400">400</span>
404 <span id="401">401</span>
405 <span id="402">402</span>
406 <span id="403">403</span>
407 <span id="404">404</span>
408 <span id="405">405</span>
409 <span id="406">406</span>
410 <span id="407">407</span>
411 <span id="408">408</span>
412 <span id="409">409</span>
413 <span id="410">410</span>
414 <span id="411">411</span>
415 <span id="412">412</span>
416 <span id="413">413</span>
417 <span id="414">414</span>
418 <span id="415">415</span>
419 <span id="416">416</span>
420 <span id="417">417</span>
421 <span id="418">418</span>
422 <span id="419">419</span>
423 <span id="420">420</span>
424 <span id="421">421</span>
425 <span id="422">422</span>
426 <span id="423">423</span>
427 <span id="424">424</span>
428 <span id="425">425</span>
429 <span id="426">426</span>
430 <span id="427">427</span>
431 <span id="428">428</span>
432 <span id="429">429</span>
433 <span id="430">430</span>
434 <span id="431">431</span>
435 <span id="432">432</span>
436 <span id="433">433</span>
437 <span id="434">434</span>
438 <span id="435">435</span>
439 <span id="436">436</span>
440 <span id="437">437</span>
441 <span id="438">438</span>
442 <span id="439">439</span>
443 <span id="440">440</span>
444 <span id="441">441</span>
445 <span id="442">442</span>
446 <span id="443">443</span>
447 <span id="444">444</span>
448 <span id="445">445</span>
449 <span id="446">446</span>
450 <span id="447">447</span>
451 <span id="448">448</span>
452 <span id="449">449</span>
453 <span id="450">450</span>
454 <span id="451">451</span>
455 <span id="452">452</span>
456 <span id="453">453</span>
457 <span id="454">454</span>
458 <span id="455">455</span>
459 <span id="456">456</span>
460 <span id="457">457</span>
461 <span id="458">458</span>
462 <span id="459">459</span>
463 <span id="460">460</span>
464 <span id="461">461</span>
465 <span id="462">462</span>
466 <span id="463">463</span>
467 <span id="464">464</span>
468 <span id="465">465</span>
469 <span id="466">466</span>
470 <span id="467">467</span>
471 <span id="468">468</span>
472 <span id="469">469</span>
473 <span id="470">470</span>
474 <span id="471">471</span>
475 <span id="472">472</span>
476 <span id="473">473</span>
477 <span id="474">474</span>
478 <span id="475">475</span>
479 <span id="476">476</span>
480 <span id="477">477</span>
481 <span id="478">478</span>
482 <span id="479">479</span>
483 <span id="480">480</span>
484 <span id="481">481</span>
485 <span id="482">482</span>
486 <span id="483">483</span>
487 <span id="484">484</span>
488 <span id="485">485</span>
489 <span id="486">486</span>
490 <span id="487">487</span>
491 <span id="488">488</span>
492 <span id="489">489</span>
493 <span id="490">490</span>
494 <span id="491">491</span>
495 <span id="492">492</span>
496 <span id="493">493</span>
497 <span id="494">494</span>
498 <span id="495">495</span>
499 <span id="496">496</span>
500 <span id="497">497</span>
501 <span id="498">498</span>
502 <span id="499">499</span>
503 <span id="500">500</span>
504 <span id="501">501</span>
505 <span id="502">502</span>
506 <span id="503">503</span>
507 <span id="504">504</span>
508 <span id="505">505</span>
509 <span id="506">506</span>
510 <span id="507">507</span>
511 <span id="508">508</span>
512 <span id="509">509</span>
513 <span id="510">510</span>
514 <span id="511">511</span>
515 <span id="512">512</span>
516 <span id="513">513</span>
517 <span id="514">514</span>
518 <span id="515">515</span>
519 <span id="516">516</span>
520 <span id="517">517</span>
521 <span id="518">518</span>
522 <span id="519">519</span>
523 <span id="520">520</span>
524 <span id="521">521</span>
525 <span id="522">522</span>
526 <span id="523">523</span>
527 <span id="524">524</span>
528 <span id="525">525</span>
529 <span id="526">526</span>
530 <span id="527">527</span>
531 <span id="528">528</span>
532 <span id="529">529</span>
533 <span id="530">530</span>
534 <span id="531">531</span>
535 <span id="532">532</span>
536 <span id="533">533</span>
537 <span id="534">534</span>
538 <span id="535">535</span>
539 <span id="536">536</span>
540 <span id="537">537</span>
541 <span id="538">538</span>
542 <span id="539">539</span>
543 <span id="540">540</span>
544 <span id="541">541</span>
545 <span id="542">542</span>
546 <span id="543">543</span>
547 <span id="544">544</span>
548 <span id="545">545</span>
549 <span id="546">546</span>
550 <span id="547">547</span>
551 <span id="548">548</span>
552 <span id="549">549</span>
553 <span id="550">550</span>
554 <span id="551">551</span>
555 <span id="552">552</span>
556 <span id="553">553</span>
557 <span id="554">554</span>
558 <span id="555">555</span>
559 <span id="556">556</span>
560 <span id="557">557</span>
561 <span id="558">558</span>
562 <span id="559">559</span>
563 <span id="560">560</span>
564 <span id="561">561</span>
565 <span id="562">562</span>
566 <span id="563">563</span>
567 <span id="564">564</span>
568 <span id="565">565</span>
569 <span id="566">566</span>
570 <span id="567">567</span>
571 <span id="568">568</span>
572 <span id="569">569</span>
573 <span id="570">570</span>
574 <span id="571">571</span>
575 <span id="572">572</span>
576 <span id="573">573</span>
577 <span id="574">574</span>
578 <span id="575">575</span>
579 <span id="576">576</span>
580 <span id="577">577</span>
581 <span id="578">578</span>
582 <span id="579">579</span>
583 <span id="580">580</span>
584 <span id="581">581</span>
585 <span id="582">582</span>
586 <span id="583">583</span>
587 <span id="584">584</span>
588 <span id="585">585</span>
589 <span id="586">586</span>
590 <span id="587">587</span>
591 <span id="588">588</span>
592 <span id="589">589</span>
593 <span id="590">590</span>
594 <span id="591">591</span>
595 <span id="592">592</span>
596 <span id="593">593</span>
597 <span id="594">594</span>
598 <span id="595">595</span>
599 <span id="596">596</span>
600 <span id="597">597</span>
601 <span id="598">598</span>
602 <span id="599">599</span>
603 <span id="600">600</span>
604 <span id="601">601</span>
605 <span id="602">602</span>
606 <span id="603">603</span>
607 <span id="604">604</span>
608 <span id="605">605</span>
609 <span id="606">606</span>
610 <span id="607">607</span>
611 <span id="608">608</span>
612 <span id="609">609</span>
613 <span id="610">610</span>
614 <span id="611">611</span>
615 <span id="612">612</span>
616 <span id="613">613</span>
617 <span id="614">614</span>
618 <span id="615">615</span>
619 <span id="616">616</span>
620 <span id="617">617</span>
621 <span id="618">618</span>
622 <span id="619">619</span>
623 <span id="620">620</span>
624 <span id="621">621</span>
625 <span id="622">622</span>
626 <span id="623">623</span>
627 <span id="624">624</span>
628 <span id="625">625</span>
629 <span id="626">626</span>
630 <span id="627">627</span>
631 <span id="628">628</span>
632 <span id="629">629</span>
633 <span id="630">630</span>
634 <span id="631">631</span>
635 <span id="632">632</span>
636 <span id="633">633</span>
637 <span id="634">634</span>
638 <span id="635">635</span>
639 <span id="636">636</span>
640 <span id="637">637</span>
641 <span id="638">638</span>
642 <span id="639">639</span>
643 <span id="640">640</span>
644 <span id="641">641</span>
645 <span id="642">642</span>
646 <span id="643">643</span>
647 <span id="644">644</span>
648 <span id="645">645</span>
649 <span id="646">646</span>
650 <span id="647">647</span>
651 <span id="648">648</span>
652 <span id="649">649</span>
653 <span id="650">650</span>
654 <span id="651">651</span>
655 <span id="652">652</span>
656 <span id="653">653</span>
657 <span id="654">654</span>
658 <span id="655">655</span>
659 <span id="656">656</span>
660 <span id="657">657</span>
661 <span id="658">658</span>
662 <span id="659">659</span>
663 <span id="660">660</span>
664 <span id="661">661</span>
665 <span id="662">662</span>
666 <span id="663">663</span>
667 <span id="664">664</span>
668 <span id="665">665</span>
669 <span id="666">666</span>
670 <span id="667">667</span>
671 <span id="668">668</span>
672 <span id="669">669</span>
673 <span id="670">670</span>
674 <span id="671">671</span>
675 <span id="672">672</span>
676 <span id="673">673</span>
677 <span id="674">674</span>
678 <span id="675">675</span>
679 <span id="676">676</span>
680 <span id="677">677</span>
681 <span id="678">678</span>
682 <span id="679">679</span>
683 <span id="680">680</span>
684 <span id="681">681</span>
685 <span id="682">682</span>
686 <span id="683">683</span>
687 <span id="684">684</span>
688 <span id="685">685</span>
689 <span id="686">686</span>
690 <span id="687">687</span>
691 <span id="688">688</span>
692 <span id="689">689</span>
693 <span id="690">690</span>
694 <span id="691">691</span>
695 <span id="692">692</span>
696 <span id="693">693</span>
697 <span id="694">694</span>
698 <span id="695">695</span>
699 <span id="696">696</span>
700 <span id="697">697</span>
701 <span id="698">698</span>
702 <span id="699">699</span>
703 <span id="700">700</span>
704 <span id="701">701</span>
705 <span id="702">702</span>
706 <span id="703">703</span>
707 <span id="704">704</span>
708 <span id="705">705</span>
709 <span id="706">706</span>
710 <span id="707">707</span>
711 <span id="708">708</span>
712 <span id="709">709</span>
713 <span id="710">710</span>
714 <span id="711">711</span>
715 <span id="712">712</span>
716 <span id="713">713</span>
717 <span id="714">714</span>
718 <span id="715">715</span>
719 <span id="716">716</span>
720 <span id="717">717</span>
721 <span id="718">718</span>
722 <span id="719">719</span>
723 <span id="720">720</span>
724 <span id="721">721</span>
725 <span id="722">722</span>
726 <span id="723">723</span>
727 <span id="724">724</span>
728 <span id="725">725</span>
729 <span id="726">726</span>
730 <span id="727">727</span>
731 <span id="728">728</span>
732 <span id="729">729</span>
733 <span id="730">730</span>
734 <span id="731">731</span>
735 <span id="732">732</span>
736 <span id="733">733</span>
737 <span id="734">734</span>
738 <span id="735">735</span>
739 <span id="736">736</span>
740 <span id="737">737</span>
741 <span id="738">738</span>
742 <span id="739">739</span>
743 <span id="740">740</span>
744 <span id="741">741</span>
745 <span id="742">742</span>
746 <span id="743">743</span>
747 <span id="744">744</span>
748 <span id="745">745</span>
749 <span id="746">746</span>
750 <span id="747">747</span>
751 <span id="748">748</span>
752 <span id="749">749</span>
753 <span id="750">750</span>
754 <span id="751">751</span>
755 <span id="752">752</span>
756 <span id="753">753</span>
757 <span id="754">754</span>
758 <span id="755">755</span>
759 <span id="756">756</span>
760 <span id="757">757</span>
761 <span id="758">758</span>
762 <span id="759">759</span>
763 <span id="760">760</span>
764 <span id="761">761</span>
765 <span id="762">762</span>
766 <span id="763">763</span>
767 <span id="764">764</span>
768 <span id="765">765</span>
769 <span id="766">766</span>
770 <span id="767">767</span>
771 <span id="768">768</span>
772 <span id="769">769</span>
773 <span id="770">770</span>
774 <span id="771">771</span>
775 <span id="772">772</span>
776 <span id="773">773</span>
777 <span id="774">774</span>
778 <span id="775">775</span>
779 <span id="776">776</span>
780 <span id="777">777</span>
781 <span id="778">778</span>
782 <span id="779">779</span>
783 <span id="780">780</span>
784 <span id="781">781</span>
785 <span id="782">782</span>
786 <span id="783">783</span>
787 <span id="784">784</span>
788 <span id="785">785</span>
789 <span id="786">786</span>
790 <span id="787">787</span>
791 <span id="788">788</span>
792 <span id="789">789</span>
793 <span id="790">790</span>
794 <span id="791">791</span>
795 <span id="792">792</span>
796 <span id="793">793</span>
797 <span id="794">794</span>
798 <span id="795">795</span>
799 <span id="796">796</span>
800 <span id="797">797</span>
801 <span id="798">798</span>
802 <span id="799">799</span>
803 <span id="800">800</span>
804 <span id="801">801</span>
805 <span id="802">802</span>
806 <span id="803">803</span>
807 <span id="804">804</span>
808 <span id="805">805</span>
809 <span id="806">806</span>
810 <span id="807">807</span>
811 <span id="808">808</span>
812 <span id="809">809</span>
813 <span id="810">810</span>
814 <span id="811">811</span>
815 <span id="812">812</span>
816 <span id="813">813</span>
817 <span id="814">814</span>
818 <span id="815">815</span>
819 <span id="816">816</span>
820 <span id="817">817</span>
821 <span id="818">818</span>
822 <span id="819">819</span>
823 <span id="820">820</span>
824 <span id="821">821</span>
825 <span id="822">822</span>
826 <span id="823">823</span>
827 <span id="824">824</span>
828 <span id="825">825</span>
829 <span id="826">826</span>
830 <span id="827">827</span>
831 <span id="828">828</span>
832 <span id="829">829</span>
833 <span id="830">830</span>
834 <span id="831">831</span>
835 <span id="832">832</span>
836 <span id="833">833</span>
837 <span id="834">834</span>
838 <span id="835">835</span>
839 <span id="836">836</span>
840 <span id="837">837</span>
841 <span id="838">838</span>
842 <span id="839">839</span>
843 <span id="840">840</span>
844 <span id="841">841</span>
845 <span id="842">842</span>
846 <span id="843">843</span>
847 <span id="844">844</span>
848 <span id="845">845</span>
849 <span id="846">846</span>
850 <span id="847">847</span>
851 <span id="848">848</span>
852 <span id="849">849</span>
853 <span id="850">850</span>
854 <span id="851">851</span>
855 <span id="852">852</span>
856 <span id="853">853</span>
857 <span id="854">854</span>
858 <span id="855">855</span>
859 <span id="856">856</span>
860 <span id="857">857</span>
861 <span id="858">858</span>
862 <span id="859">859</span>
863 <span id="860">860</span>
864 <span id="861">861</span>
865 <span id="862">862</span>
866 <span id="863">863</span>
867 <span id="864">864</span>
868 <span id="865">865</span>
869 <span id="866">866</span>
870 <span id="867">867</span>
871 <span id="868">868</span>
872 <span id="869">869</span>
873 <span id="870">870</span>
874 <span id="871">871</span>
875 <span id="872">872</span>
876 <span id="873">873</span>
877 <span id="874">874</span>
878 <span id="875">875</span>
879 <span id="876">876</span>
880 <span id="877">877</span>
881 <span id="878">878</span>
882 <span id="879">879</span>
883 <span id="880">880</span>
884 <span id="881">881</span>
885 <span id="882">882</span>
886 <span id="883">883</span>
887 <span id="884">884</span>
888 <span id="885">885</span>
889 <span id="886">886</span>
890 <span id="887">887</span>
891 <span id="888">888</span>
892 <span id="889">889</span>
893 <span id="890">890</span>
894 <span id="891">891</span>
895 <span id="892">892</span>
896 <span id="893">893</span>
897 <span id="894">894</span>
898 <span id="895">895</span>
899 <span id="896">896</span>
900 <span id="897">897</span>
901 <span id="898">898</span>
902 <span id="899">899</span>
903 <span id="900">900</span>
904 <span id="901">901</span>
905 <span id="902">902</span>
906 <span id="903">903</span>
907 <span id="904">904</span>
908 <span id="905">905</span>
909 <span id="906">906</span>
910 <span id="907">907</span>
911 <span id="908">908</span>
912 <span id="909">909</span>
913 <span id="910">910</span>
914 <span id="911">911</span>
915 <span id="912">912</span>
916 <span id="913">913</span>
917 <span id="914">914</span>
918 <span id="915">915</span>
919 <span id="916">916</span>
920 <span id="917">917</span>
921 <span id="918">918</span>
922 <span id="919">919</span>
923 <span id="920">920</span>
924 <span id="921">921</span>
925 <span id="922">922</span>
926 <span id="923">923</span>
927 <span id="924">924</span>
928 <span id="925">925</span>
929 <span id="926">926</span>
930 <span id="927">927</span>
931 <span id="928">928</span>
932 <span id="929">929</span>
933 <span id="930">930</span>
934 <span id="931">931</span>
935 <span id="932">932</span>
936 <span id="933">933</span>
937 <span id="934">934</span>
938 <span id="935">935</span>
939 <span id="936">936</span>
940 <span id="937">937</span>
941 <span id="938">938</span>
942 <span id="939">939</span>
943 <span id="940">940</span>
944 <span id="941">941</span>
945 <span id="942">942</span>
946 <span id="943">943</span>
947 <span id="944">944</span>
948 <span id="945">945</span>
949 <span id="946">946</span>
950 <span id="947">947</span>
951 <span id="948">948</span>
952 <span id="949">949</span>
953 <span id="950">950</span>
954 <span id="951">951</span>
955 <span id="952">952</span>
956 <span id="953">953</span>
957 <span id="954">954</span>
958 <span id="955">955</span>
959 <span id="956">956</span>
960 <span id="957">957</span>
961 <span id="958">958</span>
962 <span id="959">959</span>
963 <span id="960">960</span>
964 <span id="961">961</span>
965 <span id="962">962</span>
966 <span id="963">963</span>
967 <span id="964">964</span>
968 <span id="965">965</span>
969 <span id="966">966</span>
970 <span id="967">967</span>
971 <span id="968">968</span>
972 <span id="969">969</span>
973 <span id="970">970</span>
974 <span id="971">971</span>
975 <span id="972">972</span>
976 <span id="973">973</span>
977 <span id="974">974</span>
978 <span id="975">975</span>
979 <span id="976">976</span>
980 <span id="977">977</span>
981 <span id="978">978</span>
982 <span id="979">979</span>
983 <span id="980">980</span>
984 <span id="981">981</span>
985 <span id="982">982</span>
986 <span id="983">983</span>
987 <span id="984">984</span>
988 <span id="985">985</span>
989 <span id="986">986</span>
990 <span id="987">987</span>
991 <span id="988">988</span>
992 <span id="989">989</span>
993 <span id="990">990</span>
994 <span id="991">991</span>
995 <span id="992">992</span>
996 <span id="993">993</span>
997 <span id="994">994</span>
998 <span id="995">995</span>
999 <span id="996">996</span>
1000 <span id="997">997</span>
1001 </pre><div class="example-wrap"><pre class="rust ">
1002 <span class="comment">// Magical Bitcoin Library</span>
1003 <span class="comment">// Written in 2020 by</span>
1004 <span class="comment">// Alekos Filini &lt;alekos.filini@gmail.com&gt;</span>
1005 <span class="comment">//</span>
1006 <span class="comment">// Copyright (c) 2020 Magical Bitcoin</span>
1007 <span class="comment">//</span>
1008 <span class="comment">// Permission is hereby granted, free of charge, to any person obtaining a copy</span>
1009 <span class="comment">// of this software and associated documentation files (the &quot;Software&quot;), to deal</span>
1010 <span class="comment">// in the Software without restriction, including without limitation the rights</span>
1011 <span class="comment">// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell</span>
1012 <span class="comment">// copies of the Software, and to permit persons to whom the Software is</span>
1013 <span class="comment">// furnished to do so, subject to the following conditions:</span>
1014 <span class="comment">//</span>
1015 <span class="comment">// The above copyright notice and this permission notice shall be included in all</span>
1016 <span class="comment">// copies or substantial portions of the Software.</span>
1017 <span class="comment">//</span>
1018 <span class="comment">// THE SOFTWARE IS PROVIDED &quot;AS IS&quot;, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR</span>
1019 <span class="comment">// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,</span>
1020 <span class="comment">// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE</span>
1021 <span class="comment">// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER</span>
1022 <span class="comment">// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,</span>
1023 <span class="comment">// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE</span>
1024 <span class="comment">// SOFTWARE.</span>
1025
1026 <span class="doccomment">//! Coin selection</span>
1027 <span class="doccomment">//!</span>
1028 <span class="doccomment">//! This module provides the trait [`CoinSelectionAlgorithm`] that can be implemented to</span>
1029 <span class="doccomment">//! define custom coin selection algorithms.</span>
1030 <span class="doccomment">//!</span>
1031 <span class="doccomment">//! The coin selection algorithm is not globally part of a [`Wallet`](super::Wallet), instead it</span>
1032 <span class="doccomment">//! is selected whenever a [`Wallet::create_tx`](super::Wallet::create_tx) call is made, through</span>
1033 <span class="doccomment">//! the use of the [`TxBuilder`] structure, specifically with</span>
1034 <span class="doccomment">//! [`TxBuilder::coin_selection`](super::tx_builder::TxBuilder::coin_selection) method.</span>
1035 <span class="doccomment">//!</span>
1036 <span class="doccomment">//! The [`DefaultCoinSelectionAlgorithm`] selects the default coin selection algorithm that</span>
1037 <span class="doccomment">//! [`TxBuilder`] uses, if it&#39;s not explicitly overridden.</span>
1038 <span class="doccomment">//!</span>
1039 <span class="doccomment">//! [`TxBuilder`]: super::tx_builder::TxBuilder</span>
1040 <span class="doccomment">//!</span>
1041 <span class="doccomment">//! ## Example</span>
1042 <span class="doccomment">//!</span>
1043 <span class="doccomment">//! ```no_run</span>
1044 <span class="doccomment">//! # use std::str::FromStr;</span>
1045 <span class="doccomment">//! # use bitcoin::*;</span>
1046 <span class="doccomment">//! # use bdk::wallet::coin_selection::*;</span>
1047 <span class="doccomment">//! # use bdk::database::Database;</span>
1048 <span class="doccomment">//! # use bdk::*;</span>
1049 <span class="doccomment">//! # const TXIN_BASE_WEIGHT: usize = (32 + 4 + 4 + 1) * 4;</span>
1050 <span class="doccomment">//! #[derive(Debug)]</span>
1051 <span class="doccomment">//! struct AlwaysSpendEverything;</span>
1052 <span class="doccomment">//!</span>
1053 <span class="doccomment">//! impl&lt;D: Database&gt; CoinSelectionAlgorithm&lt;D&gt; for AlwaysSpendEverything {</span>
1054 <span class="doccomment">//! fn coin_select(</span>
1055 <span class="doccomment">//! &amp;self,</span>
1056 <span class="doccomment">//! database: &amp;D,</span>
1057 <span class="doccomment">//! required_utxos: Vec&lt;(UTXO, usize)&gt;,</span>
1058 <span class="doccomment">//! optional_utxos: Vec&lt;(UTXO, usize)&gt;,</span>
1059 <span class="doccomment">//! fee_rate: FeeRate,</span>
1060 <span class="doccomment">//! amount_needed: u64,</span>
1061 <span class="doccomment">//! fee_amount: f32,</span>
1062 <span class="doccomment">//! ) -&gt; Result&lt;CoinSelectionResult, bdk::Error&gt; {</span>
1063 <span class="doccomment">//! let mut selected_amount = 0;</span>
1064 <span class="doccomment">//! let mut additional_weight = 0;</span>
1065 <span class="doccomment">//! let all_utxos_selected = required_utxos</span>
1066 <span class="doccomment">//! .into_iter().chain(optional_utxos)</span>
1067 <span class="doccomment">//! .scan((&amp;mut selected_amount, &amp;mut additional_weight), |(selected_amount, additional_weight), (utxo, weight)| {</span>
1068 <span class="doccomment">//! **selected_amount += utxo.txout.value;</span>
1069 <span class="doccomment">//! **additional_weight += TXIN_BASE_WEIGHT + weight;</span>
1070 <span class="doccomment">//!</span>
1071 <span class="doccomment">//! Some(utxo)</span>
1072 <span class="doccomment">//! })</span>
1073 <span class="doccomment">//! .collect::&lt;Vec&lt;_&gt;&gt;();</span>
1074 <span class="doccomment">//! let additional_fees = additional_weight as f32 * fee_rate.as_sat_vb() / 4.0;</span>
1075 <span class="doccomment">//!</span>
1076 <span class="doccomment">//! if (fee_amount + additional_fees).ceil() as u64 + amount_needed &gt; selected_amount {</span>
1077 <span class="doccomment">//! return Err(bdk::Error::InsufficientFunds);</span>
1078 <span class="doccomment">//! }</span>
1079 <span class="doccomment">//!</span>
1080 <span class="doccomment">//! Ok(CoinSelectionResult {</span>
1081 <span class="doccomment">//! selected: all_utxos_selected,</span>
1082 <span class="doccomment">//! selected_amount,</span>
1083 <span class="doccomment">//! fee_amount: fee_amount + additional_fees,</span>
1084 <span class="doccomment">//! })</span>
1085 <span class="doccomment">//! }</span>
1086 <span class="doccomment">//! }</span>
1087 <span class="doccomment">//!</span>
1088 <span class="doccomment">//! # let wallet: OfflineWallet&lt;_&gt; = Wallet::new_offline(&quot;&quot;, None, Network::Testnet, bdk::database::MemoryDatabase::default())?;</span>
1089 <span class="doccomment">//! // create wallet, sync, ...</span>
1090 <span class="doccomment">//!</span>
1091 <span class="doccomment">//! let to_address = Address::from_str(&quot;2N4eQYCbKUHCCTUjBJeHcJp9ok6J2GZsTDt&quot;).unwrap();</span>
1092 <span class="doccomment">//! let (psbt, details) = wallet.create_tx(</span>
1093 <span class="doccomment">//! TxBuilder::with_recipients(vec![(to_address.script_pubkey(), 50_000)])</span>
1094 <span class="doccomment">//! .coin_selection(AlwaysSpendEverything),</span>
1095 <span class="doccomment">//! )?;</span>
1096 <span class="doccomment">//!</span>
1097 <span class="doccomment">//! // inspect, sign, broadcast, ...</span>
1098 <span class="doccomment">//!</span>
1099 <span class="doccomment">//! # Ok::&lt;(), bdk::Error&gt;(())</span>
1100 <span class="doccomment">//! ```</span>
1101
1102 <span class="kw">use</span> <span class="kw">crate</span>::<span class="ident">database</span>::<span class="ident">Database</span>;
1103 <span class="kw">use</span> <span class="kw">crate</span>::<span class="ident">error</span>::<span class="ident">Error</span>;
1104 <span class="kw">use</span> <span class="kw">crate</span>::<span class="ident">types</span>::{<span class="ident">FeeRate</span>, <span class="ident">UTXO</span>};
1105
1106 <span class="kw">use</span> <span class="ident">rand</span>::<span class="ident">seq</span>::<span class="ident">SliceRandom</span>;
1107 <span class="attribute">#[<span class="ident">cfg</span>(<span class="ident">not</span>(<span class="ident">test</span>))]</span>
1108 <span class="kw">use</span> <span class="ident">rand</span>::<span class="ident">thread_rng</span>;
1109 <span class="attribute">#[<span class="ident">cfg</span>(<span class="ident">test</span>)]</span>
1110 <span class="kw">use</span> <span class="ident">rand</span>::{<span class="ident">rngs</span>::<span class="ident">StdRng</span>, <span class="ident">SeedableRng</span>};
1111
1112 <span class="doccomment">/// Default coin selection algorithm used by [`TxBuilder`](super::tx_builder::TxBuilder) if not</span>
1113 <span class="doccomment">/// overridden</span>
1114 <span class="attribute">#[<span class="ident">cfg</span>(<span class="ident">not</span>(<span class="ident">test</span>))]</span>
1115 <span class="kw">pub</span> <span class="kw">type</span> <span class="ident">DefaultCoinSelectionAlgorithm</span> <span class="op">=</span> <span class="ident">BranchAndBoundCoinSelection</span>;
1116 <span class="attribute">#[<span class="ident">cfg</span>(<span class="ident">test</span>)]</span>
1117 <span class="kw">pub</span> <span class="kw">type</span> <span class="ident">DefaultCoinSelectionAlgorithm</span> <span class="op">=</span> <span class="ident">LargestFirstCoinSelection</span>; <span class="comment">// make the tests more predictable</span>
1118
1119 <span class="comment">// Base weight of a Txin, not counting the weight needed for satisfying it.</span>
1120 <span class="comment">// prev_txid (32 bytes) + prev_vout (4 bytes) + sequence (4 bytes) + script_len (1 bytes)</span>
1121 <span class="kw">pub</span>(<span class="kw">crate</span>) <span class="kw">const</span> <span class="ident">TXIN_BASE_WEIGHT</span>: <span class="ident">usize</span> <span class="op">=</span> (<span class="number">32</span> <span class="op">+</span> <span class="number">4</span> <span class="op">+</span> <span class="number">4</span> <span class="op">+</span> <span class="number">1</span>) <span class="op">*</span> <span class="number">4</span>;
1122
1123 <span class="doccomment">/// Result of a successful coin selection</span>
1124 <span class="attribute">#[<span class="ident">derive</span>(<span class="ident">Debug</span>)]</span>
1125 <span class="kw">pub</span> <span class="kw">struct</span> <span class="ident">CoinSelectionResult</span> {
1126 <span class="doccomment">/// List of outputs selected for use as inputs</span>
1127 <span class="kw">pub</span> <span class="ident">selected</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">UTXO</span><span class="op">&gt;</span>,
1128 <span class="doccomment">/// Sum of the selected inputs&#39; value</span>
1129 <span class="kw">pub</span> <span class="ident">selected_amount</span>: <span class="ident">u64</span>,
1130 <span class="doccomment">/// Total fee amount in satoshi</span>
1131 <span class="kw">pub</span> <span class="ident">fee_amount</span>: <span class="ident">f32</span>,
1132 }
1133
1134 <span class="doccomment">/// Trait for generalized coin selection algorithms</span>
1135 <span class="doccomment">///</span>
1136 <span class="doccomment">/// This trait can be implemented to make the [`Wallet`](super::Wallet) use a customized coin</span>
1137 <span class="doccomment">/// selection algorithm when it creates transactions.</span>
1138 <span class="doccomment">///</span>
1139 <span class="doccomment">/// For an example see [this module](crate::wallet::coin_selection)&#39;s documentation.</span>
1140 <span class="kw">pub</span> <span class="kw">trait</span> <span class="ident">CoinSelectionAlgorithm</span><span class="op">&lt;</span><span class="ident">D</span>: <span class="ident">Database</span><span class="op">&gt;</span>: <span class="ident">std</span>::<span class="ident">fmt</span>::<span class="ident">Debug</span> {
1141 <span class="doccomment">/// Perform the coin selection</span>
1142 <span class="doccomment">///</span>
1143 <span class="doccomment">/// - `database`: a reference to the wallet&#39;s database that can be used to lookup additional</span>
1144 <span class="doccomment">/// details for a specific UTXO</span>
1145 <span class="doccomment">/// - `required_utxos`: the utxos that must be spent regardless of `amount_needed` with their</span>
1146 <span class="doccomment">/// weight cost</span>
1147 <span class="doccomment">/// - `optional_utxos`: the remaining available utxos to satisfy `amount_needed` with their</span>
1148 <span class="doccomment">/// weight cost</span>
1149 <span class="doccomment">/// - `fee_rate`: fee rate to use</span>
1150 <span class="doccomment">/// - `amount_needed`: the amount in satoshi to select</span>
1151 <span class="doccomment">/// - `fee_amount`: the amount of fees in satoshi already accumulated from adding outputs and</span>
1152 <span class="doccomment">/// the transaction&#39;s header</span>
1153 <span class="kw">fn</span> <span class="ident">coin_select</span>(
1154 <span class="kw-2">&amp;</span><span class="self">self</span>,
1155 <span class="ident">database</span>: <span class="kw-2">&amp;</span><span class="ident">D</span>,
1156 <span class="ident">required_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span>(<span class="ident">UTXO</span>, <span class="ident">usize</span>)<span class="op">&gt;</span>,
1157 <span class="ident">optional_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span>(<span class="ident">UTXO</span>, <span class="ident">usize</span>)<span class="op">&gt;</span>,
1158 <span class="ident">fee_rate</span>: <span class="ident">FeeRate</span>,
1159 <span class="ident">amount_needed</span>: <span class="ident">u64</span>,
1160 <span class="ident">fee_amount</span>: <span class="ident">f32</span>,
1161 ) <span class="op">-</span><span class="op">&gt;</span> <span class="prelude-ty">Result</span><span class="op">&lt;</span><span class="ident">CoinSelectionResult</span>, <span class="ident">Error</span><span class="op">&gt;</span>;
1162 }
1163
1164 <span class="doccomment">/// Simple and dumb coin selection</span>
1165 <span class="doccomment">///</span>
1166 <span class="doccomment">/// This coin selection algorithm sorts the available UTXOs by value and then picks them starting</span>
1167 <span class="doccomment">/// from the largest ones until the required amount is reached.</span>
1168 <span class="attribute">#[<span class="ident">derive</span>(<span class="ident">Debug</span>, <span class="ident">Default</span>)]</span>
1169 <span class="kw">pub</span> <span class="kw">struct</span> <span class="ident">LargestFirstCoinSelection</span>;
1170
1171 <span class="kw">impl</span><span class="op">&lt;</span><span class="ident">D</span>: <span class="ident">Database</span><span class="op">&gt;</span> <span class="ident">CoinSelectionAlgorithm</span><span class="op">&lt;</span><span class="ident">D</span><span class="op">&gt;</span> <span class="kw">for</span> <span class="ident">LargestFirstCoinSelection</span> {
1172 <span class="kw">fn</span> <span class="ident">coin_select</span>(
1173 <span class="kw-2">&amp;</span><span class="self">self</span>,
1174 <span class="ident">_database</span>: <span class="kw-2">&amp;</span><span class="ident">D</span>,
1175 <span class="ident">required_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span>(<span class="ident">UTXO</span>, <span class="ident">usize</span>)<span class="op">&gt;</span>,
1176 <span class="kw-2">mut</span> <span class="ident">optional_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span>(<span class="ident">UTXO</span>, <span class="ident">usize</span>)<span class="op">&gt;</span>,
1177 <span class="ident">fee_rate</span>: <span class="ident">FeeRate</span>,
1178 <span class="ident">amount_needed</span>: <span class="ident">u64</span>,
1179 <span class="kw-2">mut</span> <span class="ident">fee_amount</span>: <span class="ident">f32</span>,
1180 ) <span class="op">-</span><span class="op">&gt;</span> <span class="prelude-ty">Result</span><span class="op">&lt;</span><span class="ident">CoinSelectionResult</span>, <span class="ident">Error</span><span class="op">&gt;</span> {
1181 <span class="kw">let</span> <span class="ident">calc_fee_bytes</span> <span class="op">=</span> <span class="op">|</span><span class="ident">wu</span><span class="op">|</span> (<span class="ident">wu</span> <span class="kw">as</span> <span class="ident">f32</span>) <span class="op">*</span> <span class="ident">fee_rate</span>.<span class="ident">as_sat_vb</span>() <span class="op">/</span> <span class="number">4.0</span>;
1182
1183 <span class="ident">log</span>::<span class="macro">debug</span><span class="macro">!</span>(
1184 <span class="string">&quot;amount_needed = `{}`, fee_amount = `{}`, fee_rate = `{:?}`&quot;</span>,
1185 <span class="ident">amount_needed</span>,
1186 <span class="ident">fee_amount</span>,
1187 <span class="ident">fee_rate</span>
1188 );
1189
1190 <span class="comment">// We put the &quot;required UTXOs&quot; first and make sure the optional UTXOs are sorted,</span>
1191 <span class="comment">// initially smallest to largest, before being reversed with `.rev()`.</span>
1192 <span class="kw">let</span> <span class="ident">utxos</span> <span class="op">=</span> {
1193 <span class="ident">optional_utxos</span>.<span class="ident">sort_unstable_by_key</span>(<span class="op">|</span>(<span class="ident">utxo</span>, <span class="kw">_</span>)<span class="op">|</span> <span class="ident">utxo</span>.<span class="ident">txout</span>.<span class="ident">value</span>);
1194 <span class="ident">required_utxos</span>
1195 .<span class="ident">into_iter</span>()
1196 .<span class="ident">map</span>(<span class="op">|</span><span class="ident">utxo</span><span class="op">|</span> (<span class="bool-val">true</span>, <span class="ident">utxo</span>))
1197 .<span class="ident">chain</span>(<span class="ident">optional_utxos</span>.<span class="ident">into_iter</span>().<span class="ident">rev</span>().<span class="ident">map</span>(<span class="op">|</span><span class="ident">utxo</span><span class="op">|</span> (<span class="bool-val">false</span>, <span class="ident">utxo</span>)))
1198 };
1199
1200 <span class="comment">// Keep including inputs until we&#39;ve got enough.</span>
1201 <span class="comment">// Store the total input value in selected_amount and the total fee being paid in fee_amount</span>
1202 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">selected_amount</span> <span class="op">=</span> <span class="number">0</span>;
1203 <span class="kw">let</span> <span class="ident">selected</span> <span class="op">=</span> <span class="ident">utxos</span>
1204 .<span class="ident">scan</span>(
1205 (<span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">selected_amount</span>, <span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">fee_amount</span>),
1206 <span class="op">|</span>(<span class="ident">selected_amount</span>, <span class="ident">fee_amount</span>), (<span class="ident">must_use</span>, (<span class="ident">utxo</span>, <span class="ident">weight</span>))<span class="op">|</span> {
1207 <span class="kw">if</span> <span class="ident">must_use</span> <span class="op">|</span><span class="op">|</span> <span class="kw-2">*</span><span class="kw-2">*</span><span class="ident">selected_amount</span> <span class="op">&lt;</span> <span class="ident">amount_needed</span> <span class="op">+</span> (<span class="ident">fee_amount</span>.<span class="ident">ceil</span>() <span class="kw">as</span> <span class="ident">u64</span>) {
1208 <span class="kw-2">*</span><span class="kw-2">*</span><span class="ident">fee_amount</span> <span class="op">+</span><span class="op">=</span> <span class="ident">calc_fee_bytes</span>(<span class="ident">TXIN_BASE_WEIGHT</span> <span class="op">+</span> <span class="ident">weight</span>);
1209 <span class="kw-2">*</span><span class="kw-2">*</span><span class="ident">selected_amount</span> <span class="op">+</span><span class="op">=</span> <span class="ident">utxo</span>.<span class="ident">txout</span>.<span class="ident">value</span>;
1210
1211 <span class="ident">log</span>::<span class="macro">debug</span><span class="macro">!</span>(
1212 <span class="string">&quot;Selected {}, updated fee_amount = `{}`&quot;</span>,
1213 <span class="ident">utxo</span>.<span class="ident">outpoint</span>,
1214 <span class="ident">fee_amount</span>
1215 );
1216
1217 <span class="prelude-val">Some</span>(<span class="ident">utxo</span>)
1218 } <span class="kw">else</span> {
1219 <span class="prelude-val">None</span>
1220 }
1221 },
1222 )
1223 .<span class="ident">collect</span>::<span class="op">&lt;</span><span class="ident">Vec</span><span class="op">&lt;</span><span class="kw">_</span><span class="op">&gt;</span><span class="op">&gt;</span>();
1224
1225 <span class="kw">if</span> <span class="ident">selected_amount</span> <span class="op">&lt;</span> <span class="ident">amount_needed</span> <span class="op">+</span> (<span class="ident">fee_amount</span>.<span class="ident">ceil</span>() <span class="kw">as</span> <span class="ident">u64</span>) {
1226 <span class="kw">return</span> <span class="prelude-val">Err</span>(<span class="ident">Error</span>::<span class="ident">InsufficientFunds</span>);
1227 }
1228
1229 <span class="prelude-val">Ok</span>(<span class="ident">CoinSelectionResult</span> {
1230 <span class="ident">selected</span>,
1231 <span class="ident">fee_amount</span>,
1232 <span class="ident">selected_amount</span>,
1233 })
1234 }
1235 }
1236
1237 <span class="attribute">#[<span class="ident">derive</span>(<span class="ident">Debug</span>, <span class="ident">Clone</span>)]</span>
1238 <span class="comment">// Adds fee information to an UTXO.</span>
1239 <span class="kw">struct</span> <span class="ident">OutputGroup</span> {
1240 <span class="ident">utxo</span>: <span class="ident">UTXO</span>,
1241 <span class="comment">// weight needed to satisfy the UTXO, as described in `Descriptor::max_satisfaction_weight`</span>
1242 <span class="ident">satisfaction_weight</span>: <span class="ident">usize</span>,
1243 <span class="comment">// Amount of fees for spending a certain utxo, calculated using a certain FeeRate</span>
1244 <span class="ident">fee</span>: <span class="ident">f32</span>,
1245 <span class="comment">// The effective value of the UTXO, i.e., the utxo value minus the fee for spending it</span>
1246 <span class="ident">effective_value</span>: <span class="ident">i64</span>,
1247 }
1248
1249 <span class="kw">impl</span> <span class="ident">OutputGroup</span> {
1250 <span class="kw">fn</span> <span class="ident">new</span>(<span class="ident">utxo</span>: <span class="ident">UTXO</span>, <span class="ident">satisfaction_weight</span>: <span class="ident">usize</span>, <span class="ident">fee_rate</span>: <span class="ident">FeeRate</span>) <span class="op">-</span><span class="op">&gt;</span> <span class="self">Self</span> {
1251 <span class="kw">let</span> <span class="ident">fee</span> <span class="op">=</span> (<span class="ident">TXIN_BASE_WEIGHT</span> <span class="op">+</span> <span class="ident">satisfaction_weight</span>) <span class="kw">as</span> <span class="ident">f32</span> <span class="op">/</span> <span class="number">4.0</span> <span class="op">*</span> <span class="ident">fee_rate</span>.<span class="ident">as_sat_vb</span>();
1252 <span class="kw">let</span> <span class="ident">effective_value</span> <span class="op">=</span> <span class="ident">utxo</span>.<span class="ident">txout</span>.<span class="ident">value</span> <span class="kw">as</span> <span class="ident">i64</span> <span class="op">-</span> <span class="ident">fee</span>.<span class="ident">ceil</span>() <span class="kw">as</span> <span class="ident">i64</span>;
1253 <span class="ident">OutputGroup</span> {
1254 <span class="ident">utxo</span>,
1255 <span class="ident">satisfaction_weight</span>,
1256 <span class="ident">effective_value</span>,
1257 <span class="ident">fee</span>,
1258 }
1259 }
1260 }
1261
1262 <span class="doccomment">/// Branch and bound coin selection</span>
1263 <span class="doccomment">///</span>
1264 <span class="doccomment">/// Code adapted from Bitcoin Core&#39;s implementation and from Mark Erhardt Master&#39;s Thesis: &lt;http://murch.one/wp-content/uploads/2016/11/erhardt2016coinselection.pdf&gt;</span>
1265 <span class="attribute">#[<span class="ident">derive</span>(<span class="ident">Debug</span>)]</span>
1266 <span class="kw">pub</span> <span class="kw">struct</span> <span class="ident">BranchAndBoundCoinSelection</span> {
1267 <span class="ident">size_of_change</span>: <span class="ident">u64</span>,
1268 }
1269
1270 <span class="kw">impl</span> <span class="ident">Default</span> <span class="kw">for</span> <span class="ident">BranchAndBoundCoinSelection</span> {
1271 <span class="kw">fn</span> <span class="ident">default</span>() <span class="op">-</span><span class="op">&gt;</span> <span class="self">Self</span> {
1272 <span class="self">Self</span> {
1273 <span class="comment">// P2WPKH cost of change -&gt; value (8 bytes) + script len (1 bytes) + script (22 bytes)</span>
1274 <span class="ident">size_of_change</span>: <span class="number">8</span> <span class="op">+</span> <span class="number">1</span> <span class="op">+</span> <span class="number">22</span>,
1275 }
1276 }
1277 }
1278
1279 <span class="kw">impl</span> <span class="ident">BranchAndBoundCoinSelection</span> {
1280 <span class="doccomment">/// Create new instance with target size for change output</span>
1281 <span class="kw">pub</span> <span class="kw">fn</span> <span class="ident">new</span>(<span class="ident">size_of_change</span>: <span class="ident">u64</span>) <span class="op">-</span><span class="op">&gt;</span> <span class="self">Self</span> {
1282 <span class="self">Self</span> { <span class="ident">size_of_change</span> }
1283 }
1284 }
1285
1286 <span class="kw">const</span> <span class="ident">BNB_TOTAL_TRIES</span>: <span class="ident">usize</span> <span class="op">=</span> <span class="number">100_000</span>;
1287
1288 <span class="kw">impl</span><span class="op">&lt;</span><span class="ident">D</span>: <span class="ident">Database</span><span class="op">&gt;</span> <span class="ident">CoinSelectionAlgorithm</span><span class="op">&lt;</span><span class="ident">D</span><span class="op">&gt;</span> <span class="kw">for</span> <span class="ident">BranchAndBoundCoinSelection</span> {
1289 <span class="kw">fn</span> <span class="ident">coin_select</span>(
1290 <span class="kw-2">&amp;</span><span class="self">self</span>,
1291 <span class="ident">_database</span>: <span class="kw-2">&amp;</span><span class="ident">D</span>,
1292 <span class="ident">required_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span>(<span class="ident">UTXO</span>, <span class="ident">usize</span>)<span class="op">&gt;</span>,
1293 <span class="ident">optional_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span>(<span class="ident">UTXO</span>, <span class="ident">usize</span>)<span class="op">&gt;</span>,
1294 <span class="ident">fee_rate</span>: <span class="ident">FeeRate</span>,
1295 <span class="ident">amount_needed</span>: <span class="ident">u64</span>,
1296 <span class="ident">fee_amount</span>: <span class="ident">f32</span>,
1297 ) <span class="op">-</span><span class="op">&gt;</span> <span class="prelude-ty">Result</span><span class="op">&lt;</span><span class="ident">CoinSelectionResult</span>, <span class="ident">Error</span><span class="op">&gt;</span> {
1298 <span class="comment">// Mapping every (UTXO, usize) to an output group</span>
1299 <span class="kw">let</span> <span class="ident">required_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">OutputGroup</span><span class="op">&gt;</span> <span class="op">=</span> <span class="ident">required_utxos</span>
1300 .<span class="ident">into_iter</span>()
1301 .<span class="ident">map</span>(<span class="op">|</span><span class="ident">u</span><span class="op">|</span> <span class="ident">OutputGroup</span>::<span class="ident">new</span>(<span class="ident">u</span>.<span class="number">0</span>, <span class="ident">u</span>.<span class="number">1</span>, <span class="ident">fee_rate</span>))
1302 .<span class="ident">collect</span>();
1303
1304 <span class="comment">// Mapping every (UTXO, usize) to an output group.</span>
1305 <span class="comment">// Filtering UTXOs with an effective_value &lt; 0, as the fee paid for</span>
1306 <span class="comment">// adding them is more than their value</span>
1307 <span class="kw">let</span> <span class="ident">optional_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">OutputGroup</span><span class="op">&gt;</span> <span class="op">=</span> <span class="ident">optional_utxos</span>
1308 .<span class="ident">into_iter</span>()
1309 .<span class="ident">map</span>(<span class="op">|</span><span class="ident">u</span><span class="op">|</span> <span class="ident">OutputGroup</span>::<span class="ident">new</span>(<span class="ident">u</span>.<span class="number">0</span>, <span class="ident">u</span>.<span class="number">1</span>, <span class="ident">fee_rate</span>))
1310 .<span class="ident">filter</span>(<span class="op">|</span><span class="ident">u</span><span class="op">|</span> <span class="ident">u</span>.<span class="ident">effective_value</span> <span class="op">&gt;</span> <span class="number">0</span>)
1311 .<span class="ident">collect</span>();
1312
1313 <span class="kw">let</span> <span class="ident">curr_value</span> <span class="op">=</span> <span class="ident">required_utxos</span>
1314 .<span class="ident">iter</span>()
1315 .<span class="ident">fold</span>(<span class="number">0</span>, <span class="op">|</span><span class="ident">acc</span>, <span class="ident">x</span><span class="op">|</span> <span class="ident">acc</span> <span class="op">+</span> <span class="ident">x</span>.<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>);
1316
1317 <span class="kw">let</span> <span class="ident">curr_available_value</span> <span class="op">=</span> <span class="ident">optional_utxos</span>
1318 .<span class="ident">iter</span>()
1319 .<span class="ident">fold</span>(<span class="number">0</span>, <span class="op">|</span><span class="ident">acc</span>, <span class="ident">x</span><span class="op">|</span> <span class="ident">acc</span> <span class="op">+</span> <span class="ident">x</span>.<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>);
1320
1321 <span class="kw">let</span> <span class="ident">actual_target</span> <span class="op">=</span> <span class="ident">fee_amount</span>.<span class="ident">ceil</span>() <span class="kw">as</span> <span class="ident">u64</span> <span class="op">+</span> <span class="ident">amount_needed</span>;
1322 <span class="kw">let</span> <span class="ident">cost_of_change</span> <span class="op">=</span> <span class="self">self</span>.<span class="ident">size_of_change</span> <span class="kw">as</span> <span class="ident">f32</span> <span class="op">*</span> <span class="ident">fee_rate</span>.<span class="ident">as_sat_vb</span>();
1323
1324 <span class="kw">if</span> <span class="ident">curr_available_value</span> <span class="op">+</span> <span class="ident">curr_value</span> <span class="op">&lt;</span> <span class="ident">actual_target</span> {
1325 <span class="kw">return</span> <span class="prelude-val">Err</span>(<span class="ident">Error</span>::<span class="ident">InsufficientFunds</span>);
1326 }
1327
1328 <span class="prelude-val">Ok</span>(<span class="self">self</span>
1329 .<span class="ident">bnb</span>(
1330 <span class="ident">required_utxos</span>.<span class="ident">clone</span>(),
1331 <span class="ident">optional_utxos</span>.<span class="ident">clone</span>(),
1332 <span class="ident">curr_value</span>,
1333 <span class="ident">curr_available_value</span>,
1334 <span class="ident">actual_target</span>,
1335 <span class="ident">fee_amount</span>,
1336 <span class="ident">cost_of_change</span>,
1337 )
1338 .<span class="ident">unwrap_or_else</span>(<span class="op">|</span><span class="kw">_</span><span class="op">|</span> {
1339 <span class="self">self</span>.<span class="ident">single_random_draw</span>(
1340 <span class="ident">required_utxos</span>,
1341 <span class="ident">optional_utxos</span>,
1342 <span class="ident">curr_value</span>,
1343 <span class="ident">actual_target</span>,
1344 <span class="ident">fee_amount</span>,
1345 )
1346 }))
1347 }
1348 }
1349
1350 <span class="kw">impl</span> <span class="ident">BranchAndBoundCoinSelection</span> {
1351 <span class="comment">// TODO: make this more Rust-onic :)</span>
1352 <span class="comment">// (And perhpaps refactor with less arguments?)</span>
1353 <span class="attribute">#[<span class="ident">allow</span>(<span class="ident">clippy</span>::<span class="ident">too_many_arguments</span>)]</span>
1354 <span class="kw">fn</span> <span class="ident">bnb</span>(
1355 <span class="kw-2">&amp;</span><span class="self">self</span>,
1356 <span class="ident">required_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">OutputGroup</span><span class="op">&gt;</span>,
1357 <span class="kw-2">mut</span> <span class="ident">optional_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">OutputGroup</span><span class="op">&gt;</span>,
1358 <span class="kw-2">mut</span> <span class="ident">curr_value</span>: <span class="ident">u64</span>,
1359 <span class="kw-2">mut</span> <span class="ident">curr_available_value</span>: <span class="ident">u64</span>,
1360 <span class="ident">actual_target</span>: <span class="ident">u64</span>,
1361 <span class="ident">fee_amount</span>: <span class="ident">f32</span>,
1362 <span class="ident">cost_of_change</span>: <span class="ident">f32</span>,
1363 ) <span class="op">-</span><span class="op">&gt;</span> <span class="prelude-ty">Result</span><span class="op">&lt;</span><span class="ident">CoinSelectionResult</span>, <span class="ident">Error</span><span class="op">&gt;</span> {
1364 <span class="comment">// current_selection[i] will contain true if we are using optional_utxos[i],</span>
1365 <span class="comment">// false otherwise. Note that current_selection.len() could be less than</span>
1366 <span class="comment">// optional_utxos.len(), it just means that we still haven&#39;t decided if we should keep</span>
1367 <span class="comment">// certain optional_utxos or not.</span>
1368 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">current_selection</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">bool</span><span class="op">&gt;</span> <span class="op">=</span> <span class="ident">Vec</span>::<span class="ident">with_capacity</span>(<span class="ident">optional_utxos</span>.<span class="ident">len</span>());
1369
1370 <span class="comment">// Sort the utxo_pool</span>
1371 <span class="ident">optional_utxos</span>.<span class="ident">sort_unstable_by_key</span>(<span class="op">|</span><span class="ident">a</span><span class="op">|</span> <span class="ident">a</span>.<span class="ident">effective_value</span>);
1372 <span class="ident">optional_utxos</span>.<span class="ident">reverse</span>();
1373
1374 <span class="comment">// Contains the best selection we found</span>
1375 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">best_selection</span> <span class="op">=</span> <span class="ident">Vec</span>::<span class="ident">new</span>();
1376 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">best_selection_value</span> <span class="op">=</span> <span class="prelude-val">None</span>;
1377
1378 <span class="comment">// Depth First search loop for choosing the UTXOs</span>
1379 <span class="kw">for</span> <span class="kw">_</span> <span class="kw">in</span> <span class="number">0</span>..<span class="ident">BNB_TOTAL_TRIES</span> {
1380 <span class="comment">// Conditions for starting a backtrack</span>
1381 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">backtrack</span> <span class="op">=</span> <span class="bool-val">false</span>;
1382 <span class="comment">// Cannot possibly reach target with the amount remaining in the curr_available_value,</span>
1383 <span class="comment">// or the selected value is out of range.</span>
1384 <span class="comment">// Go back and try other branch</span>
1385 <span class="kw">if</span> <span class="ident">curr_value</span> <span class="op">+</span> <span class="ident">curr_available_value</span> <span class="op">&lt;</span> <span class="ident">actual_target</span>
1386 <span class="op">|</span><span class="op">|</span> <span class="ident">curr_value</span> <span class="op">&gt;</span> <span class="ident">actual_target</span> <span class="op">+</span> <span class="ident">cost_of_change</span> <span class="kw">as</span> <span class="ident">u64</span>
1387 {
1388 <span class="ident">backtrack</span> <span class="op">=</span> <span class="bool-val">true</span>;
1389 } <span class="kw">else</span> <span class="kw">if</span> <span class="ident">curr_value</span> <span class="op">&gt;</span><span class="op">=</span> <span class="ident">actual_target</span> {
1390 <span class="comment">// Selected value is within range, there&#39;s no point in going forward. Start</span>
1391 <span class="comment">// backtracking</span>
1392 <span class="ident">backtrack</span> <span class="op">=</span> <span class="bool-val">true</span>;
1393
1394 <span class="comment">// If we found a solution better than the previous one, or if there wasn&#39;t previous</span>
1395 <span class="comment">// solution, update the best solution</span>
1396 <span class="kw">if</span> <span class="ident">best_selection_value</span>.<span class="ident">is_none</span>() <span class="op">|</span><span class="op">|</span> <span class="ident">curr_value</span> <span class="op">&lt;</span> <span class="ident">best_selection_value</span>.<span class="ident">unwrap</span>() {
1397 <span class="ident">best_selection</span> <span class="op">=</span> <span class="ident">current_selection</span>.<span class="ident">clone</span>();
1398 <span class="ident">best_selection_value</span> <span class="op">=</span> <span class="prelude-val">Some</span>(<span class="ident">curr_value</span>);
1399 }
1400
1401 <span class="comment">// If we found a perfect match, break here</span>
1402 <span class="kw">if</span> <span class="ident">curr_value</span> <span class="op">=</span><span class="op">=</span> <span class="ident">actual_target</span> {
1403 <span class="kw">break</span>;
1404 }
1405 }
1406
1407 <span class="comment">// Backtracking, moving backwards</span>
1408 <span class="kw">if</span> <span class="ident">backtrack</span> {
1409 <span class="comment">// Walk backwards to find the last included UTXO that still needs to have its omission branch traversed.</span>
1410 <span class="kw">while</span> <span class="kw">let</span> <span class="prelude-val">Some</span>(<span class="bool-val">false</span>) <span class="op">=</span> <span class="ident">current_selection</span>.<span class="ident">last</span>() {
1411 <span class="ident">current_selection</span>.<span class="ident">pop</span>();
1412 <span class="ident">curr_available_value</span> <span class="op">+</span><span class="op">=</span>
1413 <span class="ident">optional_utxos</span>[<span class="ident">current_selection</span>.<span class="ident">len</span>()].<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>;
1414 }
1415
1416 <span class="kw">if</span> <span class="ident">current_selection</span>.<span class="ident">last_mut</span>().<span class="ident">is_none</span>() {
1417 <span class="comment">// We have walked back to the first utxo and no branch is untraversed. All solutions searched</span>
1418 <span class="comment">// If best selection is empty, then there&#39;s no exact match</span>
1419 <span class="kw">if</span> <span class="ident">best_selection</span>.<span class="ident">is_empty</span>() {
1420 <span class="kw">return</span> <span class="prelude-val">Err</span>(<span class="ident">Error</span>::<span class="ident">BnBNoExactMatch</span>);
1421 }
1422 <span class="kw">break</span>;
1423 }
1424
1425 <span class="kw">if</span> <span class="kw">let</span> <span class="prelude-val">Some</span>(<span class="ident">c</span>) <span class="op">=</span> <span class="ident">current_selection</span>.<span class="ident">last_mut</span>() {
1426 <span class="comment">// Output was included on previous iterations, try excluding now.</span>
1427 <span class="kw-2">*</span><span class="ident">c</span> <span class="op">=</span> <span class="bool-val">false</span>;
1428 }
1429
1430 <span class="kw">let</span> <span class="ident">utxo</span> <span class="op">=</span> <span class="kw-2">&amp;</span><span class="ident">optional_utxos</span>[<span class="ident">current_selection</span>.<span class="ident">len</span>() <span class="op">-</span> <span class="number">1</span>];
1431 <span class="ident">curr_value</span> <span class="op">-</span><span class="op">=</span> <span class="ident">utxo</span>.<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>;
1432 } <span class="kw">else</span> {
1433 <span class="comment">// Moving forwards, continuing down this branch</span>
1434 <span class="kw">let</span> <span class="ident">utxo</span> <span class="op">=</span> <span class="kw-2">&amp;</span><span class="ident">optional_utxos</span>[<span class="ident">current_selection</span>.<span class="ident">len</span>()];
1435
1436 <span class="comment">// Remove this utxo from the curr_available_value utxo amount</span>
1437 <span class="ident">curr_available_value</span> <span class="op">-</span><span class="op">=</span> <span class="ident">utxo</span>.<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>;
1438
1439 <span class="comment">// Inclusion branch first (Largest First Exploration)</span>
1440 <span class="ident">current_selection</span>.<span class="ident">push</span>(<span class="bool-val">true</span>);
1441 <span class="ident">curr_value</span> <span class="op">+</span><span class="op">=</span> <span class="ident">utxo</span>.<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>;
1442 }
1443 }
1444
1445 <span class="comment">// Check for solution</span>
1446 <span class="kw">if</span> <span class="ident">best_selection</span>.<span class="ident">is_empty</span>() {
1447 <span class="kw">return</span> <span class="prelude-val">Err</span>(<span class="ident">Error</span>::<span class="ident">BnBTotalTriesExceeded</span>);
1448 }
1449
1450 <span class="comment">// Set output set</span>
1451 <span class="kw">let</span> <span class="ident">selected_utxos</span> <span class="op">=</span> <span class="ident">optional_utxos</span>
1452 .<span class="ident">into_iter</span>()
1453 .<span class="ident">zip</span>(<span class="ident">best_selection</span>)
1454 .<span class="ident">filter_map</span>(<span class="op">|</span>(<span class="ident">optional</span>, <span class="ident">is_in_best</span>)<span class="op">|</span> <span class="kw">if</span> <span class="ident">is_in_best</span> { <span class="prelude-val">Some</span>(<span class="ident">optional</span>) } <span class="kw">else</span> { <span class="prelude-val">None</span> })
1455 .<span class="ident">collect</span>();
1456
1457 <span class="prelude-val">Ok</span>(<span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">calculate_cs_result</span>(
1458 <span class="ident">selected_utxos</span>,
1459 <span class="ident">required_utxos</span>,
1460 <span class="ident">fee_amount</span>,
1461 ))
1462 }
1463
1464 <span class="kw">fn</span> <span class="ident">single_random_draw</span>(
1465 <span class="kw-2">&amp;</span><span class="self">self</span>,
1466 <span class="ident">required_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">OutputGroup</span><span class="op">&gt;</span>,
1467 <span class="kw-2">mut</span> <span class="ident">optional_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">OutputGroup</span><span class="op">&gt;</span>,
1468 <span class="ident">curr_value</span>: <span class="ident">u64</span>,
1469 <span class="ident">actual_target</span>: <span class="ident">u64</span>,
1470 <span class="ident">fee_amount</span>: <span class="ident">f32</span>,
1471 ) <span class="op">-</span><span class="op">&gt;</span> <span class="ident">CoinSelectionResult</span> {
1472 <span class="attribute">#[<span class="ident">cfg</span>(<span class="ident">not</span>(<span class="ident">test</span>))]</span>
1473 <span class="ident">optional_utxos</span>.<span class="ident">shuffle</span>(<span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">thread_rng</span>());
1474 <span class="attribute">#[<span class="ident">cfg</span>(<span class="ident">test</span>)]</span>
1475 {
1476 <span class="kw">let</span> <span class="ident">seed</span> <span class="op">=</span> [<span class="number">0</span>; <span class="number">32</span>];
1477 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">rng</span>: <span class="ident">StdRng</span> <span class="op">=</span> <span class="ident">SeedableRng</span>::<span class="ident">from_seed</span>(<span class="ident">seed</span>);
1478 <span class="ident">optional_utxos</span>.<span class="ident">shuffle</span>(<span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">rng</span>);
1479 }
1480
1481 <span class="kw">let</span> <span class="ident">selected_utxos</span> <span class="op">=</span> <span class="ident">optional_utxos</span>
1482 .<span class="ident">into_iter</span>()
1483 .<span class="ident">scan</span>(<span class="ident">curr_value</span>, <span class="op">|</span><span class="ident">curr_value</span>, <span class="ident">utxo</span><span class="op">|</span> {
1484 <span class="kw">if</span> <span class="kw-2">*</span><span class="ident">curr_value</span> <span class="op">&gt;</span><span class="op">=</span> <span class="ident">actual_target</span> {
1485 <span class="prelude-val">None</span>
1486 } <span class="kw">else</span> {
1487 <span class="kw-2">*</span><span class="ident">curr_value</span> <span class="op">+</span><span class="op">=</span> <span class="ident">utxo</span>.<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>;
1488 <span class="prelude-val">Some</span>(<span class="ident">utxo</span>)
1489 }
1490 })
1491 .<span class="ident">collect</span>::<span class="op">&lt;</span><span class="ident">Vec</span><span class="op">&lt;</span><span class="kw">_</span><span class="op">&gt;</span><span class="op">&gt;</span>();
1492
1493 <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">calculate_cs_result</span>(<span class="ident">selected_utxos</span>, <span class="ident">required_utxos</span>, <span class="ident">fee_amount</span>)
1494 }
1495
1496 <span class="kw">fn</span> <span class="ident">calculate_cs_result</span>(
1497 <span class="kw-2">mut</span> <span class="ident">selected_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">OutputGroup</span><span class="op">&gt;</span>,
1498 <span class="kw-2">mut</span> <span class="ident">required_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">OutputGroup</span><span class="op">&gt;</span>,
1499 <span class="kw-2">mut</span> <span class="ident">fee_amount</span>: <span class="ident">f32</span>,
1500 ) <span class="op">-</span><span class="op">&gt;</span> <span class="ident">CoinSelectionResult</span> {
1501 <span class="ident">selected_utxos</span>.<span class="ident">append</span>(<span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">required_utxos</span>);
1502 <span class="ident">fee_amount</span> <span class="op">+</span><span class="op">=</span> <span class="ident">selected_utxos</span>.<span class="ident">iter</span>().<span class="ident">map</span>(<span class="op">|</span><span class="ident">u</span><span class="op">|</span> <span class="ident">u</span>.<span class="ident">fee</span>).<span class="ident">sum</span>::<span class="op">&lt;</span><span class="ident">f32</span><span class="op">&gt;</span>();
1503 <span class="kw">let</span> <span class="ident">selected</span> <span class="op">=</span> <span class="ident">selected_utxos</span>
1504 .<span class="ident">into_iter</span>()
1505 .<span class="ident">map</span>(<span class="op">|</span><span class="ident">u</span><span class="op">|</span> <span class="ident">u</span>.<span class="ident">utxo</span>)
1506 .<span class="ident">collect</span>::<span class="op">&lt;</span><span class="ident">Vec</span><span class="op">&lt;</span><span class="kw">_</span><span class="op">&gt;</span><span class="op">&gt;</span>();
1507 <span class="kw">let</span> <span class="ident">selected_amount</span> <span class="op">=</span> <span class="ident">selected</span>.<span class="ident">iter</span>().<span class="ident">map</span>(<span class="op">|</span><span class="ident">u</span><span class="op">|</span> <span class="ident">u</span>.<span class="ident">txout</span>.<span class="ident">value</span>).<span class="ident">sum</span>();
1508
1509 <span class="ident">CoinSelectionResult</span> {
1510 <span class="ident">selected</span>,
1511 <span class="ident">fee_amount</span>,
1512 <span class="ident">selected_amount</span>,
1513 }
1514 }
1515 }
1516
1517 <span class="attribute">#[<span class="ident">cfg</span>(<span class="ident">test</span>)]</span>
1518 <span class="kw">mod</span> <span class="ident">test</span> {
1519 <span class="kw">use</span> <span class="ident">std</span>::<span class="ident">str</span>::<span class="ident">FromStr</span>;
1520
1521 <span class="kw">use</span> <span class="ident">bitcoin</span>::{<span class="ident">OutPoint</span>, <span class="ident">Script</span>, <span class="ident">TxOut</span>};
1522
1523 <span class="kw">use</span> <span class="kw">super</span>::<span class="kw-2">*</span>;
1524 <span class="kw">use</span> <span class="kw">crate</span>::<span class="ident">database</span>::<span class="ident">MemoryDatabase</span>;
1525 <span class="kw">use</span> <span class="kw">crate</span>::<span class="ident">types</span>::<span class="kw-2">*</span>;
1526
1527 <span class="kw">use</span> <span class="ident">rand</span>::<span class="ident">rngs</span>::<span class="ident">StdRng</span>;
1528 <span class="kw">use</span> <span class="ident">rand</span>::<span class="ident">seq</span>::<span class="ident">SliceRandom</span>;
1529 <span class="kw">use</span> <span class="ident">rand</span>::{<span class="ident">Rng</span>, <span class="ident">SeedableRng</span>};
1530
1531 <span class="kw">const</span> <span class="ident">P2WPKH_WITNESS_SIZE</span>: <span class="ident">usize</span> <span class="op">=</span> <span class="number">73</span> <span class="op">+</span> <span class="number">33</span> <span class="op">+</span> <span class="number">2</span>;
1532
1533 <span class="kw">fn</span> <span class="ident">get_test_utxos</span>() <span class="op">-</span><span class="op">&gt;</span> <span class="ident">Vec</span><span class="op">&lt;</span>(<span class="ident">UTXO</span>, <span class="ident">usize</span>)<span class="op">&gt;</span> {
1534 <span class="macro">vec</span><span class="macro">!</span>[
1535 (
1536 <span class="ident">UTXO</span> {
1537 <span class="ident">outpoint</span>: <span class="ident">OutPoint</span>::<span class="ident">from_str</span>(
1538 <span class="string">&quot;ebd9813ecebc57ff8f30797de7c205e3c7498ca950ea4341ee51a685ff2fa30a:0&quot;</span>,
1539 )
1540 .<span class="ident">unwrap</span>(),
1541 <span class="ident">txout</span>: <span class="ident">TxOut</span> {
1542 <span class="ident">value</span>: <span class="number">100_000</span>,
1543 <span class="ident">script_pubkey</span>: <span class="ident">Script</span>::<span class="ident">new</span>(),
1544 },
1545 <span class="ident">keychain</span>: <span class="ident">KeychainKind</span>::<span class="ident">External</span>,
1546 },
1547 <span class="ident">P2WPKH_WITNESS_SIZE</span>,
1548 ),
1549 (
1550 <span class="ident">UTXO</span> {
1551 <span class="ident">outpoint</span>: <span class="ident">OutPoint</span>::<span class="ident">from_str</span>(
1552 <span class="string">&quot;65d92ddff6b6dc72c89624a6491997714b90f6004f928d875bc0fd53f264fa85:0&quot;</span>,
1553 )
1554 .<span class="ident">unwrap</span>(),
1555 <span class="ident">txout</span>: <span class="ident">TxOut</span> {
1556 <span class="ident">value</span>: <span class="number">200_000</span>,
1557 <span class="ident">script_pubkey</span>: <span class="ident">Script</span>::<span class="ident">new</span>(),
1558 },
1559 <span class="ident">keychain</span>: <span class="ident">KeychainKind</span>::<span class="ident">Internal</span>,
1560 },
1561 <span class="ident">P2WPKH_WITNESS_SIZE</span>,
1562 ),
1563 ]
1564 }
1565
1566 <span class="kw">fn</span> <span class="ident">generate_random_utxos</span>(<span class="ident">rng</span>: <span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">StdRng</span>, <span class="ident">utxos_number</span>: <span class="ident">usize</span>) <span class="op">-</span><span class="op">&gt;</span> <span class="ident">Vec</span><span class="op">&lt;</span>(<span class="ident">UTXO</span>, <span class="ident">usize</span>)<span class="op">&gt;</span> {
1567 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">res</span> <span class="op">=</span> <span class="ident">Vec</span>::<span class="ident">new</span>();
1568 <span class="kw">for</span> <span class="kw">_</span> <span class="kw">in</span> <span class="number">0</span>..<span class="ident">utxos_number</span> {
1569 <span class="ident">res</span>.<span class="ident">push</span>((
1570 <span class="ident">UTXO</span> {
1571 <span class="ident">outpoint</span>: <span class="ident">OutPoint</span>::<span class="ident">from_str</span>(
1572 <span class="string">&quot;ebd9813ecebc57ff8f30797de7c205e3c7498ca950ea4341ee51a685ff2fa30a:0&quot;</span>,
1573 )
1574 .<span class="ident">unwrap</span>(),
1575 <span class="ident">txout</span>: <span class="ident">TxOut</span> {
1576 <span class="ident">value</span>: <span class="ident">rng</span>.<span class="ident">gen_range</span>(<span class="number">0</span>, <span class="number">200000000</span>),
1577 <span class="ident">script_pubkey</span>: <span class="ident">Script</span>::<span class="ident">new</span>(),
1578 },
1579 <span class="ident">keychain</span>: <span class="ident">KeychainKind</span>::<span class="ident">External</span>,
1580 },
1581 <span class="ident">P2WPKH_WITNESS_SIZE</span>,
1582 ));
1583 }
1584 <span class="ident">res</span>
1585 }
1586
1587 <span class="kw">fn</span> <span class="ident">generate_same_value_utxos</span>(<span class="ident">utxos_value</span>: <span class="ident">u64</span>, <span class="ident">utxos_number</span>: <span class="ident">usize</span>) <span class="op">-</span><span class="op">&gt;</span> <span class="ident">Vec</span><span class="op">&lt;</span>(<span class="ident">UTXO</span>, <span class="ident">usize</span>)<span class="op">&gt;</span> {
1588 <span class="kw">let</span> <span class="ident">utxo</span> <span class="op">=</span> (
1589 <span class="ident">UTXO</span> {
1590 <span class="ident">outpoint</span>: <span class="ident">OutPoint</span>::<span class="ident">from_str</span>(
1591 <span class="string">&quot;ebd9813ecebc57ff8f30797de7c205e3c7498ca950ea4341ee51a685ff2fa30a:0&quot;</span>,
1592 )
1593 .<span class="ident">unwrap</span>(),
1594 <span class="ident">txout</span>: <span class="ident">TxOut</span> {
1595 <span class="ident">value</span>: <span class="ident">utxos_value</span>,
1596 <span class="ident">script_pubkey</span>: <span class="ident">Script</span>::<span class="ident">new</span>(),
1597 },
1598 <span class="ident">keychain</span>: <span class="ident">KeychainKind</span>::<span class="ident">External</span>,
1599 },
1600 <span class="ident">P2WPKH_WITNESS_SIZE</span>,
1601 );
1602 <span class="macro">vec</span><span class="macro">!</span>[<span class="ident">utxo</span>; <span class="ident">utxos_number</span>]
1603 }
1604
1605 <span class="kw">fn</span> <span class="ident">sum_random_utxos</span>(<span class="kw-2">mut</span> <span class="ident">rng</span>: <span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">StdRng</span>, <span class="ident">utxos</span>: <span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">Vec</span><span class="op">&lt;</span>(<span class="ident">UTXO</span>, <span class="ident">usize</span>)<span class="op">&gt;</span>) <span class="op">-</span><span class="op">&gt;</span> <span class="ident">u64</span> {
1606 <span class="kw">let</span> <span class="ident">utxos_picked_len</span> <span class="op">=</span> <span class="ident">rng</span>.<span class="ident">gen_range</span>(<span class="number">2</span>, <span class="ident">utxos</span>.<span class="ident">len</span>() <span class="op">/</span> <span class="number">2</span>);
1607 <span class="ident">utxos</span>.<span class="ident">shuffle</span>(<span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">rng</span>);
1608 <span class="ident">utxos</span>[..<span class="ident">utxos_picked_len</span>]
1609 .<span class="ident">iter</span>()
1610 .<span class="ident">fold</span>(<span class="number">0</span>, <span class="op">|</span><span class="ident">acc</span>, <span class="ident">x</span><span class="op">|</span> <span class="ident">acc</span> <span class="op">+</span> <span class="ident">x</span>.<span class="number">0</span>.<span class="ident">txout</span>.<span class="ident">value</span>)
1611 }
1612
1613 <span class="attribute">#[<span class="ident">test</span>]</span>
1614 <span class="kw">fn</span> <span class="ident">test_largest_first_coin_selection_success</span>() {
1615 <span class="kw">let</span> <span class="ident">utxos</span> <span class="op">=</span> <span class="ident">get_test_utxos</span>();
1616 <span class="kw">let</span> <span class="ident">database</span> <span class="op">=</span> <span class="ident">MemoryDatabase</span>::<span class="ident">default</span>();
1617
1618 <span class="kw">let</span> <span class="ident">result</span> <span class="op">=</span> <span class="ident">LargestFirstCoinSelection</span>::<span class="ident">default</span>()
1619 .<span class="ident">coin_select</span>(
1620 <span class="kw-2">&amp;</span><span class="ident">database</span>,
1621 <span class="ident">utxos</span>,
1622 <span class="macro">vec</span><span class="macro">!</span>[],
1623 <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1.0</span>),
1624 <span class="number">250_000</span>,
1625 <span class="number">50.0</span>,
1626 )
1627 .<span class="ident">unwrap</span>();
1628
1629 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected</span>.<span class="ident">len</span>(), <span class="number">2</span>);
1630 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected_amount</span>, <span class="number">300_000</span>);
1631 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">fee_amount</span>, <span class="number">186.0</span>);
1632 }
1633
1634 <span class="attribute">#[<span class="ident">test</span>]</span>
1635 <span class="kw">fn</span> <span class="ident">test_largest_first_coin_selection_use_all</span>() {
1636 <span class="kw">let</span> <span class="ident">utxos</span> <span class="op">=</span> <span class="ident">get_test_utxos</span>();
1637 <span class="kw">let</span> <span class="ident">database</span> <span class="op">=</span> <span class="ident">MemoryDatabase</span>::<span class="ident">default</span>();
1638
1639 <span class="kw">let</span> <span class="ident">result</span> <span class="op">=</span> <span class="ident">LargestFirstCoinSelection</span>::<span class="ident">default</span>()
1640 .<span class="ident">coin_select</span>(
1641 <span class="kw-2">&amp;</span><span class="ident">database</span>,
1642 <span class="ident">utxos</span>,
1643 <span class="macro">vec</span><span class="macro">!</span>[],
1644 <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1.0</span>),
1645 <span class="number">20_000</span>,
1646 <span class="number">50.0</span>,
1647 )
1648 .<span class="ident">unwrap</span>();
1649
1650 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected</span>.<span class="ident">len</span>(), <span class="number">2</span>);
1651 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected_amount</span>, <span class="number">300_000</span>);
1652 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">fee_amount</span>, <span class="number">186.0</span>);
1653 }
1654
1655 <span class="attribute">#[<span class="ident">test</span>]</span>
1656 <span class="kw">fn</span> <span class="ident">test_largest_first_coin_selection_use_only_necessary</span>() {
1657 <span class="kw">let</span> <span class="ident">utxos</span> <span class="op">=</span> <span class="ident">get_test_utxos</span>();
1658 <span class="kw">let</span> <span class="ident">database</span> <span class="op">=</span> <span class="ident">MemoryDatabase</span>::<span class="ident">default</span>();
1659
1660 <span class="kw">let</span> <span class="ident">result</span> <span class="op">=</span> <span class="ident">LargestFirstCoinSelection</span>::<span class="ident">default</span>()
1661 .<span class="ident">coin_select</span>(
1662 <span class="kw-2">&amp;</span><span class="ident">database</span>,
1663 <span class="macro">vec</span><span class="macro">!</span>[],
1664 <span class="ident">utxos</span>,
1665 <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1.0</span>),
1666 <span class="number">20_000</span>,
1667 <span class="number">50.0</span>,
1668 )
1669 .<span class="ident">unwrap</span>();
1670
1671 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected</span>.<span class="ident">len</span>(), <span class="number">1</span>);
1672 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected_amount</span>, <span class="number">200_000</span>);
1673 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">fee_amount</span>, <span class="number">118.0</span>);
1674 }
1675
1676 <span class="attribute">#[<span class="ident">test</span>]</span>
1677 <span class="attribute">#[<span class="ident">should_panic</span>(<span class="ident">expected</span> <span class="op">=</span> <span class="string">&quot;InsufficientFunds&quot;</span>)]</span>
1678 <span class="kw">fn</span> <span class="ident">test_largest_first_coin_selection_insufficient_funds</span>() {
1679 <span class="kw">let</span> <span class="ident">utxos</span> <span class="op">=</span> <span class="ident">get_test_utxos</span>();
1680 <span class="kw">let</span> <span class="ident">database</span> <span class="op">=</span> <span class="ident">MemoryDatabase</span>::<span class="ident">default</span>();
1681
1682 <span class="ident">LargestFirstCoinSelection</span>::<span class="ident">default</span>()
1683 .<span class="ident">coin_select</span>(
1684 <span class="kw-2">&amp;</span><span class="ident">database</span>,
1685 <span class="macro">vec</span><span class="macro">!</span>[],
1686 <span class="ident">utxos</span>,
1687 <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1.0</span>),
1688 <span class="number">500_000</span>,
1689 <span class="number">50.0</span>,
1690 )
1691 .<span class="ident">unwrap</span>();
1692 }
1693
1694 <span class="attribute">#[<span class="ident">test</span>]</span>
1695 <span class="attribute">#[<span class="ident">should_panic</span>(<span class="ident">expected</span> <span class="op">=</span> <span class="string">&quot;InsufficientFunds&quot;</span>)]</span>
1696 <span class="kw">fn</span> <span class="ident">test_largest_first_coin_selection_insufficient_funds_high_fees</span>() {
1697 <span class="kw">let</span> <span class="ident">utxos</span> <span class="op">=</span> <span class="ident">get_test_utxos</span>();
1698 <span class="kw">let</span> <span class="ident">database</span> <span class="op">=</span> <span class="ident">MemoryDatabase</span>::<span class="ident">default</span>();
1699
1700 <span class="ident">LargestFirstCoinSelection</span>::<span class="ident">default</span>()
1701 .<span class="ident">coin_select</span>(
1702 <span class="kw-2">&amp;</span><span class="ident">database</span>,
1703 <span class="macro">vec</span><span class="macro">!</span>[],
1704 <span class="ident">utxos</span>,
1705 <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1000.0</span>),
1706 <span class="number">250_000</span>,
1707 <span class="number">50.0</span>,
1708 )
1709 .<span class="ident">unwrap</span>();
1710 }
1711
1712 <span class="attribute">#[<span class="ident">test</span>]</span>
1713 <span class="kw">fn</span> <span class="ident">test_bnb_coin_selection_success</span>() {
1714 <span class="comment">// In this case bnb won&#39;t find a suitable match and single random draw will</span>
1715 <span class="comment">// select three outputs</span>
1716 <span class="kw">let</span> <span class="ident">utxos</span> <span class="op">=</span> <span class="ident">generate_same_value_utxos</span>(<span class="number">100_000</span>, <span class="number">20</span>);
1717
1718 <span class="kw">let</span> <span class="ident">database</span> <span class="op">=</span> <span class="ident">MemoryDatabase</span>::<span class="ident">default</span>();
1719
1720 <span class="kw">let</span> <span class="ident">result</span> <span class="op">=</span> <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">default</span>()
1721 .<span class="ident">coin_select</span>(
1722 <span class="kw-2">&amp;</span><span class="ident">database</span>,
1723 <span class="macro">vec</span><span class="macro">!</span>[],
1724 <span class="ident">utxos</span>,
1725 <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1.0</span>),
1726 <span class="number">250_000</span>,
1727 <span class="number">50.0</span>,
1728 )
1729 .<span class="ident">unwrap</span>();
1730
1731 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected</span>.<span class="ident">len</span>(), <span class="number">3</span>);
1732 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected_amount</span>, <span class="number">300_000</span>);
1733 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">fee_amount</span>, <span class="number">254.0</span>);
1734 }
1735
1736 <span class="attribute">#[<span class="ident">test</span>]</span>
1737 <span class="kw">fn</span> <span class="ident">test_bnb_coin_selection_required_are_enough</span>() {
1738 <span class="kw">let</span> <span class="ident">utxos</span> <span class="op">=</span> <span class="ident">get_test_utxos</span>();
1739 <span class="kw">let</span> <span class="ident">database</span> <span class="op">=</span> <span class="ident">MemoryDatabase</span>::<span class="ident">default</span>();
1740
1741 <span class="kw">let</span> <span class="ident">result</span> <span class="op">=</span> <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">default</span>()
1742 .<span class="ident">coin_select</span>(
1743 <span class="kw-2">&amp;</span><span class="ident">database</span>,
1744 <span class="ident">utxos</span>.<span class="ident">clone</span>(),
1745 <span class="ident">utxos</span>,
1746 <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1.0</span>),
1747 <span class="number">20_000</span>,
1748 <span class="number">50.0</span>,
1749 )
1750 .<span class="ident">unwrap</span>();
1751
1752 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected</span>.<span class="ident">len</span>(), <span class="number">2</span>);
1753 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected_amount</span>, <span class="number">300_000</span>);
1754 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">fee_amount</span>, <span class="number">186.0</span>);
1755 }
1756
1757 <span class="attribute">#[<span class="ident">test</span>]</span>
1758 <span class="attribute">#[<span class="ident">should_panic</span>(<span class="ident">expected</span> <span class="op">=</span> <span class="string">&quot;InsufficientFunds&quot;</span>)]</span>
1759 <span class="kw">fn</span> <span class="ident">test_bnb_coin_selection_insufficient_funds</span>() {
1760 <span class="kw">let</span> <span class="ident">utxos</span> <span class="op">=</span> <span class="ident">get_test_utxos</span>();
1761 <span class="kw">let</span> <span class="ident">database</span> <span class="op">=</span> <span class="ident">MemoryDatabase</span>::<span class="ident">default</span>();
1762
1763 <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">default</span>()
1764 .<span class="ident">coin_select</span>(
1765 <span class="kw-2">&amp;</span><span class="ident">database</span>,
1766 <span class="macro">vec</span><span class="macro">!</span>[],
1767 <span class="ident">utxos</span>,
1768 <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1.0</span>),
1769 <span class="number">500_000</span>,
1770 <span class="number">50.0</span>,
1771 )
1772 .<span class="ident">unwrap</span>();
1773 }
1774
1775 <span class="attribute">#[<span class="ident">test</span>]</span>
1776 <span class="attribute">#[<span class="ident">should_panic</span>(<span class="ident">expected</span> <span class="op">=</span> <span class="string">&quot;InsufficientFunds&quot;</span>)]</span>
1777 <span class="kw">fn</span> <span class="ident">test_bnb_coin_selection_insufficient_funds_high_fees</span>() {
1778 <span class="kw">let</span> <span class="ident">utxos</span> <span class="op">=</span> <span class="ident">get_test_utxos</span>();
1779 <span class="kw">let</span> <span class="ident">database</span> <span class="op">=</span> <span class="ident">MemoryDatabase</span>::<span class="ident">default</span>();
1780
1781 <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">default</span>()
1782 .<span class="ident">coin_select</span>(
1783 <span class="kw-2">&amp;</span><span class="ident">database</span>,
1784 <span class="macro">vec</span><span class="macro">!</span>[],
1785 <span class="ident">utxos</span>,
1786 <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1000.0</span>),
1787 <span class="number">250_000</span>,
1788 <span class="number">50.0</span>,
1789 )
1790 .<span class="ident">unwrap</span>();
1791 }
1792
1793 <span class="attribute">#[<span class="ident">test</span>]</span>
1794 <span class="kw">fn</span> <span class="ident">test_bnb_coin_selection_check_fee_rate</span>() {
1795 <span class="kw">let</span> <span class="ident">utxos</span> <span class="op">=</span> <span class="ident">get_test_utxos</span>();
1796 <span class="kw">let</span> <span class="ident">database</span> <span class="op">=</span> <span class="ident">MemoryDatabase</span>::<span class="ident">default</span>();
1797
1798 <span class="kw">let</span> <span class="ident">result</span> <span class="op">=</span> <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">new</span>(<span class="number">0</span>)
1799 .<span class="ident">coin_select</span>(
1800 <span class="kw-2">&amp;</span><span class="ident">database</span>,
1801 <span class="macro">vec</span><span class="macro">!</span>[],
1802 <span class="ident">utxos</span>.<span class="ident">clone</span>(),
1803 <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1.0</span>),
1804 <span class="number">99932</span>, <span class="comment">// first utxo&#39;s effective value</span>
1805 <span class="number">0.0</span>,
1806 )
1807 .<span class="ident">unwrap</span>();
1808
1809 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected</span>.<span class="ident">len</span>(), <span class="number">1</span>);
1810 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected_amount</span>, <span class="number">100_000</span>);
1811 <span class="kw">let</span> <span class="ident">input_size</span> <span class="op">=</span> (<span class="ident">TXIN_BASE_WEIGHT</span> <span class="kw">as</span> <span class="ident">f32</span>) <span class="op">/</span> <span class="number">4.0</span> <span class="op">+</span> <span class="ident">P2WPKH_WITNESS_SIZE</span> <span class="kw">as</span> <span class="ident">f32</span> <span class="op">/</span> <span class="number">4.0</span>;
1812 <span class="kw">let</span> <span class="ident">epsilon</span> <span class="op">=</span> <span class="number">0.5</span>;
1813 <span class="macro">assert</span><span class="macro">!</span>((<span class="number">1.0</span> <span class="op">-</span> (<span class="ident">result</span>.<span class="ident">fee_amount</span> <span class="op">/</span> <span class="ident">input_size</span>)).<span class="ident">abs</span>() <span class="op">&lt;</span> <span class="ident">epsilon</span>);
1814 }
1815
1816 <span class="attribute">#[<span class="ident">test</span>]</span>
1817 <span class="kw">fn</span> <span class="ident">test_bnb_coin_selection_exact_match</span>() {
1818 <span class="kw">let</span> <span class="ident">seed</span> <span class="op">=</span> [<span class="number">0</span>; <span class="number">32</span>];
1819 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">rng</span>: <span class="ident">StdRng</span> <span class="op">=</span> <span class="ident">SeedableRng</span>::<span class="ident">from_seed</span>(<span class="ident">seed</span>);
1820 <span class="kw">let</span> <span class="ident">database</span> <span class="op">=</span> <span class="ident">MemoryDatabase</span>::<span class="ident">default</span>();
1821
1822 <span class="kw">for</span> <span class="ident">_i</span> <span class="kw">in</span> <span class="number">0</span>..<span class="number">200</span> {
1823 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">optional_utxos</span> <span class="op">=</span> <span class="ident">generate_random_utxos</span>(<span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">rng</span>, <span class="number">16</span>);
1824 <span class="kw">let</span> <span class="ident">target_amount</span> <span class="op">=</span> <span class="ident">sum_random_utxos</span>(<span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">rng</span>, <span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">optional_utxos</span>);
1825 <span class="kw">let</span> <span class="ident">result</span> <span class="op">=</span> <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">new</span>(<span class="number">0</span>)
1826 .<span class="ident">coin_select</span>(
1827 <span class="kw-2">&amp;</span><span class="ident">database</span>,
1828 <span class="macro">vec</span><span class="macro">!</span>[],
1829 <span class="ident">optional_utxos</span>,
1830 <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">0.0</span>),
1831 <span class="ident">target_amount</span>,
1832 <span class="number">0.0</span>,
1833 )
1834 .<span class="ident">unwrap</span>();
1835 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected_amount</span>, <span class="ident">target_amount</span>);
1836 }
1837 }
1838
1839 <span class="attribute">#[<span class="ident">test</span>]</span>
1840 <span class="attribute">#[<span class="ident">should_panic</span>(<span class="ident">expected</span> <span class="op">=</span> <span class="string">&quot;BnBNoExactMatch&quot;</span>)]</span>
1841 <span class="kw">fn</span> <span class="ident">test_bnb_function_no_exact_match</span>() {
1842 <span class="kw">let</span> <span class="ident">fee_rate</span> <span class="op">=</span> <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">10.0</span>);
1843 <span class="kw">let</span> <span class="ident">utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">OutputGroup</span><span class="op">&gt;</span> <span class="op">=</span> <span class="ident">get_test_utxos</span>()
1844 .<span class="ident">into_iter</span>()
1845 .<span class="ident">map</span>(<span class="op">|</span><span class="ident">u</span><span class="op">|</span> <span class="ident">OutputGroup</span>::<span class="ident">new</span>(<span class="ident">u</span>.<span class="number">0</span>, <span class="ident">u</span>.<span class="number">1</span>, <span class="ident">fee_rate</span>))
1846 .<span class="ident">collect</span>();
1847
1848 <span class="kw">let</span> <span class="ident">curr_available_value</span> <span class="op">=</span> <span class="ident">utxos</span>
1849 .<span class="ident">iter</span>()
1850 .<span class="ident">fold</span>(<span class="number">0</span>, <span class="op">|</span><span class="ident">acc</span>, <span class="ident">x</span><span class="op">|</span> <span class="ident">acc</span> <span class="op">+</span> <span class="ident">x</span>.<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>);
1851
1852 <span class="kw">let</span> <span class="ident">size_of_change</span> <span class="op">=</span> <span class="number">31</span>;
1853 <span class="kw">let</span> <span class="ident">cost_of_change</span> <span class="op">=</span> <span class="ident">size_of_change</span> <span class="kw">as</span> <span class="ident">f32</span> <span class="op">*</span> <span class="ident">fee_rate</span>.<span class="ident">as_sat_vb</span>();
1854 <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">new</span>(<span class="ident">size_of_change</span>)
1855 .<span class="ident">bnb</span>(
1856 <span class="macro">vec</span><span class="macro">!</span>[],
1857 <span class="ident">utxos</span>,
1858 <span class="number">0</span>,
1859 <span class="ident">curr_available_value</span>,
1860 <span class="number">20_000</span>,
1861 <span class="number">50.0</span>,
1862 <span class="ident">cost_of_change</span>,
1863 )
1864 .<span class="ident">unwrap</span>();
1865 }
1866
1867 <span class="attribute">#[<span class="ident">test</span>]</span>
1868 <span class="attribute">#[<span class="ident">should_panic</span>(<span class="ident">expected</span> <span class="op">=</span> <span class="string">&quot;BnBTotalTriesExceeded&quot;</span>)]</span>
1869 <span class="kw">fn</span> <span class="ident">test_bnb_function_tries_exceeded</span>() {
1870 <span class="kw">let</span> <span class="ident">fee_rate</span> <span class="op">=</span> <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">10.0</span>);
1871 <span class="kw">let</span> <span class="ident">utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">OutputGroup</span><span class="op">&gt;</span> <span class="op">=</span> <span class="ident">generate_same_value_utxos</span>(<span class="number">100_000</span>, <span class="number">100_000</span>)
1872 .<span class="ident">into_iter</span>()
1873 .<span class="ident">map</span>(<span class="op">|</span><span class="ident">u</span><span class="op">|</span> <span class="ident">OutputGroup</span>::<span class="ident">new</span>(<span class="ident">u</span>.<span class="number">0</span>, <span class="ident">u</span>.<span class="number">1</span>, <span class="ident">fee_rate</span>))
1874 .<span class="ident">collect</span>();
1875
1876 <span class="kw">let</span> <span class="ident">curr_available_value</span> <span class="op">=</span> <span class="ident">utxos</span>
1877 .<span class="ident">iter</span>()
1878 .<span class="ident">fold</span>(<span class="number">0</span>, <span class="op">|</span><span class="ident">acc</span>, <span class="ident">x</span><span class="op">|</span> <span class="ident">acc</span> <span class="op">+</span> <span class="ident">x</span>.<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>);
1879
1880 <span class="kw">let</span> <span class="ident">size_of_change</span> <span class="op">=</span> <span class="number">31</span>;
1881 <span class="kw">let</span> <span class="ident">cost_of_change</span> <span class="op">=</span> <span class="ident">size_of_change</span> <span class="kw">as</span> <span class="ident">f32</span> <span class="op">*</span> <span class="ident">fee_rate</span>.<span class="ident">as_sat_vb</span>();
1882
1883 <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">new</span>(<span class="ident">size_of_change</span>)
1884 .<span class="ident">bnb</span>(
1885 <span class="macro">vec</span><span class="macro">!</span>[],
1886 <span class="ident">utxos</span>,
1887 <span class="number">0</span>,
1888 <span class="ident">curr_available_value</span>,
1889 <span class="number">20_000</span>,
1890 <span class="number">50.0</span>,
1891 <span class="ident">cost_of_change</span>,
1892 )
1893 .<span class="ident">unwrap</span>();
1894 }
1895
1896 <span class="comment">// The match won&#39;t be exact but still in the range</span>
1897 <span class="attribute">#[<span class="ident">test</span>]</span>
1898 <span class="kw">fn</span> <span class="ident">test_bnb_function_almost_exact_match_with_fees</span>() {
1899 <span class="kw">let</span> <span class="ident">fee_rate</span> <span class="op">=</span> <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1.0</span>);
1900 <span class="kw">let</span> <span class="ident">size_of_change</span> <span class="op">=</span> <span class="number">31</span>;
1901 <span class="kw">let</span> <span class="ident">cost_of_change</span> <span class="op">=</span> <span class="ident">size_of_change</span> <span class="kw">as</span> <span class="ident">f32</span> <span class="op">*</span> <span class="ident">fee_rate</span>.<span class="ident">as_sat_vb</span>();
1902 <span class="kw">let</span> <span class="ident">fee_amount</span> <span class="op">=</span> <span class="number">50.0</span>;
1903
1904 <span class="kw">let</span> <span class="ident">utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="kw">_</span><span class="op">&gt;</span> <span class="op">=</span> <span class="ident">generate_same_value_utxos</span>(<span class="number">50_000</span>, <span class="number">10</span>)
1905 .<span class="ident">into_iter</span>()
1906 .<span class="ident">map</span>(<span class="op">|</span><span class="ident">u</span><span class="op">|</span> <span class="ident">OutputGroup</span>::<span class="ident">new</span>(<span class="ident">u</span>.<span class="number">0</span>, <span class="ident">u</span>.<span class="number">1</span>, <span class="ident">fee_rate</span>))
1907 .<span class="ident">collect</span>();
1908
1909 <span class="kw">let</span> <span class="ident">curr_value</span> <span class="op">=</span> <span class="number">0</span>;
1910
1911 <span class="kw">let</span> <span class="ident">curr_available_value</span> <span class="op">=</span> <span class="ident">utxos</span>
1912 .<span class="ident">iter</span>()
1913 .<span class="ident">fold</span>(<span class="number">0</span>, <span class="op">|</span><span class="ident">acc</span>, <span class="ident">x</span><span class="op">|</span> <span class="ident">acc</span> <span class="op">+</span> <span class="ident">x</span>.<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>);
1914
1915 <span class="comment">// 2*(value of 1 utxo) - 2*(1 utxo fees with 1.0sat/vbyte fee rate) -</span>
1916 <span class="comment">// cost_of_change + 5.</span>
1917 <span class="kw">let</span> <span class="ident">target_amount</span> <span class="op">=</span> <span class="number">2</span> <span class="op">*</span> <span class="number">50_000</span> <span class="op">-</span> <span class="number">2</span> <span class="op">*</span> <span class="number">67</span> <span class="op">-</span> <span class="ident">cost_of_change</span>.<span class="ident">ceil</span>() <span class="kw">as</span> <span class="ident">u64</span> <span class="op">+</span> <span class="number">5</span>;
1918
1919 <span class="kw">let</span> <span class="ident">result</span> <span class="op">=</span> <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">new</span>(<span class="ident">size_of_change</span>)
1920 .<span class="ident">bnb</span>(
1921 <span class="macro">vec</span><span class="macro">!</span>[],
1922 <span class="ident">utxos</span>,
1923 <span class="ident">curr_value</span>,
1924 <span class="ident">curr_available_value</span>,
1925 <span class="ident">target_amount</span>,
1926 <span class="ident">fee_amount</span>,
1927 <span class="ident">cost_of_change</span>,
1928 )
1929 .<span class="ident">unwrap</span>();
1930 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">fee_amount</span>, <span class="number">186.0</span>);
1931 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected_amount</span>, <span class="number">100_000</span>);
1932 }
1933
1934 <span class="comment">// TODO: bnb() function should be optimized, and this test should be done with more utxos</span>
1935 <span class="attribute">#[<span class="ident">test</span>]</span>
1936 <span class="kw">fn</span> <span class="ident">test_bnb_function_exact_match_more_utxos</span>() {
1937 <span class="kw">let</span> <span class="ident">seed</span> <span class="op">=</span> [<span class="number">0</span>; <span class="number">32</span>];
1938 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">rng</span>: <span class="ident">StdRng</span> <span class="op">=</span> <span class="ident">SeedableRng</span>::<span class="ident">from_seed</span>(<span class="ident">seed</span>);
1939 <span class="kw">let</span> <span class="ident">fee_rate</span> <span class="op">=</span> <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">0.0</span>);
1940
1941 <span class="kw">for</span> <span class="kw">_</span> <span class="kw">in</span> <span class="number">0</span>..<span class="number">200</span> {
1942 <span class="kw">let</span> <span class="ident">optional_utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="kw">_</span><span class="op">&gt;</span> <span class="op">=</span> <span class="ident">generate_random_utxos</span>(<span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">rng</span>, <span class="number">40</span>)
1943 .<span class="ident">into_iter</span>()
1944 .<span class="ident">map</span>(<span class="op">|</span><span class="ident">u</span><span class="op">|</span> <span class="ident">OutputGroup</span>::<span class="ident">new</span>(<span class="ident">u</span>.<span class="number">0</span>, <span class="ident">u</span>.<span class="number">1</span>, <span class="ident">fee_rate</span>))
1945 .<span class="ident">collect</span>();
1946
1947 <span class="kw">let</span> <span class="ident">curr_value</span> <span class="op">=</span> <span class="number">0</span>;
1948
1949 <span class="kw">let</span> <span class="ident">curr_available_value</span> <span class="op">=</span> <span class="ident">optional_utxos</span>
1950 .<span class="ident">iter</span>()
1951 .<span class="ident">fold</span>(<span class="number">0</span>, <span class="op">|</span><span class="ident">acc</span>, <span class="ident">x</span><span class="op">|</span> <span class="ident">acc</span> <span class="op">+</span> <span class="ident">x</span>.<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>);
1952
1953 <span class="kw">let</span> <span class="ident">target_amount</span> <span class="op">=</span> <span class="ident">optional_utxos</span>[<span class="number">3</span>].<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>
1954 <span class="op">+</span> <span class="ident">optional_utxos</span>[<span class="number">23</span>].<span class="ident">effective_value</span> <span class="kw">as</span> <span class="ident">u64</span>;
1955
1956 <span class="kw">let</span> <span class="ident">result</span> <span class="op">=</span> <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">new</span>(<span class="number">0</span>)
1957 .<span class="ident">bnb</span>(
1958 <span class="macro">vec</span><span class="macro">!</span>[],
1959 <span class="ident">optional_utxos</span>,
1960 <span class="ident">curr_value</span>,
1961 <span class="ident">curr_available_value</span>,
1962 <span class="ident">target_amount</span>,
1963 <span class="number">0.0</span>,
1964 <span class="number">0.0</span>,
1965 )
1966 .<span class="ident">unwrap</span>();
1967 <span class="macro">assert_eq</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected_amount</span>, <span class="ident">target_amount</span>);
1968 }
1969 }
1970
1971 <span class="attribute">#[<span class="ident">test</span>]</span>
1972 <span class="kw">fn</span> <span class="ident">test_single_random_draw_function_success</span>() {
1973 <span class="kw">let</span> <span class="ident">seed</span> <span class="op">=</span> [<span class="number">0</span>; <span class="number">32</span>];
1974 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">rng</span>: <span class="ident">StdRng</span> <span class="op">=</span> <span class="ident">SeedableRng</span>::<span class="ident">from_seed</span>(<span class="ident">seed</span>);
1975 <span class="kw">let</span> <span class="kw-2">mut</span> <span class="ident">utxos</span> <span class="op">=</span> <span class="ident">generate_random_utxos</span>(<span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">rng</span>, <span class="number">300</span>);
1976 <span class="kw">let</span> <span class="ident">target_amount</span> <span class="op">=</span> <span class="ident">sum_random_utxos</span>(<span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">rng</span>, <span class="kw-2">&amp;</span><span class="kw-2">mut</span> <span class="ident">utxos</span>);
1977
1978 <span class="kw">let</span> <span class="ident">fee_rate</span> <span class="op">=</span> <span class="ident">FeeRate</span>::<span class="ident">from_sat_per_vb</span>(<span class="number">1.0</span>);
1979 <span class="kw">let</span> <span class="ident">utxos</span>: <span class="ident">Vec</span><span class="op">&lt;</span><span class="ident">OutputGroup</span><span class="op">&gt;</span> <span class="op">=</span> <span class="ident">utxos</span>
1980 .<span class="ident">into_iter</span>()
1981 .<span class="ident">map</span>(<span class="op">|</span><span class="ident">u</span><span class="op">|</span> <span class="ident">OutputGroup</span>::<span class="ident">new</span>(<span class="ident">u</span>.<span class="number">0</span>, <span class="ident">u</span>.<span class="number">1</span>, <span class="ident">fee_rate</span>))
1982 .<span class="ident">collect</span>();
1983
1984 <span class="kw">let</span> <span class="ident">result</span> <span class="op">=</span> <span class="ident">BranchAndBoundCoinSelection</span>::<span class="ident">default</span>().<span class="ident">single_random_draw</span>(
1985 <span class="macro">vec</span><span class="macro">!</span>[],
1986 <span class="ident">utxos</span>,
1987 <span class="number">0</span>,
1988 <span class="ident">target_amount</span>,
1989 <span class="number">50.0</span>,
1990 );
1991
1992 <span class="macro">assert</span><span class="macro">!</span>(<span class="ident">result</span>.<span class="ident">selected_amount</span> <span class="op">&gt;</span> <span class="ident">target_amount</span>);
1993 <span class="macro">assert_eq</span><span class="macro">!</span>(
1994 <span class="ident">result</span>.<span class="ident">fee_amount</span>,
1995 <span class="number">50.0</span> <span class="op">+</span> <span class="ident">result</span>.<span class="ident">selected</span>.<span class="ident">len</span>() <span class="kw">as</span> <span class="ident">f32</span> <span class="op">*</span> <span class="number">68.0</span>
1996 );
1997 }
1998 }
1999 </pre></div>
2000 </section><section id="search" class="content hidden"></section><section class="footer"></section><script>window.rootPath = "../../../";window.currentCrate = "bdk";</script><script src="../../../main.js"></script><script src="../../../source-script.js"></script><script src="../../../source-files.js"></script><script defer src="../../../search-index.js"></script></body></html>