Merge pull request #1127 from 21pages/fix_10054
fix 10054: change direct to relay when RST
This commit is contained in:
commit
3b3d34fabf
22
Cargo.lock
generated
22
Cargo.lock
generated
@ -1470,6 +1470,27 @@ dependencies = [
|
|||||||
"synstructure",
|
"synstructure",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "errno"
|
||||||
|
version = "0.2.8"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "f639046355ee4f37944e44f60642c6f3a7efa3cf6b78c78a0d989a8ce6c396a1"
|
||||||
|
dependencies = [
|
||||||
|
"errno-dragonfly",
|
||||||
|
"libc",
|
||||||
|
"winapi 0.3.9",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "errno-dragonfly"
|
||||||
|
version = "0.1.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "aa68f1b12764fab894d2755d2518754e71b4fd80ecfb822714a1206c2aab39bf"
|
||||||
|
dependencies = [
|
||||||
|
"cc",
|
||||||
|
"libc",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "error-code"
|
name = "error-code"
|
||||||
version = "2.3.1"
|
version = "2.3.1"
|
||||||
@ -4168,6 +4189,7 @@ dependencies = [
|
|||||||
"default-net",
|
"default-net",
|
||||||
"dispatch",
|
"dispatch",
|
||||||
"enigo",
|
"enigo",
|
||||||
|
"errno",
|
||||||
"evdev",
|
"evdev",
|
||||||
"flexi_logger",
|
"flexi_logger",
|
||||||
"flutter_rust_bridge",
|
"flutter_rust_bridge",
|
||||||
|
@ -58,6 +58,7 @@ num_cpus = "1.13"
|
|||||||
bytes = { version = "1.2", features = ["serde"] }
|
bytes = { version = "1.2", features = ["serde"] }
|
||||||
default-net = "0.11.0"
|
default-net = "0.11.0"
|
||||||
wol-rs = "0.9.1"
|
wol-rs = "0.9.1"
|
||||||
|
errno = "0.2.8"
|
||||||
|
|
||||||
[target.'cfg(not(target_os = "linux"))'.dependencies]
|
[target.'cfg(not(target_os = "linux"))'.dependencies]
|
||||||
reqwest = { version = "0.11", features = ["json", "rustls-tls"], default-features=false }
|
reqwest = { version = "0.11", features = ["json", "rustls-tls"], default-features=false }
|
||||||
|
@ -21,6 +21,7 @@ use std::{
|
|||||||
|
|
||||||
pub const RENDEZVOUS_TIMEOUT: u64 = 12_000;
|
pub const RENDEZVOUS_TIMEOUT: u64 = 12_000;
|
||||||
pub const CONNECT_TIMEOUT: u64 = 18_000;
|
pub const CONNECT_TIMEOUT: u64 = 18_000;
|
||||||
|
pub const READ_TIMEOUT: u64 = 30_000;
|
||||||
pub const REG_INTERVAL: i64 = 12_000;
|
pub const REG_INTERVAL: i64 = 12_000;
|
||||||
pub const COMPRESS_LEVEL: i32 = 3;
|
pub const COMPRESS_LEVEL: i32 = 3;
|
||||||
const SERIAL: i32 = 3;
|
const SERIAL: i32 = 3;
|
||||||
|
@ -24,7 +24,10 @@ use hbb_common::{
|
|||||||
allow_err,
|
allow_err,
|
||||||
anyhow::{anyhow, Context},
|
anyhow::{anyhow, Context},
|
||||||
bail,
|
bail,
|
||||||
config::{Config, PeerConfig, PeerInfoSerde, CONNECT_TIMEOUT, RELAY_PORT, RENDEZVOUS_TIMEOUT},
|
config::{
|
||||||
|
Config, PeerConfig, PeerInfoSerde, CONNECT_TIMEOUT, READ_TIMEOUT, RELAY_PORT,
|
||||||
|
RENDEZVOUS_TIMEOUT,
|
||||||
|
},
|
||||||
log,
|
log,
|
||||||
message_proto::{option_message::BoolOption, *},
|
message_proto::{option_message::BoolOption, *},
|
||||||
protobuf::Message as _,
|
protobuf::Message as _,
|
||||||
@ -116,8 +119,9 @@ impl Client {
|
|||||||
key: &str,
|
key: &str,
|
||||||
token: &str,
|
token: &str,
|
||||||
conn_type: ConnType,
|
conn_type: ConnType,
|
||||||
|
interface: impl Interface,
|
||||||
) -> ResultType<(Stream, bool)> {
|
) -> ResultType<(Stream, bool)> {
|
||||||
match Self::_start(peer, key, token, conn_type).await {
|
match Self::_start(peer, key, token, conn_type, interface).await {
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
let err_str = err.to_string();
|
let err_str = err.to_string();
|
||||||
if err_str.starts_with("Failed") {
|
if err_str.starts_with("Failed") {
|
||||||
@ -135,6 +139,7 @@ impl Client {
|
|||||||
key: &str,
|
key: &str,
|
||||||
token: &str,
|
token: &str,
|
||||||
conn_type: ConnType,
|
conn_type: ConnType,
|
||||||
|
interface: impl Interface,
|
||||||
) -> ResultType<(Stream, bool)> {
|
) -> ResultType<(Stream, bool)> {
|
||||||
// to-do: remember the port for each peer, so that we can retry easier
|
// to-do: remember the port for each peer, so that we can retry easier
|
||||||
let any_addr = Config::get_any_listen_addr();
|
let any_addr = Config::get_any_listen_addr();
|
||||||
@ -181,7 +186,11 @@ impl Client {
|
|||||||
log::info!("#{} punch attempt with {}, id: {}", i, my_addr, peer);
|
log::info!("#{} punch attempt with {}, id: {}", i, my_addr, peer);
|
||||||
let mut msg_out = RendezvousMessage::new();
|
let mut msg_out = RendezvousMessage::new();
|
||||||
use hbb_common::protobuf::Enum;
|
use hbb_common::protobuf::Enum;
|
||||||
let nat_type = NatType::from_i32(my_nat_type).unwrap_or(NatType::UNKNOWN_NAT);
|
let nat_type = if interface.is_force_relay() {
|
||||||
|
NatType::SYMMETRIC
|
||||||
|
} else {
|
||||||
|
NatType::from_i32(my_nat_type).unwrap_or(NatType::UNKNOWN_NAT)
|
||||||
|
};
|
||||||
msg_out.set_punch_hole_request(PunchHoleRequest {
|
msg_out.set_punch_hole_request(PunchHoleRequest {
|
||||||
id: peer.to_owned(),
|
id: peer.to_owned(),
|
||||||
token: token.to_owned(),
|
token: token.to_owned(),
|
||||||
@ -233,7 +242,15 @@ impl Client {
|
|||||||
let mut conn =
|
let mut conn =
|
||||||
Self::create_relay(peer, rr.uuid, rr.relay_server, key, conn_type)
|
Self::create_relay(peer, rr.uuid, rr.relay_server, key, conn_type)
|
||||||
.await?;
|
.await?;
|
||||||
Self::secure_connection(peer, signed_id_pk, key, &mut conn).await?;
|
Self::secure_connection(
|
||||||
|
peer,
|
||||||
|
signed_id_pk,
|
||||||
|
key,
|
||||||
|
&mut conn,
|
||||||
|
false,
|
||||||
|
interface,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
return Ok((conn, false));
|
return Ok((conn, false));
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
@ -274,6 +291,7 @@ impl Client {
|
|||||||
key,
|
key,
|
||||||
token,
|
token,
|
||||||
conn_type,
|
conn_type,
|
||||||
|
interface,
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
}
|
}
|
||||||
@ -292,6 +310,7 @@ impl Client {
|
|||||||
key: &str,
|
key: &str,
|
||||||
token: &str,
|
token: &str,
|
||||||
conn_type: ConnType,
|
conn_type: ConnType,
|
||||||
|
interface: impl Interface,
|
||||||
) -> ResultType<(Stream, bool)> {
|
) -> ResultType<(Stream, bool)> {
|
||||||
let direct_failures = PeerConfig::load(peer_id).direct_failures;
|
let direct_failures = PeerConfig::load(peer_id).direct_failures;
|
||||||
let mut connect_timeout = 0;
|
let mut connect_timeout = 0;
|
||||||
@ -329,8 +348,8 @@ impl Client {
|
|||||||
let start = std::time::Instant::now();
|
let start = std::time::Instant::now();
|
||||||
// NOTICE: Socks5 is be used event in intranet. Which may be not a good way.
|
// NOTICE: Socks5 is be used event in intranet. Which may be not a good way.
|
||||||
let mut conn = socket_client::connect_tcp(peer, local_addr, connect_timeout).await;
|
let mut conn = socket_client::connect_tcp(peer, local_addr, connect_timeout).await;
|
||||||
let direct = !conn.is_err();
|
let mut direct = !conn.is_err();
|
||||||
if conn.is_err() {
|
if interface.is_force_relay() || conn.is_err() {
|
||||||
if !relay_server.is_empty() {
|
if !relay_server.is_empty() {
|
||||||
conn = Self::request_relay(
|
conn = Self::request_relay(
|
||||||
peer_id,
|
peer_id,
|
||||||
@ -348,6 +367,7 @@ impl Client {
|
|||||||
conn.err().unwrap()
|
conn.err().unwrap()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
direct = false;
|
||||||
} else {
|
} else {
|
||||||
bail!("Failed to make direct connection to remote desktop");
|
bail!("Failed to make direct connection to remote desktop");
|
||||||
}
|
}
|
||||||
@ -360,7 +380,7 @@ impl Client {
|
|||||||
}
|
}
|
||||||
let mut conn = conn?;
|
let mut conn = conn?;
|
||||||
log::info!("{:?} used to establish connection", start.elapsed());
|
log::info!("{:?} used to establish connection", start.elapsed());
|
||||||
Self::secure_connection(peer_id, signed_id_pk, key, &mut conn).await?;
|
Self::secure_connection(peer_id, signed_id_pk, key, &mut conn, direct, interface).await?;
|
||||||
Ok((conn, direct))
|
Ok((conn, direct))
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -369,6 +389,8 @@ impl Client {
|
|||||||
signed_id_pk: Vec<u8>,
|
signed_id_pk: Vec<u8>,
|
||||||
key: &str,
|
key: &str,
|
||||||
conn: &mut Stream,
|
conn: &mut Stream,
|
||||||
|
direct: bool,
|
||||||
|
mut interface: impl Interface,
|
||||||
) -> ResultType<()> {
|
) -> ResultType<()> {
|
||||||
let rs_pk = get_rs_pk(if key.is_empty() {
|
let rs_pk = get_rs_pk(if key.is_empty() {
|
||||||
hbb_common::config::RS_PUB_KEY
|
hbb_common::config::RS_PUB_KEY
|
||||||
@ -394,9 +416,15 @@ impl Client {
|
|||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
match timeout(CONNECT_TIMEOUT, conn.next()).await? {
|
match timeout(READ_TIMEOUT, conn.next()).await? {
|
||||||
Some(res) => {
|
Some(res) => {
|
||||||
let bytes = res?;
|
let bytes = match res {
|
||||||
|
Ok(bytes) => bytes,
|
||||||
|
Err(err) => {
|
||||||
|
interface.set_force_relay(direct, false);
|
||||||
|
bail!("{}", err);
|
||||||
|
}
|
||||||
|
};
|
||||||
if let Ok(msg_in) = Message::parse_from_bytes(&bytes) {
|
if let Ok(msg_in) = Message::parse_from_bytes(&bytes) {
|
||||||
if let Some(message::Union::SignedId(si)) = msg_in.union {
|
if let Some(message::Union::SignedId(si)) = msg_in.union {
|
||||||
if let Ok((id, their_pk_b)) = decode_id_pk(&si.id, &sign_pk) {
|
if let Ok((id, their_pk_b)) = decode_id_pk(&si.id, &sign_pk) {
|
||||||
@ -786,6 +814,7 @@ pub struct LoginConfigHandler {
|
|||||||
session_id: u64,
|
session_id: u64,
|
||||||
pub supported_encoding: Option<(bool, bool)>,
|
pub supported_encoding: Option<(bool, bool)>,
|
||||||
pub restarting_remote_device: bool,
|
pub restarting_remote_device: bool,
|
||||||
|
pub force_relay: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Deref for LoginConfigHandler {
|
impl Deref for LoginConfigHandler {
|
||||||
@ -812,6 +841,7 @@ impl LoginConfigHandler {
|
|||||||
self.session_id = rand::random();
|
self.session_id = rand::random();
|
||||||
self.supported_encoding = None;
|
self.supported_encoding = None;
|
||||||
self.restarting_remote_device = false;
|
self.restarting_remote_device = false;
|
||||||
|
self.force_relay = !self.get_option("force-always-relay").is_empty();
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn should_auto_login(&self) -> String {
|
pub fn should_auto_login(&self) -> String {
|
||||||
@ -1418,6 +1448,8 @@ pub trait Interface: Send + Clone + 'static + Sized {
|
|||||||
fn msgbox(&self, msgtype: &str, title: &str, text: &str);
|
fn msgbox(&self, msgtype: &str, title: &str, text: &str);
|
||||||
fn handle_login_error(&mut self, err: &str) -> bool;
|
fn handle_login_error(&mut self, err: &str) -> bool;
|
||||||
fn handle_peer_info(&mut self, pi: PeerInfo);
|
fn handle_peer_info(&mut self, pi: PeerInfo);
|
||||||
|
fn set_force_relay(&mut self, direct: bool, received: bool);
|
||||||
|
fn is_force_relay(&self) -> bool;
|
||||||
async fn handle_hash(&mut self, pass: &str, hash: Hash, peer: &mut Stream);
|
async fn handle_hash(&mut self, pass: &str, hash: Hash, peer: &mut Stream);
|
||||||
async fn handle_login_from_ui(&mut self, password: String, remember: bool, peer: &mut Stream);
|
async fn handle_login_from_ui(&mut self, password: String, remember: bool, peer: &mut Stream);
|
||||||
async fn handle_test_delay(&mut self, t: TestDelay, peer: &mut Stream);
|
async fn handle_test_delay(&mut self, t: TestDelay, peer: &mut Stream);
|
||||||
@ -1579,14 +1611,16 @@ lazy_static::lazy_static! {
|
|||||||
pub fn check_if_retry(msgtype: &str, title: &str, text: &str) -> bool {
|
pub fn check_if_retry(msgtype: &str, title: &str, text: &str) -> bool {
|
||||||
msgtype == "error"
|
msgtype == "error"
|
||||||
&& title == "Connection Error"
|
&& title == "Connection Error"
|
||||||
&& !text.to_lowercase().contains("offline")
|
&& (text.contains("10054")
|
||||||
&& !text.to_lowercase().contains("exist")
|
|| text.contains("104")
|
||||||
&& !text.to_lowercase().contains("handshake")
|
|| (!text.to_lowercase().contains("offline")
|
||||||
&& !text.to_lowercase().contains("failed")
|
&& !text.to_lowercase().contains("exist")
|
||||||
&& !text.to_lowercase().contains("resolve")
|
&& !text.to_lowercase().contains("handshake")
|
||||||
&& !text.to_lowercase().contains("mismatch")
|
&& !text.to_lowercase().contains("failed")
|
||||||
&& !text.to_lowercase().contains("manually")
|
&& !text.to_lowercase().contains("resolve")
|
||||||
&& !text.to_lowercase().contains("not allowed")
|
&& !text.to_lowercase().contains("mismatch")
|
||||||
|
&& !text.to_lowercase().contains("manually")
|
||||||
|
&& !text.to_lowercase().contains("not allowed")))
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
use crate::client::*;
|
use crate::client::*;
|
||||||
use hbb_common::{
|
use hbb_common::{
|
||||||
allow_err, bail,
|
allow_err, bail,
|
||||||
config::CONNECT_TIMEOUT,
|
config::READ_TIMEOUT,
|
||||||
futures::{SinkExt, StreamExt},
|
futures::{SinkExt, StreamExt},
|
||||||
log,
|
log,
|
||||||
message_proto::*,
|
message_proto::*,
|
||||||
@ -105,22 +105,61 @@ async fn connect_and_login(
|
|||||||
key: &str,
|
key: &str,
|
||||||
token: &str,
|
token: &str,
|
||||||
is_rdp: bool,
|
is_rdp: bool,
|
||||||
|
) -> ResultType<Option<Stream>> {
|
||||||
|
let mut res = connect_and_login_2(
|
||||||
|
id,
|
||||||
|
password,
|
||||||
|
ui_receiver,
|
||||||
|
interface.clone(),
|
||||||
|
forward,
|
||||||
|
key,
|
||||||
|
token,
|
||||||
|
is_rdp,
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
if res.is_err() && interface.is_force_relay() {
|
||||||
|
res = connect_and_login_2(
|
||||||
|
id,
|
||||||
|
password,
|
||||||
|
ui_receiver,
|
||||||
|
interface,
|
||||||
|
forward,
|
||||||
|
key,
|
||||||
|
token,
|
||||||
|
is_rdp,
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
res
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn connect_and_login_2(
|
||||||
|
id: &str,
|
||||||
|
password: &str,
|
||||||
|
ui_receiver: &mut mpsc::UnboundedReceiver<Data>,
|
||||||
|
interface: impl Interface,
|
||||||
|
forward: &mut Framed<TcpStream, BytesCodec>,
|
||||||
|
key: &str,
|
||||||
|
token: &str,
|
||||||
|
is_rdp: bool,
|
||||||
) -> ResultType<Option<Stream>> {
|
) -> ResultType<Option<Stream>> {
|
||||||
let conn_type = if is_rdp {
|
let conn_type = if is_rdp {
|
||||||
ConnType::RDP
|
ConnType::RDP
|
||||||
} else {
|
} else {
|
||||||
ConnType::PORT_FORWARD
|
ConnType::PORT_FORWARD
|
||||||
};
|
};
|
||||||
let (mut stream, _) = Client::start(id, key, token, conn_type).await?;
|
let (mut stream, direct) = Client::start(id, key, token, conn_type, interface.clone()).await?;
|
||||||
let mut interface = interface;
|
let mut interface = interface;
|
||||||
let mut buffer = Vec::new();
|
let mut buffer = Vec::new();
|
||||||
|
let mut received = false;
|
||||||
loop {
|
loop {
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
res = timeout(CONNECT_TIMEOUT, stream.next()) => match res {
|
res = timeout(READ_TIMEOUT, stream.next()) => match res {
|
||||||
Err(_) => {
|
Err(_) => {
|
||||||
bail!("Timeout");
|
bail!("Timeout");
|
||||||
}
|
}
|
||||||
Ok(Some(Ok(bytes))) => {
|
Ok(Some(Ok(bytes))) => {
|
||||||
|
received = true;
|
||||||
let msg_in = Message::parse_from_bytes(&bytes)?;
|
let msg_in = Message::parse_from_bytes(&bytes)?;
|
||||||
match msg_in.union {
|
match msg_in.union {
|
||||||
Some(message::Union::Hash(hash)) => {
|
Some(message::Union::Hash(hash)) => {
|
||||||
@ -143,6 +182,11 @@ async fn connect_and_login(
|
|||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Ok(Some(Err(err))) => {
|
||||||
|
log::error!("Connection closed: {}", err);
|
||||||
|
interface.set_force_relay(direct, received);
|
||||||
|
bail!("Connection closed: {}", err);
|
||||||
|
}
|
||||||
_ => {
|
_ => {
|
||||||
bail!("Reset by the peer");
|
bail!("Reset by the peer");
|
||||||
}
|
}
|
||||||
|
@ -316,7 +316,7 @@ class SessionList: Reactor.Component {
|
|||||||
<li #connect>{translate('Connect')}</li>
|
<li #connect>{translate('Connect')}</li>
|
||||||
<li #transfer>{translate('Transfer File')}</li>
|
<li #transfer>{translate('Transfer File')}</li>
|
||||||
<li #tunnel>{translate('TCP Tunneling')}</li>
|
<li #tunnel>{translate('TCP Tunneling')}</li>
|
||||||
{false && !handler.using_public_server() && <li #force-always-relay><span>{svg_checkmark}</span>{translate('Always connect via relay')}</li>}
|
<li #force-always-relay><span>{svg_checkmark}</span>{translate('Always connect via relay')}</li>
|
||||||
<li #rdp>RDP<EditRdpPort /></li>
|
<li #rdp>RDP<EditRdpPort /></li>
|
||||||
<li #wol>{translate('WOL')}</li>
|
<li #wol>{translate('WOL')}</li>
|
||||||
<div .separator />
|
<div .separator />
|
||||||
@ -396,7 +396,6 @@ class SessionList: Reactor.Component {
|
|||||||
if (el) {
|
if (el) {
|
||||||
var force = handler.get_peer_option(id, "force-always-relay");
|
var force = handler.get_peer_option(id, "force-always-relay");
|
||||||
el.attributes.toggleClass("selected", force == "Y");
|
el.attributes.toggleClass("selected", force == "Y");
|
||||||
el.attributes.toggleClass("line-through", force != "Y");
|
|
||||||
}
|
}
|
||||||
var conn = this.$(menu #connect);
|
var conn = this.$(menu #connect);
|
||||||
if (conn) {
|
if (conn) {
|
||||||
|
@ -53,6 +53,7 @@ use crate::{
|
|||||||
client::*,
|
client::*,
|
||||||
common::{self, check_clipboard, update_clipboard, ClipboardContext, CLIPBOARD_INTERVAL},
|
common::{self, check_clipboard, update_clipboard, ClipboardContext, CLIPBOARD_INTERVAL},
|
||||||
};
|
};
|
||||||
|
use errno;
|
||||||
|
|
||||||
type Video = AssetPtr<video_destination>;
|
type Video = AssetPtr<video_destination>;
|
||||||
|
|
||||||
@ -1456,12 +1457,21 @@ impl Remote {
|
|||||||
async fn io_loop(&mut self, key: &str, token: &str) {
|
async fn io_loop(&mut self, key: &str, token: &str) {
|
||||||
let stop_clipboard = self.start_clipboard();
|
let stop_clipboard = self.start_clipboard();
|
||||||
let mut last_recv_time = Instant::now();
|
let mut last_recv_time = Instant::now();
|
||||||
|
let mut received = false;
|
||||||
let conn_type = if self.handler.is_file_transfer() {
|
let conn_type = if self.handler.is_file_transfer() {
|
||||||
ConnType::FILE_TRANSFER
|
ConnType::FILE_TRANSFER
|
||||||
} else {
|
} else {
|
||||||
ConnType::default()
|
ConnType::default()
|
||||||
};
|
};
|
||||||
match Client::start(&self.handler.id, key, token, conn_type).await {
|
match Client::start(
|
||||||
|
&self.handler.id,
|
||||||
|
key,
|
||||||
|
token,
|
||||||
|
conn_type,
|
||||||
|
self.handler.clone(),
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
{
|
||||||
Ok((mut peer, direct)) => {
|
Ok((mut peer, direct)) => {
|
||||||
SERVER_KEYBOARD_ENABLED.store(true, Ordering::SeqCst);
|
SERVER_KEYBOARD_ENABLED.store(true, Ordering::SeqCst);
|
||||||
SERVER_CLIPBOARD_ENABLED.store(true, Ordering::SeqCst);
|
SERVER_CLIPBOARD_ENABLED.store(true, Ordering::SeqCst);
|
||||||
@ -1484,11 +1494,13 @@ impl Remote {
|
|||||||
match res {
|
match res {
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
log::error!("Connection closed: {}", err);
|
log::error!("Connection closed: {}", err);
|
||||||
|
self.handler.set_force_relay(direct, received);
|
||||||
self.handler.msgbox("error", "Connection Error", &err.to_string());
|
self.handler.msgbox("error", "Connection Error", &err.to_string());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
Ok(ref bytes) => {
|
Ok(ref bytes) => {
|
||||||
last_recv_time = Instant::now();
|
last_recv_time = Instant::now();
|
||||||
|
received = true;
|
||||||
self.data_count.fetch_add(bytes.len(), Ordering::Relaxed);
|
self.data_count.fetch_add(bytes.len(), Ordering::Relaxed);
|
||||||
if !self.handle_msg_from_peer(bytes, &mut peer).await {
|
if !self.handle_msg_from_peer(bytes, &mut peer).await {
|
||||||
break
|
break
|
||||||
@ -2073,18 +2085,18 @@ impl Remote {
|
|||||||
|
|
||||||
async fn send_opts_after_login(&self, peer: &mut Stream) {
|
async fn send_opts_after_login(&self, peer: &mut Stream) {
|
||||||
if let Some(opts) = self
|
if let Some(opts) = self
|
||||||
.handler
|
.handler
|
||||||
.lc
|
.lc
|
||||||
.read()
|
.read()
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.get_option_message_after_login()
|
.get_option_message_after_login()
|
||||||
{
|
{
|
||||||
let mut misc = Misc::new();
|
let mut misc = Misc::new();
|
||||||
misc.set_option(opts);
|
misc.set_option(opts);
|
||||||
let mut msg_out = Message::new();
|
let mut msg_out = Message::new();
|
||||||
msg_out.set_misc(misc);
|
msg_out.set_misc(misc);
|
||||||
allow_err!(peer.send(&msg_out).await);
|
allow_err!(peer.send(&msg_out).await);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn handle_msg_from_peer(&mut self, data: &[u8], peer: &mut Stream) -> bool {
|
async fn handle_msg_from_peer(&mut self, data: &[u8], peer: &mut Stream) -> bool {
|
||||||
@ -2695,6 +2707,24 @@ impl Interface for Handler {
|
|||||||
handle_test_delay(t, peer).await;
|
handle_test_delay(t, peer).await;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn set_force_relay(&mut self, direct: bool, received: bool) {
|
||||||
|
let mut lc = self.lc.write().unwrap();
|
||||||
|
lc.force_relay = false;
|
||||||
|
if direct && !received {
|
||||||
|
let errno = errno::errno().0;
|
||||||
|
log::info!("errno is {}", errno);
|
||||||
|
// TODO: check mac and ios
|
||||||
|
if cfg!(windows) && errno == 10054 || !cfg!(windows) && errno == 104 {
|
||||||
|
lc.force_relay = true;
|
||||||
|
lc.set_option("force-always-relay".to_owned(), "Y".to_owned());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_force_relay(&self) -> bool {
|
||||||
|
self.lc.read().unwrap().force_relay
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Handler {
|
impl Handler {
|
||||||
|
Loading…
Reference in New Issue
Block a user