@@ -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.
127
+ PeerBufferFull ,
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 ( & pending_per_peer_msgs) { return Err ( SendError :: PeerBufferFull ) }
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,20 @@ impl<Signer: Sign, K: Deref, L: Deref> OnionMessenger<Signer, K, L>
192
196
}
193
197
}
194
198
199
+ fn outbound_buffer_full ( buffer : & HashMap < PublicKey , VecDeque < msgs:: OnionMessage > > ) -> bool {
200
+ const MAX_BUFFER_SIZE : usize = 262144 ; // 256KiB
201
+ let mut buffered_bytes = 0 ;
202
+ for peer_buf in buffer. values ( ) {
203
+ for om in peer_buf {
204
+ buffered_bytes += om. serialized_length ( ) ;
205
+ if buffered_bytes >= MAX_BUFFER_SIZE {
206
+ return true
207
+ }
208
+ }
209
+ }
210
+ false
211
+ }
212
+
195
213
impl < Signer : Sign , K : Deref , L : Deref > OnionMessageHandler for OnionMessenger < Signer , K , L >
196
214
where K :: Target : KeysInterface < Signer = Signer > ,
197
215
L :: Target : Logger ,
@@ -260,6 +278,10 @@ impl<Signer: Sign, K: Deref, L: Deref> OnionMessageHandler for OnionMessenger<Si
260
278
#[ cfg( fuzzing) ]
261
279
pending_per_peer_msgs. entry ( next_node_id) . or_insert_with ( || VecDeque :: new ( ) ) ;
262
280
281
+ if outbound_buffer_full ( & pending_per_peer_msgs) {
282
+ log_trace ! ( self . logger, "Dropping forwarded onion message to peer {:?}: outbound buffer full" , next_node_id) ;
283
+ return
284
+ }
263
285
match pending_per_peer_msgs. entry ( next_node_id) {
264
286
hash_map:: Entry :: Vacant ( _) => {
265
287
log_trace ! ( self . logger, "Dropping forwarded onion message to disconnected peer {:?}" , next_node_id) ;
0 commit comments