last connection work

This commit is contained in:
John Smith 2022-08-06 12:36:07 -04:00
parent e7ae6b04bd
commit 5e1176db5a
5 changed files with 52 additions and 19 deletions

View File

@ -806,7 +806,7 @@ impl NetworkManager {
// Do our half of the hole punch by sending an empty packet // Do our half of the hole punch by sending an empty packet
// Both sides will do this and then the receipt will get sent over the punched hole // Both sides will do this and then the receipt will get sent over the punched hole
network_result_try!( let connection_descriptor = network_result_try!(
self.net() self.net()
.send_data_to_dial_info( .send_data_to_dial_info(
hole_punch_dial_info_detail.dial_info.clone(), hole_punch_dial_info_detail.dial_info.clone(),
@ -817,6 +817,9 @@ impl NetworkManager {
// XXX: do we need a delay here? or another hole punch packet? // XXX: do we need a delay here? or another hole punch packet?
// Set the hole punch as our 'last connection' to ensure we return the receipt over the direct hole punch
peer_nr.set_last_connection(connection_descriptor, intf::get_timestamp());
// Return the receipt using the same dial info send the receipt to it // Return the receipt using the same dial info send the receipt to it
rpc.rpc_call_return_receipt(Destination::Direct(peer_nr), None, receipt) rpc.rpc_call_return_receipt(Destination::Direct(peer_nr), None, receipt)
.await .await
@ -1178,6 +1181,8 @@ impl NetworkManager {
// Do our half of the hole punch by sending an empty packet // Do our half of the hole punch by sending an empty packet
// Both sides will do this and then the receipt will get sent over the punched hole // Both sides will do this and then the receipt will get sent over the punched hole
// Don't bother storing the returned connection descriptor as the 'last connection' because the other side of the hole
// punch should come through and create a real 'last connection' for us if this succeeds
network_result_try!( network_result_try!(
self.net() self.net()
.send_data_to_dial_info(hole_punch_did.dial_info, Vec::new()) .send_data_to_dial_info(hole_punch_did.dial_info, Vec::new())
@ -1294,7 +1299,12 @@ impl NetworkManager {
} else { } else {
SendDataKind::GlobalDirect SendDataKind::GlobalDirect
}; };
network_result_try!(this.net().send_data_to_dial_info(dial_info, data).await?); let connection_descriptor = network_result_try!(
this.net().send_data_to_dial_info(dial_info, data).await?
);
// If we connected to this node directly, save off the last connection so we can use it again
node_ref.set_last_connection(connection_descriptor, intf::get_timestamp());
Ok(NetworkResult::value(send_data_kind)) Ok(NetworkResult::value(send_data_kind))
} }
ContactMethod::SignalReverse(relay_nr, target_node_ref) => { ContactMethod::SignalReverse(relay_nr, target_node_ref) => {

View File

@ -493,13 +493,15 @@ impl Network {
} }
// Send data directly to a dial info, possibly without knowing which node it is going to // Send data directly to a dial info, possibly without knowing which node it is going to
// Returns a descriptor for the connection used to send the data
#[instrument(level="trace", err, skip(self, data), fields(data.len = data.len()))] #[instrument(level="trace", err, skip(self, data), fields(data.len = data.len()))]
pub async fn send_data_to_dial_info( pub async fn send_data_to_dial_info(
&self, &self,
dial_info: DialInfo, dial_info: DialInfo,
data: Vec<u8>, data: Vec<u8>,
) -> EyreResult<NetworkResult<()>> { ) -> EyreResult<NetworkResult<ConnectionDescriptor>> {
let data_len = data.len(); let data_len = data.len();
let connection_descriptor;
if dial_info.protocol_type() == ProtocolType::UDP { if dial_info.protocol_type() == ProtocolType::UDP {
// Handle connectionless protocol // Handle connectionless protocol
let peer_socket_addr = dial_info.to_socket_addr(); let peer_socket_addr = dial_info.to_socket_addr();
@ -507,10 +509,9 @@ impl Network {
Some(ph) => ph, Some(ph) => ph,
None => bail!("no appropriate UDP protocol handler for dial_info"), None => bail!("no appropriate UDP protocol handler for dial_info"),
}; };
let _ = network_result_try!(ph connection_descriptor = network_result_try!(ph
.send_message(data, peer_socket_addr) .send_message(data, peer_socket_addr)
.await .await
.into_network_result()
.wrap_err("failed to send data to dial info")?); .wrap_err("failed to send data to dial info")?);
} else { } else {
// Handle connection-oriented protocols // Handle connection-oriented protocols
@ -527,13 +528,14 @@ impl Network {
"failed to send", "failed to send",
))); )));
} }
connection_descriptor = conn.connection_descriptor();
} }
// Network accounting // Network accounting
self.network_manager() self.network_manager()
.stats_packet_sent(dial_info.to_ip_addr(), data_len as u64); .stats_packet_sent(dial_info.to_ip_addr(), data_len as u64);
Ok(NetworkResult::value(())) Ok(NetworkResult::value(connection_descriptor))
} }
///////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////

