Proxies and Tor

Make an interface route only certain websites

This makes the browser use a specific interface only for specific URLs. I use it with USB phone tethering, that's why many things are called that way. But it should work with any interface.

How it works: The default route of the interface is within an own ip table that only gets entered for requests bound to the IP of that interface. Locally running tinyproxy binds to that one. Then the browser decides via proxyconfig.pac for what URL to use that proxy.

1. Create connection in Network Manager

...with the name DHCP as Proxy

a) per cli

nmcli con add type ethernet ipv6.method disabled ipv4.never-default true connection.zone public 802-3-ethernet.auto-negotiate true con-name "DHCP as Proxy"

Edit to whatever you need, i.e. you might not need connection.zone, which is the firewall zone.

b) or per GUI:

  1. Create connection in NetworkManager called DHCP as Proxy
  2. set to DHCP IPv4, disable IPv6
  3. disable setting the default route (IPv4 → Routes → [X] Use only for Resources of this connection)

2. Install tinyproxy

dnf install tinyproxy

/etc/tinyproxy/tinyproxy.conf

1
2
3
4
5
6
7
[...]
Port 8888
Listen 127.0.0.1
# Bind will be overwritten by nm dispatch script
# just make sure it's uncommented, doesn't matter what IP it is
Bind 192.168.143.191
[...]

3. Add routing table

/etc/iproute2/rt_tables

[...]
2   phone

4. Add Network Manager dispatch script

This script will run whenever a connection gets established, set the routes correctly, bind tinyproxy to its IP and start it.

/etc/NetworkManager/dispatcher.d/tether_toggle.sh

#!/bin/sh
#
# Make sure table phone is defined in /etc/iproute2/rt_tables
# with priority 2

PHONE_CON_ID="DHCP as Proxy"
TINYPROXY_CONFIG="/etc/tinyproxy/tinyproxy.conf"

if [ "${CONNECTION_ID}" = "${PHONE_CON_ID}" ]; then
  case "${2}" in
    up)
      # getting connection paramters:
      [ "${DEVICE_IP_IFACE}" ] || DEVICE_IP_IFACE=`ls /sys/bus/usb/drivers/rndis_host/*.0/net 2> /dev/null`
      [ "${DHCP4_ROUTERS}" ] || DHCP4_ROUTERS=`ip -4 route show table all default dev "${DEVICE_IP_IFACE}" | sed 's/.*via \([^ ]*\).*/\1/g'`
      [ "${DHCP4_IP_ADDRESS}" ] || DHCP4_IP_ADDRESS=`ip -4 addr show dev "${DEVICE_IP_IFACE}" | grep inet | sed 's/.*inet \([^\/]*\)\/.*/\1/'`
      # create default route on phone table
      ip route add default via "${DHCP4_ROUTERS}" table phone

      # example rule for a custom IPs to route through
      #ip route add "${EXTERNAL_IP}" via ${DHCP4_ROUTERS}

      # traffic bound to DHCP4_IP_ADDRESS over phone
      ip rule add from ${DHCP4_IP_ADDRESS} lookup phone prio 1000
      # bind tinyproxy to DHCP4_IP_ADDRESS and start it
      sed -i "s/Bind .*/Bind ${DHCP4_IP_ADDRESS}/" "${TINYPROXY_CONFIG}"
      systemctl start tinyproxy

      {
        # make really sure that there is no general default route for device
        sleep 3
        CNT=0
        while [ "$(ip -4 route show default dev ${DEVICE_IP_IFACE})" != "" ] && [ ${CNT} -lt 10 ]; do
          CNT=$((CNT+1))
          ip route del default dev "${DEVICE_IP_IFACE}"
          sleep 1
        done
      } &
      ;;
    down)
      systemctl stop tinyproxy
      ;;
  esac
fi

make it executable

chmod +x /etc/NetworkManager/dispatcher.d/tether_toggle.sh

5. setup proxy autoconfig for browser

~/.config/proxyconfig.pac

function FindProxyForURL(url, host) { 
  if (dnsDomainIs(host, "sys.4channel.org")
    || dnsDomainIs(host, "sys.4chan.org")
    || dnsDomainIs(host, "boards.4channel.org")
    || dnsDomainIs(host, "boards.4chan.org")
    || dnsDomainIs(host, "www.4channel.org")
    || dnsDomainIs(host, "www.4chan.org")
  )
      return "PROXY 127.0.0.1:8888; DIRECT";
  if (dnsDomainIs(host, ".onion"))
      return "SOCKS 127.0.0.1:9050";
  return "DIRECT"; 
}

(note that 127.0.0.1:9050 is local running TOR here, if you use that, you will also want "network.http.referer.hideOnionSource": true in firefox)

Now set ~/.config/proxyconfig.pac as Automatic Proxy Configuration in whatever browser you need, like in firefox user.js:

1
2
3
user_pref("network.proxy.autoconfig_url", "file:///home/username/.config/proxyconfig.pac");
user_pref("network.proxy.socks_remote_dns", true);
user_pref("network.proxy.type", 2);

or in chromium browsers by launching with the parameter:

--proxy-pac-url='data:application/x-javascript-config;base64,'$(base64 -w0 ~/.config/proxyconfig.pac)

Script to enable / disable phone tethering on KDE

On enable, we send the phone to airplane mode for a temporary time, to reroll the IP. Disabling doesn't work on Android 13+ phones anymore. This uses kdialog as graphical selection window.

Install adb

dnf install android-tools

~/.local/bin/tether

#!/bin/sh

TEXT="Android USB Tethering"
IFACE=`ls /sys/bus/usb/drivers/rndis_host/*.0/net 2> /dev/null`

if [ "${IFACE}" ]; then
  INPT=`kdialog --title "Tethering" --radiolist "${TEXT}" enable "Enabled" on disable "Disabled" off`
else
  INPT=`kdialog --title "Tethering" --radiolist "${TEXT}" enable "Enabled" off disable "Disabled" on`
fi

[ "${?}" != 0 ] && exit

if [ "${INPT}" = "enable" ] && [ -z "${IFACE}" ]; then
  # disable WIFI
  echo "Disable WiFi on phone"
  adb shell svc wifi disable
  sleep 2
  # airplane mode on/off to change IP
  echo "Rolling new IP"
  adb shell cmd connectivity airplane-mode enable
  sleep 2
  adb shell cmd connectivity airplane-mode disable
  sleep 2
  # adb over network, to keep phone accessible even during tethering
  echo "Enabling adb on TCP"
  adb tcpip 5555
  sleep 2
  # enable usb tethering
  echo "Enabling USB Tethering"
  adb shell svc usb setFunctions rndis
  adb kill-server
elif [ "${INPT}" = "disable" ] && [ "${IFACE}" ]; then 
  PHONE_IP=`ip -4 route show table all default dev "${IFACE}" | sed 's/.*via \([^ ]*\).*/\1/g'`
  if (( $(grep -c . <<<"${PHONE_IP}") > 1 )); then
    PHONE_IP="[`grep : <<<"${PHONE_IP}"`%${IFACE}]:5555"
  fi
  echo "Phone IP: ${PHONE_IP}"
  if [ "${PHONE_IP}" ]; then
    adb connect "${PHONE_IP}"
    adb shell svc wifi enable
    adb shell svc usb setFunctions mtp
    adb disconnect
    adb kill-server
  fi
fi

exit ${?}
chmod +x ~/.local/bin/tether
Edit Report
Pub: 03 Sep 2023 11:41 UTC
Edit: 03 Sep 2023 11:43 UTC
Views: 543