Skip to content

Cloudflared Tunnel on Termux

This guide explains how to start a Cloudflared tunnel on an Android device using the Termux terminal. With this setup, you can connect to your device remotely from the public Internet, even if it's behind NAT or has a changing public IP.

You'll be able to use SSH and ADB to connect, and you won't need root access.

Configuration

You'll need to configure some variables during the setup process. Some of them will be known and have to updated once some steps are finished. Some will be known upfront, while others will need updating as you go. Save these in a setup.env file on both your Android device (server) and Linux desktop (client):

# Cloudflared tunnel ID
TUNNEL_ID=
# DNS name for SSH tunnel
TUNNEL_DNS_SSH=
# DNS name for ADB tunnel
TUNNEL_DNS_ADB=
# Slack webhook URL for notifications
SLACK_WEBHOOK=
# SSH public key of the client
SSH_CLIENT_PUB_KEY=
# Termux user ID, assigned by Android during installation
TERMUX_USER_ID=
# Local network IP address of the Android device
SERVER_LAN_IP=

Cloudflare Setup

Open your Cloudflare Dashboard and create a tunnel under Zero Trust / Networks / Tunnels. Create a cloudflared tunnel with the following public hostnames:

  • termux-ssh.example.com pointing to a service tcp://localhost:8022
  • termux-adb.example.com pointing to a service tcp://localhost:5555

After creating the tunnel, update your setup.env file with the new tunnel ID:

TUNNEL_ID=d6ab1533
TUNNEL_DNS_SSH=termux-ssh.example.com
TUNNEL_DNS_ADB=termux-adb.example.com

Optional: Slack Notifications

Configure Slack incoming webhook to receive notifications as soon as the tunnel is online. Update setup.env:

SLACK_WEBHOOK=https://hooks.slack.com/services/DUP4

Add your SSH key

Get your client's public SSH key using cat ~/.ssh/id_rsa.pub and update setup.env:

SSH_CLIENT_PUB_KEY="ssh-rsa SSHRSAKEY me@box"

Termux Setup

Install Termux on your Android device.

Install the necessary packages:

pkg --check-mirror update
pkg install vim git openssh curl curlie dnsutils make python uv zsh termux-api
uv pip install --system supervisor

Find your user ID and LAN IP address with:

id -u
ifconfig
Update your setup.env with this information:
TERMUX_USER_ID=10000
SERVER_LAN_IP=192.168.0.2

Create and activate setup.env:

vim setup.env  # paste the content
. setup.env

Add client's SSH key to ~/.ssh/authorized_keys.

mkdir -p ~/.ssh
echo "$SSH_CLIENT_PUB_KEY" >> ~/.ssh/authorized_keys

For easier typing, you can run the SSH server on your Android device and connect to it from the same LAN.

# On the Android device:
sshd -D -d -p 8023
# On the client device:
. setup.env
ssh $TERMUX_USER_ID@$SERVER_LAN_IP -p 8023
# Now you can run the rest of the commands remotely

Setup Termux for convenient use:

cat << 'EOF' > ~/.termux/termux.properties
extra-keys = [ \
 ['(', ')', '[', ']', '<', '>', '`', '*'], \
 [':', '~', '-', '_', '/', '|', 'BACKSLASH', '='], \
 ['ESC','QUOTE', 'APOSTROPHE', 'PGUP', 'HOME','UP','END','+'], \
 ['TAB','CTRL','ALT','PGDN','LEFT','DOWN','RIGHT','DEL'] \
]
allow-external-apps = true
EOF

Install Termux:API and Termux:Widget

mkdir -p tmp
wget https://github.com/termux/termux-api/releases/download/v0.50.1/termux-api_v0.50.1+github-debug.apk -O tmp/termux-api_v0.50.1+github-debug.apk
termux-open tmp/termux-api_v0.50.1+github-debug.apk

wget https://github.com/termux/termux-widget/releases/download/v0.14.0/termux-widget-app_v0.14.0+github.debug.apk -O tmp/termux-widget-app_v0.14.0+github.debug.apk
termux-open tmp/termux-widget-app_v0.14.0+github.debug.apk

mkdir -p /data/data/com.termux/files/home/.shortcuts/tasks
chmod 700 -R /data/data/com.termux/files/home/.shortcuts

