šļøš Zero Trust Samba with Cloudflare Private Routing
The Portuguese immigration process loves paperwork. I purchased a printer for the first time in years when I moved here and, now that I have been here for a while, the scanner function does a lot of work converting stamped papers back to digital copies. Thatās fine and sometimes my brain prefers knowing that I can physically hold the original copies of my documents.
I still want a digital back-up though. I use iCloud for most things, but Iād like to have control over a redundant copy of those files. A USB drive isnāt sufficient - I need to grab these (mostly PDF) files periodically from different locations, like when Iām standing in line at a SEF office, and I donāt want to carry the drive with me.
Instead, Iām going to set up my own Samba server, move the files there, and connect to them when I need them. I still want the level of identity-driven control that a SaaS equivalent like iCloud or Google Drive offers me. Not only that, but I want to add redundant identity options in case one has an incident. This tutorial walks through how to use Cloudflare for Teams to accomplish all the above - at no cost with the free plan.
šÆ I have a few goals for this project:
- Deploy a Samba file server on an Ubuntu machine
- Make that machine inaccessible to the Internet
- Connect to that file server without adding back hauls that degrade performance
- Control who can reach the server using my identity provider of choise, instead of relying entirely on simple password auth in Samba
šŗļø This walkthrough covers how to:
- Deploy a Samba file server on an Ubuntu machine
- Use Cloudflare Tunnel to create a private network and connect that server to Cloudflareās network
- Connect to that server through Cloudflareās network using the WARP agent
ā²ļøTime to complete: ~45 minutes
š I work there. I work at Cloudflare. Several of my posts on this blog that discuss Cloudflare focus on building things with Cloudflare Workers. Iām a Workers customer and pay my invoice to use it. However, this experiment uses products where Iām on the team (Access, Gateway, WARP, and Tunnel).
Samba
The Server Messaging Block (SMB) protocol provides connectivity from client device to file shares, printers, and other destinations. The vast majority of SMB use cases depend on Microsoft Windows.
What about the rest of the operating systems out there? 30 years ago, Andrew Trigdell first built Samba to provide for SMB connectivity from UNIX-like and BSD systems, like macOS. Samba is free, GNU-licensed, software and Iām going to run it on my Ubuntu machine and use the native Samba client on my Mac to connect.
I have an Ubuntu VM, running in Digital Ocean, where Iām going to run a Samba file server. You could repeat this process with other operating systems.
First, Iāll install Samba using apt
. I followed the directions available from Ubuntu here with a couple of modifications beginning on step 3, which Iāll call out here.
I have poor user hygiene and am running as root, so I created the Samba directory with this command.
mkdir /root/sambashare/
Next, I can edit the configuration of the Samba service.
sudo vim /etc/samba/smb.conf
Iāll add the following five lines to the end of the configuration file.
[sambashare]
comment = Samba on Ubuntu
path = /root/sambashare
read only = no
browsable = yes
I can now add a user and set a password. Iāll be relying on my identity provider login, but this password gives me an additional second factor if I want to think of it that way. Again, Iām running as root and Samba requires the user to also be a user on the machine, so I need to run the following command.
sudo smbpasswd -a root
Alright, I can now restart the service, update the firewall, and check its status.
sudo service smbd restart
sudo ufw allow samba
sudo service smbd status
Now that my Samba server is up and running, I need to connect to it. I do not want this service exposed to the public Internet, so I want to reach it through a private network. However, most private networks trust all users inside of them. I want to only let certain users in this private network reach this service (and no other services). To do that, Iām going to use Cloudflare Tunnel and Zero Trust network rules in Cloudflare Gateway.
Cloudflare Tunnel
Cloudflare Tunnel connects applications, resources, and networks to Cloudflareās global network without requiring me to open up holes in my own firewall. The Cloudflare Tunnel daemon, cloudflared
, will create outbound-only connections to Cloudflare. The daemon is open-sourced and releases are available for several different operating systems.
Cloudflare Tunnel can be used for public-facing applications, internal resources that need Zero Trust rules, or in a deployment similar to a private network, like what Iām doing in this tutorial.
Install and Authenticate
First, Iām going to install cloudflared
.
sudo wget https://bin.equinox.io/c/VdrWdbjqyF/cloudflared-stable-linux-amd64.deb
sudo dpkg -i ./cloudflared-stable-linux-amd64.deb
Next, Iāll authenticate cloudflared
into my Cloudflare account.
cloudflared login
Iāll follow that link in a browser, login to my Cloudflare account, and choose a site in my account. This part is a little clunky (and weāre going to fix this) but I can pick any site listed for this private networking use case. The certificate downloaded will be account-wide.
Once selected, I can return to my terminal and confirm that the instance of cloudflared
has received the certificate.
Configure and Connect
Now that I have authenticated cloudflared
, I can create my Tunnel. The following command will create a Cloudflare Tunnel - but it wonāt run it just yet, weāll do that after we configure it.
cloudflared tunnel create smb-machine
The command above created a Tunnel in my account and issued credentials for that Tunnel to this instance of cloudflared
. The Tunnel I created is not ephemeral or dependent on cloudflared
to be running. For example, when cloudflared
restarts, I do not need to recreate this Tunnel.
Next, I am going to configure the private network functionality of Cloudflare Tunnel. The command below will tell Cloudflare to send traffic inside of my private network, bound for the specified IP CIDR, to the Tunnel I just created.
cloudflared tunnel route ip add 10.0.0.4/32 smb-machine
I can now finish configuring the Tunnel itself. Iām going to create a configuration file and edit it (in Vim) with the following command.
vim /root/.cloudflared/config.yaml
Inside of that file, Iāll add the following lines. The tunnel:
value is the ID of the Tunnel I created earlier and the credentials-file:
value is the location of the credentials file for that Tunnel.
tunnel: 3c152f92-62d0-4195-b4e9-213e5b93fb5b
credentials-file: /root/.cloudflared/3c152f92-62d0-4195-b4e9-213e5b93fb5b.json
warp-routing:
enabled: true
Iām going to knock out one final step. You might notice that the Samba server is not listening on the IP address that I just configured. To handle that, Iām going to run the following command to configure my network interface to receive traffic for that IP. You might not need this step if your service is available at the IP you configured.
ifconfig lo:0 10.0.0.4 up
ifconfig
commands do not survive a restart, so Iāll go back later and make sure to add this to my etc/network/interfaces
file, but for now this allows me to test.
Run
Alright, Iām ready to run the Tunnel. I could run this Tunnel in a one-off way, to test the functionality, with the command below:
cloudflared tunnel run smb-machine
However, I want access to this file server even if cloudflared
or the Ubuntu machine restarts, so Iām going to run cloudflared
as a service.
sudo cloudflared service install
sudo systemctl enable cloudflared
With that, cloudflared
is now connected to Cloudflareās network, ready to receive traffic bound for 10.0.0.4/32
, and able to survive a restart.
Building Zero Trust rules
The only users able to connect to that private IP will have to first enroll in my account. I can control who can enroll in my account (more on that later), but imagine I am part of a large organization. I donāt want just anyone in that organization to be able to reach this resource.
Iām going to build a rule that only allows me to connect. I could modify this to allow users in a specific Okta group or a list of users, but in this case itās just me.
To create that rule, Iāll navigate to the Cloudflare for Teams dashboard. In the dashboard, Iām going to open the Gateway
page and find the Network
tab.
First, Iāll create a rule to allow my user identity to reach the IP specified.
Next, Iām going to create a second rule to block everyone else.
These rules are enforced in top-to-bottom order, so I need to make sure the Allow rule is listed first.
Connecting from WARP
I can now enroll my client machine into this private network and connect to the Samba server.
Configuring enrollment
First, I need to define who can enroll into my private network using the WARP agent. To do that, Iām going to stay in the Cloudflare for Teams dashboard and open the Settings
page.
Iāll select Device
settings.
Iāll configure the enrollment rules by clicking Manage. I already have configured a handful of identity providers, but if you havenāt yet then this guide will help you get started.
Iām going to only allow myself to enroll, but I could add rules to allow my entire team, identity provider groups, or even users from multiple identity providers.
Once saved, I can leave this page and configure my accountās network settings.
Configuring my network
Second, I need to configure the settings that will be applied when users enroll. Iāll stay in the Settings
page, but navigate over to the Network
section. In this case, I need to make sure that TLS inspection
and the Proxy
mode settings are both enabled.
One last thing here - the WARP agent, which is going to be my on-ramp to connect to this resource, excludes a list of private IP ranges by default. I need to delete any ranges that include the IP I configured previously.
Enrolling my device
Iāll begin the third step by adding a certificate to my device. Iāll navigate to the Certificates
card of the Devices
page in the Settings
section to download the certificate, then Iāll follow these instructions to add the certificate to my machine.
The WARP agent, and this private routing use case, can work alongside Cloudflareās Secure Web Gateway, which performs traffic inspection using the certificate installed above. Weāre going to make this easier and remove this requirement for purely private routing use cases.
I can now download the WARP agent (links are available in the same Device settings page).
Once installed, I need to enroll the agent into my Cloudflare account.
To do so, Iāll click the gear icon and navigate to the Account view.
Iām going to input my Cloudflare for Teams name; if you donāt remember this value, you can find it in the General
page of the Settings
section. Once entered, Iāll be prompted to authenticate with my identity provider.
Finally, I need to make sure the agent is running in āWARPā mode - the proxy version - rather than just DNS mode.
For large organizations, these steps can be completed via an MDM deployment to avoid requiring users to manually complete them.
Connecting to the Server
With WARP enabled, I can now connect to the Samba share for my device. Iāll input the IP of the resource.
Iāll authenticate with the credentials created earlier.
And Iām connected! Behind the scenes, Cloudflare checked that my user identity (stored in WARP) is allowed to reach this IP and connected me through.
Whatās next?
I like this. I now have a private file storage that doesnāt rely on any one consumer-focused service. I can also keep my device safe by adding Secure Web Gateway or DNS filtering rules with just a few clicks. I can access it from any location (and even see logs).
That said, a few of these steps could be consolidated and easier (or removed for this specific use case). Weāre going to work on that next. Iām also going to move the Samba server to a machine I run in a safe location.
Donāt forget your public IPs The goal of this setup is that nothing is exposed to the Internet; Iām going to configure my Droplet to block inbound connections to its public IP. If I need to reach it again, Iāll use the SSH functionality of Cloudflare Tunnel.