View File

@ -42,7 +42,7 @@ impl RawUdpProtocolHandler {
&self, &self,
data: Vec<u8>, data: Vec<u8>,
socket_addr: SocketAddr, socket_addr: SocketAddr,
) -> io::Result<NetworkResult<()>> { ) -> io::Result<NetworkResult<ConnectionDescriptor>> {
if data.len() > MAX_MESSAGE_SIZE { if data.len() > MAX_MESSAGE_SIZE {
bail_io_error_other!("sending too large UDP message"); bail_io_error_other!("sending too large UDP message");
} }
@ -56,7 +56,17 @@ impl RawUdpProtocolHandler {
bail_io_error_other!("UDP partial send") bail_io_error_other!("UDP partial send")
} }
Ok(NetworkResult::value(())) let peer_addr = PeerAddress::new(
SocketAddress::from_socket_addr(socket_addr),
ProtocolType::UDP,
);
let local_socket_addr = self.socket.local_addr()?;
let descriptor = ConnectionDescriptor::new(
peer_addr,
SocketAddress::from_socket_addr(local_socket_addr),
);
Ok(NetworkResult::value(descriptor))
} }
#[instrument(level = "trace", err)] #[instrument(level = "trace", err)]

View File

@ -191,7 +191,7 @@ impl Network {
&self, &self,
dial_info: DialInfo, dial_info: DialInfo,
data: Vec<u8>, data: Vec<u8>,
) -> EyreResult<NetworkResult<()>> { ) -> EyreResult<NetworkResult<ConnectionDescriptor>> {
let data_len = data.len(); let data_len = data.len();
if dial_info.protocol_type() == ProtocolType::UDP { if dial_info.protocol_type() == ProtocolType::UDP {
bail!("no support for UDP protocol"); bail!("no support for UDP protocol");
@ -201,18 +201,25 @@ impl Network {
} }
// Handle connection-oriented protocols // Handle connection-oriented protocols
let conn = self let conn = network_result_try!(
.connection_manager() self.connection_manager()
.get_or_create_connection(None, dial_info.clone()) .get_or_create_connection(Some(local_addr), dial_info.clone())
.await?; .await?
);
let res = conn.send_async(data).await; if let ConnectionHandleSendResult::NotSent(_) = conn.send_async(data).await {
if res.is_ok() { return Ok(NetworkResult::NoConnection(io::Error::new(
// Network accounting io::ErrorKind::ConnectionReset,
self.network_manager() "failed to send",
.stats_packet_sent(dial_info.to_ip_addr(), data_len as u64); )));
} }
res let connection_descriptor = conn.connection_descriptor();
// Network accounting
self.network_manager()
.stats_packet_sent(dial_info.to_ip_addr(), data_len as u64);
Ok(NetworkResult::value(connection_descriptor))
} }
///////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////

View File

@ -267,6 +267,10 @@ impl NodeRef {
self.operate_mut(|e| e.clear_last_connection()) self.operate_mut(|e| e.clear_last_connection())
} }
pub fn set_last_connection(&self, connection_descriptor: ConnectionDescriptor, ts: u64) {
self.operate_mut(|e| e.set_last_connection(connection_descriptor, ts))
}
pub fn has_any_dial_info(&self) -> bool { pub fn has_any_dial_info(&self) -> bool {
self.operate(|e| { self.operate(|e| {
e.node_info() e.node_info()