@@ -23,6 +23,7 @@ use super::packet::{BIG_PACKET_HOP_DATA_LEN, ForwardControlTlvs, Packet, Payload
23
23
use super :: utils;
24
24
use util:: events:: OnionMessageProvider ;
25
25
use util:: logger:: Logger ;
26
+ use util:: ser:: Writeable ;
26
27
27
28
use core:: ops:: Deref ;
28
29
use sync:: { Arc , Mutex } ;
@@ -122,6 +123,8 @@ pub enum SendError {
122
123
TooFewBlindedHops ,
123
124
/// Our next-hop peer was offline or does not support onion message forwarding.
124
125
InvalidFirstHop ,
126
+ /// Our next-hop peer's buffer was full or our total outbound buffer was full.
127
+ BufferFull ,
125
128
}
126
129
127
130
impl < Signer : Sign , K : Deref , L : Deref > OnionMessenger < Signer , K , L >
@@ -169,6 +172,7 @@ impl<Signer: Sign, K: Deref, L: Deref> OnionMessenger<Signer, K, L>
169
172
packet_payloads, packet_keys, prng_seed) . map_err ( |( ) | SendError :: TooBigPacket ) ?;
170
173
171
174
let mut pending_per_peer_msgs = self . pending_messages . lock ( ) . unwrap ( ) ;
175
+ if outbound_buffer_full ( & introduction_node_id, & pending_per_peer_msgs) { return Err ( SendError :: BufferFull ) }
172
176
match pending_per_peer_msgs. entry ( introduction_node_id) {
173
177
hash_map:: Entry :: Vacant ( _) => Err ( SendError :: InvalidFirstHop ) ,
174
178
hash_map:: Entry :: Occupied ( mut e) => {
@@ -192,6 +196,29 @@ impl<Signer: Sign, K: Deref, L: Deref> OnionMessenger<Signer, K, L>
192
196
}
193
197
}
194
198
199
+ fn outbound_buffer_full ( peer_node_id : & PublicKey , buffer : & HashMap < PublicKey , VecDeque < msgs:: OnionMessage > > ) -> bool {
200
+ const MAX_TOTAL_BUFFER_SIZE : usize = ( 2 << 20 ) * 128 ;
201
+ const MAX_PER_PEER_BUFFER_SIZE : usize = ( 2 << 10 ) * 256 ;
202
+ let mut total_buffered_bytes = 0 ;
203
+ let mut peer_buffered_bytes = 0 ;
204
+ for ( pk, peer_buf) in buffer {
205
+ for om in peer_buf {
206
+ let om_len = om. serialized_length ( ) ;
207
+ if pk == peer_node_id {
208
+ peer_buffered_bytes += om_len;
209
+ }
210
+ total_buffered_bytes += om_len;
211
+
212
+ if total_buffered_bytes >= MAX_TOTAL_BUFFER_SIZE ||
213
+ peer_buffered_bytes >= MAX_PER_PEER_BUFFER_SIZE
214
+ {
215
+ return true
216
+ }
217
+ }
218
+ }
219
+ false
220
+ }
221
+
195
222
impl < Signer : Sign , K : Deref , L : Deref > OnionMessageHandler for OnionMessenger < Signer , K , L >
196
223
where K :: Target : KeysInterface < Signer = Signer > ,
197
224
L :: Target : Logger ,
@@ -278,6 +305,10 @@ impl<Signer: Sign, K: Deref, L: Deref> OnionMessageHandler for OnionMessenger<Si
278
305
} ;
279
306
280
307
let mut pending_per_peer_msgs = self . pending_messages . lock ( ) . unwrap ( ) ;
308
+ if outbound_buffer_full ( & next_node_id, & pending_per_peer_msgs) {
309
+ log_trace ! ( self . logger, "Dropping forwarded onion message to peer {:?}: outbound buffer full" , next_node_id) ;
310
+ return
311
+ }
281
312
282
313
#[ cfg( fuzzing) ]
283
314
pending_per_peer_msgs. entry ( next_node_id) . or_insert_with ( || VecDeque :: new ( ) ) ;
0 commit comments