@@ -25,7 +25,7 @@ use crate::{
25
25
utp:: {
26
26
packets:: { ExtensionType , Packet , PacketType , HEADER_SIZE } ,
27
27
time:: { now_microseconds, Delay , Timestamp } ,
28
- trin_helpers:: { UtpMessage , UtpMessageId } ,
28
+ trin_helpers:: { UtpMessage , UtpStreamId } ,
29
29
util:: { abs_diff, ewma, generate_sequential_identifiers} ,
30
30
} ,
31
31
} ;
@@ -111,6 +111,8 @@ pub enum UtpListenerRequest {
111
111
FindContentData ( ConnId , ByteList ) ,
112
112
/// Request to listen for FindContent stream
113
113
FindContentStream ( ConnId ) ,
114
+ /// Process all streams where uTP socket state is "Closed"
115
+ ProcessClosedStreams ( oneshot:: Sender < Vec < ( UtpSocket , UtpStreamId ) > > ) ,
114
116
/// Request to listen for Offer stream
115
117
OfferStream ( ConnId ) ,
116
118
}
@@ -122,7 +124,7 @@ pub struct UtpListener {
122
124
/// Store all active connections
123
125
utp_connections : HashMap < ConnectionKey , UtpSocket > ,
124
126
/// uTP connection ids to listen for
125
- listening : HashMap < ConnId , UtpMessageId > ,
127
+ listening : HashMap < ConnId , UtpStreamId > ,
126
128
/// Receiver for uTP events sent from the main portal event handler
127
129
utp_event_rx : UnboundedReceiver < TalkRequest > ,
128
130
/// Receiver for uTP requests sent from the overlay layer
@@ -221,9 +223,9 @@ impl UtpListener {
221
223
// TODO: Probably there is a better way with lifetimes to pass the HashMap value to a
222
224
// different thread without removing the key and re-adding it.
223
225
self . listening
224
- . insert ( conn. sender_connection_id , UtpMessageId :: FindContentStream ) ;
226
+ . insert ( conn. sender_connection_id , UtpStreamId :: FindContentStream ) ;
225
227
226
- if let Some ( UtpMessageId :: FindContentData ( Content ( content_data) ) ) =
228
+ if let Some ( UtpStreamId :: FindContentData ( Content ( content_data) ) ) =
227
229
utp_message_id
228
230
{
229
231
// We want to send uTP data only if the content is Content(ByteList)
@@ -264,12 +266,17 @@ impl UtpListener {
264
266
return ;
265
267
}
266
268
269
+ let mut result = Vec :: new ( ) ;
270
+
267
271
let mut buf = [ 0 ; BUF_SIZE ] ;
268
- if let Err ( msg) = conn. recv ( & mut buf) . await {
269
- error ! ( "Unable to receive uTP DATA packet: {msg}" )
270
- } else {
271
- conn. recv_data_stream
272
- . append ( & mut Vec :: from ( packet. payload ( ) ) ) ;
272
+ match conn. recv ( & mut buf) . await {
273
+ Ok ( bytes_read) => {
274
+ if let Some ( bytes) = bytes_read {
275
+ result. extend_from_slice ( & buf[ ..bytes] ) ;
276
+ conn. recv_data_stream . append ( & mut result) ;
277
+ }
278
+ }
279
+ Err ( err) => error ! ( "Unable to receive uTP DATA packet: {err}" ) ,
273
280
}
274
281
}
275
282
}
@@ -314,24 +321,29 @@ impl UtpListener {
314
321
match request {
315
322
UtpListenerRequest :: FindContentStream ( conn_id) => {
316
323
self . listening
317
- . insert ( conn_id, UtpMessageId :: FindContentStream ) ;
324
+ . insert ( conn_id, UtpStreamId :: FindContentStream ) ;
318
325
}
319
326
UtpListenerRequest :: Connect ( conn_id, node_id, tx) => {
320
327
let conn = self . connect ( conn_id, node_id) . await ;
321
328
if tx. send ( conn) . is_err ( ) {
322
- warn ! ( "Unable to send uTP socket to requester" )
329
+ error ! ( "Unable to send uTP socket to requester" )
323
330
} ;
324
331
}
325
332
UtpListenerRequest :: OfferStream ( conn_id) => {
326
- self . listening . insert ( conn_id, UtpMessageId :: OfferStream ) ;
333
+ self . listening . insert ( conn_id, UtpStreamId :: OfferStream ) ;
327
334
}
328
335
UtpListenerRequest :: FindContentData ( conn_id, content) => {
329
336
self . listening
330
- . insert ( conn_id, UtpMessageId :: FindContentData ( Content ( content) ) ) ;
337
+ . insert ( conn_id, UtpStreamId :: FindContentData ( Content ( content) ) ) ;
331
338
}
332
339
UtpListenerRequest :: AcceptStream ( conn_id, accepted_keys) => {
333
340
self . listening
334
- . insert ( conn_id, UtpMessageId :: AcceptStream ( accepted_keys) ) ;
341
+ . insert ( conn_id, UtpStreamId :: AcceptStream ( accepted_keys) ) ;
342
+ }
343
+ UtpListenerRequest :: ProcessClosedStreams ( tx) => {
344
+ if tx. send ( self . process_closed_streams ( ) ) . is_err ( ) {
345
+ error ! ( "Unable to send closed uTP streams to requester" )
346
+ } ;
335
347
}
336
348
}
337
349
}
@@ -355,28 +367,32 @@ impl UtpListener {
355
367
}
356
368
}
357
369
358
- // https://github.com/ethereum/portal-network-specs/pull/98\
359
- // Currently the way to handle data over uTP isn't finalized yet, so we are going to use the
360
- // handle data on connection closed method, as that seems to be the accepted method for now.
361
- pub async fn process_utp_byte_stream ( & mut self ) {
362
- let mut utp_connections = self . utp_connections . clone ( ) ;
363
- for ( conn_key, conn) in self . utp_connections . iter_mut ( ) {
364
- if conn. state == SocketState :: Closed {
365
- let received_stream = conn. recv_data_stream . clone ( ) ;
366
- debug ! ( "Received data: with len: {}" , received_stream. len( ) ) ;
367
-
368
- match self . listening . get ( & conn. receiver_connection_id ) {
369
- Some ( message_type) => {
370
- if let UtpMessageId :: AcceptStream ( content_keys) = message_type {
371
- // TODO: Implement this with overlay store and decode receiver stream if multiple content values are send
372
- debug ! ( "Store {content_keys:?}, {received_stream:?}" ) ;
373
- }
374
- }
375
- _ => warn ! ( "uTP listening HashMap doesn't have uTP stream message type" ) ,
376
- }
377
- utp_connections. remove ( conn_key) ;
378
- }
379
- }
370
+ /// Return and cleanup all active uTP streams where socket state is "Closed"
371
+ pub fn process_closed_streams ( & mut self ) -> Vec < ( UtpSocket , UtpStreamId ) > {
372
+ // This seems to be a hot loop, we may need to optimise it and find a better way to filter by closed
373
+ // connections without cloning all records. One reasonable way is to use some data-oriented
374
+ // design principles like Struct of Arrays vs. Array of Structs.
375
+ self . utp_connections
376
+ . clone ( )
377
+ . iter ( )
378
+ . filter ( |conn| conn. 1 . state == SocketState :: Closed )
379
+ . map ( |conn| {
380
+ // Remove the closed connections from active connections
381
+ let receiver_stream_id = self
382
+ . listening
383
+ . remove ( & conn. 1 . receiver_connection_id )
384
+ . expect ( "Receiver connection id should match active listening connections." ) ;
385
+ self . listening
386
+ . remove ( & conn. 1 . sender_connection_id )
387
+ . expect ( "Sender connection id should match active listening connections." ) ;
388
+ let utp_socket = self
389
+ . utp_connections
390
+ . remove ( conn. 0 )
391
+ . expect ( "uTP socket should match asctive utp connections." ) ;
392
+
393
+ ( utp_socket, receiver_stream_id)
394
+ } )
395
+ . collect ( )
380
396
}
381
397
}
382
398
0 commit comments