Ditching ngrok for frp
Sunday, 18 August 2024
Intro
(feel free too skip it if you’re here just for the config)
In recent weeks, I’ve been spending a lot of time building on the backend (and like it a lot!). After the initial spike where I was running stuff on my laptop, time came to actually expose the server to the outside world, so that my friend’s frontend (hosted on Vercel) will be able to reach it.
Ah, the age old question, once again. How to expose my thing to the world?
A minute or two to get the container up and running on Google Cloud Run seems to not be that long, but compared to ~5 seconds it takes to rebuild my server image locally, it’s a night and day difference.
So, I reached for the old, good ngrok – or is it? Well, last time I used it a few years ago, it was this cute little tool. Now it seems to have grown into something much bigger, that I don’t care enough about to understand, and gives me low-key enshittification vibes. Also, I ain’t paying $10/month for stable IP address just to tunnel public traffic to my laptop.1
Meet frp
I went on the lookout for a free, self-hosted alternatives, and
quickly found the
frp
project (Fast Reverse Proxy). It was exactly
what I needed.
To get started, download the latest frp release for your
machine from
here. Once you unzip, you should find two binaries in there: -
frps
- Fast Reverse Proxy server - frpc
- Fast Reverse Proxy client
It was mildly annoying to me is that neither
frps
norfrpc
are available on apt and Homebrew. I’ll try to get frp on Homebrew when I’ll be sufficiently bored and free.
Config
Armed with knowledge from frp
’s README and some two
blogposts,
I started configuring it on my VPS with stable IP.
On the public server
Do this setup on the server that has a stable IP address. For
simplicity, I decided to keep things in my user’s
$HOME
:
~
├── frp/
│ ├── frps
│ └── frps.toml
│ └── frps.service
frps
is the binary, and frps.toml
is
literally a single line:
bindPort = 7000
This is frps.service
:
[Unit]
Description=Fast Reverse Proxy server
After=network-online.target
Wants=network-online.target
[Service]
Type=simple
ExecStart=%h/frp/frps -c %h/frp/frps.toml
Restart=on-failure
RestartSec=10s
[Install]
WantedBy=default.target
Heads up: to be able to use %h
in
systemd unit file – which expands to /home/charlie
(assuming your user is charlie
) – this service has to
run as a user
service, not a system one. Also, you have to enable lingering
($ loginctl enable-linger
) to avoid our
service being killed on user logout.
With that out of the way, let’s register the service in systemd:
cp ~/.frp/frps.service ~/.config/systemd/user
Enable it to always run on startup:
systemctl --user enable frps.service
And start it right now:
systemctl --user start frps.service
Even though I’m not a fan of systemd, I have a slight preference for it over containers when it comes to self hosting, so this config is systemd-based. The simpler, the better.
On the local server
Do this setup on the server that cannot be reached from the public internet (e.g. because of being behind NAT or firewalls) – such as your laptop.
frpc.toml
I put this file in ~/.config/frp
.
serverAddr = "106.248.27.132"
serverPort = 7000 # connect to frp server running on this port
[[proxies]]
name = "my-tcp-proxy"
type = "tcp"
localIP = "127.0.0.1" # my server runs on localhost
localPort = 2137 # my server runs on this port on localhost
remotePort = 8080 # this port will be exposed on public server
Then run the frp client – frpc
– and specify the
config file:
frpc -c ~/.config/frp/frpc.toml
Now open a new terminal tab and run your server, listening on
the port you specified in fprc.toml
(2137 in our
case). If you don’t have any backend at hand, you can use Eli
Bendersky’s static fileserver:
go run github.com/eliben/static-server@latest --addr localhost:2137
Test it
Assuming IP of your server is 106.248.27.132, open
http://106.248.27.132:8080
in the browser.
You should be able to see contents of the directory you run
static-server
in.
Wrapping up
The whole thing is hosted on the cheapest $5/month VPS from
DigitalOcean, and it works perfectly. Now, whenever I start
developing my backend locally, but want to also have it exposed to
the grand World Wide Web, I use frpc
.
I didn’t bother to set up TLS, so it’s only http
for now, but that’s fine for my use case. I might get to it in the
future.
2024-12-30 – Gradle as task runner
in Flutter projects
2024-11-12 – Liminal spaces
2024-08-18
– Ditching ngrok for frp
2024-08-16 – Cirrus CI is the
best CI system out there
2024-08-14 – Going to Berlin for
Droidcon/Fluttercon
2024-06-25 – I was awarded Google Open Source Peer
Bonus
2024-06-04 – My journey to Google I/O
’24
2024-05-11 – GitHub Actions
beg for a supply chain attack
2024-03-19 – Writing a custom Dart
VM service extension (part 1)
2024-02-08 – On using smartphone for things
that make sense
2023-11-30 – Semantics in Flutter - under
the hood
2023-11-25 – Flutter Engine notes
2023-09-17 – Creating
and managing Android Virtual Devices using the terminal
2023-05-27 – Suckless
Android SDK setup
2023-05-26 – Let’s start over
2023-05-21 –
Short thought on “The Zen of
Unix”
2023-05-15 – Notes about “flutter
assemble”
2019-01-07 – Google Code-in
2018
If you want ngrok-like experience and don’t want to self-host, check out Cloudflare Tunnels. Thanks Luis Duarte for pointing this out.↩︎