| | 1 | | using NBitcoin; |
| | 2 | |
|
| | 3 | | namespace NLightning.Infrastructure.Bitcoin.Services; |
| | 4 | |
|
| | 5 | | using Domain.Bitcoin.Interfaces; |
| | 6 | | using Domain.Protocol.Services; |
| | 7 | |
|
| | 8 | | public class DustService : IDustService |
| | 9 | | { |
| | 10 | | private readonly IFeeService _feeService; |
| | 11 | |
|
| 0 | 12 | | public DustService(IFeeService feeService) |
| | 13 | | { |
| 0 | 14 | | _feeService = feeService; |
| 0 | 15 | | } |
| | 16 | |
|
| | 17 | | public ulong CalculateP2PkhDustLimit() |
| | 18 | | { |
| | 19 | | const uint outputSize = 34; |
| | 20 | | const uint inputSize = 148; |
| 0 | 21 | | return CalculateDustLimit(outputSize, inputSize); |
| | 22 | | } |
| | 23 | |
|
| | 24 | | public ulong CalculateP2ShDustLimit() |
| | 25 | | { |
| | 26 | | const uint outputSize = 32; |
| | 27 | | const uint inputSize = 148; // Lower bound |
| 0 | 28 | | return CalculateDustLimit(outputSize, inputSize); |
| | 29 | | } |
| | 30 | |
|
| | 31 | | public ulong CalculateP2WpkhDustLimit() |
| | 32 | | { |
| | 33 | | const uint outputSize = 31; |
| | 34 | | const uint inputSize = 67; // Lower bound |
| 0 | 35 | | return CalculateDustLimit(outputSize, inputSize); |
| | 36 | | } |
| | 37 | |
|
| | 38 | | public ulong CalculateP2WshDustLimit() |
| | 39 | | { |
| | 40 | | const uint outputSize = 43; |
| | 41 | | const uint inputSize = 67; // Lower bound |
| 0 | 42 | | return CalculateDustLimit(outputSize, inputSize); |
| | 43 | | } |
| | 44 | |
|
| | 45 | | public ulong CalculateUnknownSegwitVersionDustLimit() |
| | 46 | | { |
| | 47 | | const uint outputSize = 51; |
| | 48 | | const uint inputSize = 67; // Lower bound |
| 0 | 49 | | return CalculateDustLimit(outputSize, inputSize); |
| | 50 | | } |
| | 51 | |
|
| | 52 | | private ulong CalculateDustLimit(uint outputSize, uint inputSize) |
| | 53 | | { |
| 0 | 54 | | var totalSize = outputSize + inputSize; |
| 0 | 55 | | return (totalSize * _feeService.GetCachedFeeRatePerKw() / 1000); |
| | 56 | | } |
| | 57 | |
|
| | 58 | | public bool IsDust(ulong amount, Script scriptPubKey) |
| | 59 | | { |
| 0 | 60 | | if (scriptPubKey.IsScriptType(ScriptType.P2PKH)) |
| | 61 | | { |
| 0 | 62 | | return amount < CalculateP2PkhDustLimit(); |
| | 63 | | } |
| | 64 | |
|
| 0 | 65 | | if (scriptPubKey.IsScriptType(ScriptType.P2SH)) |
| | 66 | | { |
| 0 | 67 | | return amount < CalculateP2ShDustLimit(); |
| | 68 | | } |
| | 69 | |
|
| 0 | 70 | | if (scriptPubKey.IsScriptType(ScriptType.P2WPKH)) |
| | 71 | | { |
| 0 | 72 | | return amount < CalculateP2WpkhDustLimit(); |
| | 73 | | } |
| | 74 | |
|
| 0 | 75 | | if (scriptPubKey.IsScriptType(ScriptType.P2WSH)) |
| | 76 | | { |
| 0 | 77 | | return amount < CalculateP2WshDustLimit(); |
| | 78 | | } |
| | 79 | |
|
| 0 | 80 | | if (scriptPubKey.ToBytes()[0] == (byte)OpcodeType.OP_RETURN) |
| | 81 | | { |
| 0 | 82 | | return false; // OP_RETURN outputs are never dust |
| | 83 | | } |
| | 84 | |
|
| 0 | 85 | | return amount < CalculateUnknownSegwitVersionDustLimit(); |
| | 86 | | } |
| | 87 | | } |