Install the correct version of cloudflared (don't use pkg install cloudflared):

mkdir -p tmp
wget https://github.com/igrek51/cloudflared-termux/releases/download/2025.2.0/cloudflared -O tmp/cloudflared
chmod +x tmp/cloudflared
install cloudflared /data/data/com.termux/files/usr/bin
Or build it manually.

Install the local ADB client. Enable Developer Options and USB debugging, and authorize the device:

curl -s https://raw.githubusercontent.com/rendiix/termux-adb-fastboot/master/install.sh | bash
adb devices  # This client should have access to a local device
adb shell

Authenticate and configure cloudflared:

mkdir -p ~/cloudflare
cloudflared tunnel login
cloudflared tunnel token --cred-file ~/.cloudflared/$TUNNEL_ID.json $TUNNEL_ID

cat << EOF > ~/.cloudflared/config.yml
# protocol: http2
url: tcp://localhost:8022
tunnel: $TUNNEL_ID
credentials-file: /data/data/com.termux/files/home/.cloudflared/$TUNNEL_ID.json
EOF

Configure a Slack notifier script.

cat << EOF > ~/cloudflare/slack-notifier.sh
#!/data/data/com.termux/files/usr/bin/bash -ex
MESSAGE=\$(cat <<EOF2
Opening cloudflared tunnel...
User ID: \$(id -u)
Device: \$(getprop ro.product.brand) \$(getprop ro.product.model), \$(getprop ro.product.device), \$(uname -m)
Android: version \$(getprop ro.build.version.release), API level \$(getprop ro.build.version.sdk)
EOF2
)
curl -v -X POST -H 'Content-type: application/json' --data "{\"text\":\"\$MESSAGE\"}" \
    $SLACK_WEBHOOK
EOF
chmod +x ~/cloudflare/slack-notifier.sh

Set up supervisord to run services in background:

  • SSH server,
  • cloudflared tunnel
  • Slack notifier
cat << EOF > ~/cloudflare/supervisord.conf
[supervisord]
logfile=$HOME/cloudflare/supervisord.log

[program:sshd]
command=sshd -D -d -p 8022
autostart=true
autorestart=true
redirect_stderr=true
stdout_logfile=$HOME/cloudflare/sshd.log

[program:cloudflared]
command=cloudflared tunnel --loglevel debug run $TUNNEL_ID
autostart=true
autorestart=false
redirect_stderr=true
stdout_logfile=$HOME/cloudflare/cloudflared.log

[program:slack-notifier]
command=$HOME/cloudflare/slack-notifier.sh 2>&1
autostart=true
autorestart=false
redirect_stderr=true
stdout_logfile=$HOME/cloudflare/slack-notifier.log
startsecs=0
exitcodes=0

[supervisorctl]

[inet_http_server]
port = 127.0.0.1:9001

[rpcinterface:supervisor]
supervisor.rpcinterface_factory = supervisor.rpcinterface:make_main_rpcinterface
EOF

Prepare useful scripts:

cat << EOF > ~/cloudflare/Makefile
supervisor-start:
    supervisord -c $HOME/cloudflare/supervisord.conf --pidfile $HOME/cloudflare/supervisord.pid

tunnel-start:
    cloudflared tunnel --loglevel debug run $TUNNEL_ID

supervisor-status:
    supervisorctl -c $HOME/cloudflare/supervisord.conf status

supervisor-stop:
    supervisorctl -c $HOME/cloudflare/supervisord.conf shutdown
EOF

# Open Tunnel Script
cat << EOF > ~/.shortcuts/tasks/open-tunnel.sh
termux-wake-lock
supervisord -c $HOME/cloudflare/supervisord.conf --pidfile $HOME/cloudflare/supervisord.pid
termux-notification --group termux --title "Termux Tunnel" --content "Remote tunnel opened"
EOF

# Close Tunnel Script
cat << EOF > ~/.shortcuts/tasks/close-tunnel.sh
supervisorctl -c $HOME/cloudflare/supervisord.conf shutdown
termux-wake-unlock
termux-notification --group termux --title "Termux Tunnel" --content "Remote tunnel closed"
EOF

On your Android device, add widgets from Termux:Widget for:

  • ~/.shortcuts/tasks/open-tunnel.sh
  • ~/.shortcuts/tasks/close-tunnel.sh

Opening the tunnel

Tap the open-tunnel.sh widget. You should get the slack notification when the tunnel is open.

Client prerequisites

You'll need:

  • SSH client with generated public key
  • Cloudflared client
  • ADB client
  • scrcpy client

Connecting from the client

Access the Cloudflared tunnel from the client device. This opens local port 8023 to forward SSH traffic through Cloudflare to the remote server.

. setup.env
cloudflared access tcp --loglevel debug --hostname $TUNNEL_DNS_SSH --url localhost:8023
Now, connect to the SSH tunnel:
ssh $TERMUX_USER_ID@localhost -p 8023

Once you're in, open ADB port on the server for remote debugging:

adb tcpip 5555
adb devices  # This should list local emulator-5554

Open another tunnel to forward the ADB port.

cloudflared access tcp --loglevel debug --hostname $TUNNEL_DNS_ADB --url localhost:5556

Finally, connect to the remote device through the ADB tunnel:

adb connect localhost:5556
scrcpy -s localhost:5556

Connecting though SSH Proxy command

Add this to your .ssh/config (evaluate :

. ./setup.env
cat << EOF >&1
Host $TUNNEL_DNS_SSH
  ProxyCommand /usr/local/bin/cloudflared access ssh --hostname %h
  User $TERMUX_USER_ID
EOF
and then you can connect with just:
ssh $TUNNEL_DNS_SSH