Skip to content

Commit

Permalink
auto route on macos
Browse files Browse the repository at this point in the history
  • Loading branch information
ibigbug committed Sep 16, 2024
1 parent 41b5993 commit f6b9abe
Show file tree
Hide file tree
Showing 4 changed files with 134 additions and 10 deletions.
10 changes: 5 additions & 5 deletions clash/tests/data/config/rules.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -4,13 +4,13 @@ socks-port: 8889
mixed-port: 8899

tun:
enable: false
enable: true
device-id: "dev://utun1989"
route-all: false
route-all: true
gateway: "198.19.0.1/32"
routes:
- 0.0.0.0/1
- 128.0.0.0/1
# routes:
# - 0.0.0.0/1
# - 128.0.0.0/1

ipv6: true

Expand Down
11 changes: 10 additions & 1 deletion clash_lib/src/proxy/tun/inbound.rs
Original file line number Diff line number Diff line change
Expand Up @@ -11,8 +11,10 @@ use crate::{
app::{dispatcher::Dispatcher, dns::ThreadSafeDNSResolver},
common::errors::{map_io_error, new_io_error},
config::internal::config::TunConfig,
defer,

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / aarch64-unknown-linux-gnu-static-crt

unused imports: `defer` and `self`

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / x86_64-unknown-linux-gnu-static-crt

unused imports: `defer` and `self`

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / i686-unknown-linux-gnu-static-crt

unused imports: `defer` and `self`

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / armv7-unknown-linux-gnueabi

unused imports: `defer` and `self`

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / x86_64-unknown-linux-musl

unused imports: `defer` and `self`

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / i686-unknown-linux-gnu

unused imports: `defer` and `self`

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / x86_64-unknown-linux-gnu

unused imports: `defer` and `self`

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / armv7-unknown-linux-gnueabi-static-crt

unused imports: `defer` and `self`

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / aarch64-unknown-linux-musl

unused imports: `defer` and `self`

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / aarch64-unknown-linux-gnu

unused imports: `defer` and `self`

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / armv7-unknown-linux-gnueabihf

unused imports: `defer` and `self`

Check failure on line 14 in clash_lib/src/proxy/tun/inbound.rs

View workflow job for this annotation

GitHub Actions / armv7-unknown-linux-musleabihf

unused imports: `defer` and `self`
proxy::{
datagram::UdpPacket, tun::routes::maybe_add_routes,
datagram::UdpPacket,
tun::routes::{self, maybe_add_routes},
utils::get_outbound_interface,
},
session::{Network, Session, SocksAddr, Type},
Expand Down Expand Up @@ -200,6 +202,13 @@ pub fn get_runner(
netstack::NetStack::with_buffer_size(512, 256).map_err(map_io_error)?;

Ok(Some(Box::pin(async move {
#[cfg(target_os = "macos")]
defer! {
warn!("cleaning up routes");

let _ = routes::del_default_route();
}

let framed = tun.into_framed();

let (mut tun_sink, mut tun_stream) = framed.split();
Expand Down
116 changes: 112 additions & 4 deletions clash_lib/src/proxy/tun/routes/macos.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,117 @@
use std::net::Ipv4Addr;

use ipnet::IpNet;
use tracing::warn;

use crate::proxy::utils::OutboundInterface;
use crate::{
common::errors::new_io_error,
proxy::utils::{get_outbound_interface, OutboundInterface},
};

/// let's assume that the `route` command is available on macOS
pub fn add_route(via: &OutboundInterface, dest: &IpNet) -> std::io::Result<()> {
let cmd = std::process::Command::new("route")
.arg("add")
.arg("-net")
.arg(&dest.to_string())

Check failure on line 16 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / aarch64-apple-darwin-static-crt

the borrowed expression implements the required traits

Check failure on line 16 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / x86_64-apple-darwin

the borrowed expression implements the required traits

Check failure on line 16 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / x86_64-apple-darwin-static-crt

the borrowed expression implements the required traits

Check failure on line 16 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / aarch64-apple-darwin

the borrowed expression implements the required traits
.arg("-interface")
.arg(&via.name)
.output()?;

warn!("executing: route add -net {} -interface {}", dest, via.name);
if !cmd.status.success() {
Err(new_io_error("add route failed"))
} else {
Ok(())
}
}

fn get_default_gateway() -> std::io::Result<Option<Ipv4Addr>> {
let cmd = std::process::Command::new("route")
.arg("-n")
.arg("get")
.arg("default")
.output()?;

if !cmd.status.success() {
return Ok(None);
}

let output = String::from_utf8_lossy(&cmd.stdout);

let mut gateway = None;
for line in output.lines() {
if line.trim().contains("gateway:") {
gateway = line
.split_whitespace()
.last()
.and_then(|x| x.parse::<Ipv4Addr>().ok());
break;
}
}

Ok(gateway)
}

/// it seems to be fine to add the default route multiple times
pub fn maybe_add_default_route() -> std::io::Result<()> {
let gateway = get_default_gateway()?;
if let Some(gateway) = gateway {
let default_interface =
get_outbound_interface().ok_or(new_io_error("get default interface"))?;

let cmd = std::process::Command::new("route")
.arg("add")
.arg("-ifscope")
.arg(&default_interface.name)
.arg("0/0")
.arg(&gateway.to_string())

Check failure on line 68 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / aarch64-apple-darwin-static-crt

the borrowed expression implements the required traits

Check failure on line 68 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / x86_64-apple-darwin

the borrowed expression implements the required traits

Check failure on line 68 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / x86_64-apple-darwin-static-crt

the borrowed expression implements the required traits

Check failure on line 68 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / aarch64-apple-darwin

the borrowed expression implements the required traits
.output()?;

warn!(
"executing: route add -ifscope {} 0/0 {}",
default_interface.name, gateway
);

if !cmd.status.success() {
Err(new_io_error("add default route failed"))
} else {
Ok(())
}
} else {
Err(new_io_error(
"cant set default route, default gateway not found",
))
}
}

/// failing to delete the default route won't cause route failure
pub fn del_default_route() -> std::io::Result<()> {
let gateway = get_default_gateway()?;
if let Some(gateway) = gateway {
let default_interface =
get_outbound_interface().ok_or(new_io_error("get default interface"))?;
let cmd = std::process::Command::new("route")
.arg("delete")
.arg("-ifscope")
.arg(&default_interface.name)
.arg("0/0")
.arg(&gateway.to_string())

Check failure on line 99 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / aarch64-apple-darwin-static-crt

the borrowed expression implements the required traits

Check failure on line 99 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / x86_64-apple-darwin

the borrowed expression implements the required traits

Check failure on line 99 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / x86_64-apple-darwin-static-crt

the borrowed expression implements the required traits

Check failure on line 99 in clash_lib/src/proxy/tun/routes/macos.rs

View workflow job for this annotation

GitHub Actions / aarch64-apple-darwin

the borrowed expression implements the required traits
.output()?;

warn!(
"executing: route delete -ifscope {} 0/0 {}",
default_interface.name, gateway
);

pub fn add_route(_: &OutboundInterface, _: &IpNet) -> std::io::Result<()> {
warn!("add_route is not implemented on macOS");
Ok(())
if !cmd.status.success() {
Err(new_io_error("delete default route failed"))
} else {
Ok(())
}
} else {
Err(new_io_error(
"cant delete default route, default gateway not found",
))
}
}
7 changes: 7 additions & 0 deletions clash_lib/src/proxy/tun/routes/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ use windows::add_route;
mod macos;
#[cfg(target_os = "macos")]
use macos::add_route;
#[cfg(target_os = "macos")]
pub use macos::del_default_route;

#[cfg(target_os = "linux")]
mod linux;
Expand Down Expand Up @@ -64,6 +66,11 @@ pub fn maybe_add_routes(cfg: &TunConfig, tun_name: &str) -> std::io::Result<()>
for r in default_routes {
add_route(&tun_iface, &r).map_err(map_io_error)?;
}

#[cfg(target_os = "macos")]
{
macos::maybe_add_default_route()?;
}
} else {
for r in &cfg.routes {
add_route(&tun_iface, r).map_err(map_io_error)?;
Expand Down

0 comments on commit f6b9abe

Please sign in to comment.