Self-hosting File Browser on OpenBSD
Background
I mentioned previously that I am self-hosting Syncthing for syncing my files to multiple devices and the cloud. I also wanted to access my cloud files via web browser, so I'm self-hosting File Browser as well.
These are the steps I took to set up File Browser on OpenBSD 7.6 x64, including some of the larger errors I encountered and what I did to solve (or work around) them. Building from source was... an experience. (This GitHub comment made it seem so simple. 🙃)
Newbie lesson learned: My 1 GB memory server could not build from source, so I spun up a separate 4 GB memory server to build the binary, then transferred the binary back to my original server. (I learned what a binary is from this process.)
I referred to a bunch of guides & documentation while installing this, which I'm listing below in References. Email me suggestions for improvements.
What I did
Below, when I mention $user
, it's a placeholder for actual username (can find username with echo $USER
). Every step that has only text styled like this
means that I typed/pasted it into PowerShell and pressed enter afterwards. When I use mg
to edit a file, I save & exit by pressing ctrl + x, then ctrl + c (to exit), then y to save changes.
Attempt to build File Browser (and fail)
tl;dr — Not enough memory, copy source files over to new server, build, copy binary back to original server.
ssh $user@server-ip-address
- Install these things to build from source (I added these individually as I saw errors, but compiled them together here in the beginning)
doas pkg_add gmake node go
doas pkg_add git
doas pkg_add bash
doas npm install -g pnpm
- Build from source
git clone https://github.com/filebrowser/filebrowser
cd filebrowser
gmake build
- See error:
Error: Your current platform "openbsd" and architecture "x64" combination is not yet supported by the native Rollup build. Please use the WASM build "@rollup/wasm-node" instead.
...
Node.js v20.18.2
 ELIFECYCLE  Command failed with exit code 1.
gmake: *** [Makefile:13: build-frontend] Error 1
- Address the error by installing the WASM build
cd ~/filebrowser/frontend
pnpm add -D @rollup/wasm-node
- Add override to
package.json
(I did this via SFTP because I didn't want to do edits via command line anymore)- SFTP into server with WinSCP
- Right-click and duplicate
/filebrowser/frontend/package.json
to save a backup copy - Right-click and edit
package.json
- Add
"pnpm"
override between"engines"
section and"scripts"
section:
"engines": {
...
},
"pnpm": {
"overrides": {
"rollup": "npm:@rollup/wasm-node"
}
},
"scripts": {
...
},
- Add override to
pnpm-lock.yaml
- Right-click and duplicate
/filebrowser/frontend/pnpm-lock.yaml
to save a backup copy cd ~/filebrowser/frontend
pnpm install --no-frozen-lockfile
- Right-click and duplicate
- Try to build again
cd ~/filebrowser
gmake build
- See new error:
✓ 1140 modules transformed.
<--- Last few GCs --->
[44534:0x4326d372000] 42338 ms: Mark-Compact 494.1 (515.4) -> 490.2 (515.7) MB, 1121.35 / 0.00 ms (average mu = 0.095, current mu = 0.037) allocation failure; scavenge might not succeed
[44534:0x4326d372000] 43487 ms: Mark-Compact 494.1 (515.7) -> 490.2 (515.7) MB, 1131.97 / 0.00 ms (average mu = 0.059, current mu = 0.014) allocation failure; scavenge might not succeed
<--- JS stacktrace --->
FATAL ERROR: Ineffective mark-compacts near heap limit Allocation failed - JavaScript heap out of memory
----- Native stack trace -----
...
Abort trap (core dumped)
 ELIFECYCLE  Command failed with exit code 134.
gmake: *** [Makefile:13: build-frontend] Error 134
- Spend way too long trying to address the memory error in various ways but ultimately accept that 1 GB memory is not enough to build this thing
Create new server with 4 GB memory
I wanted to temporarily upgrade my server's memory and then downgrade it later, but that wasn't a possibility. So spinning up a second $20/mo server it is. I'll destroy it after I'm done. If we throw enough money at the problem, it'll get solved, right? (The 4 GB server costed $2.71 at the time of destruction, since I ran it only for a couple days.)
- Log into Vultr.com
- Go to Products > Compute > Deploy +
- Select Shared CPU > Select the same city as other server
- Select 4 GB memory for $20/mo
- Disable Automatic Backups and confirm that yes, I understand the risks and really don't want it
- Select Configure Software
- Select OpenBSD 7.6 x64 (the blowfish again)
- Under SSH Keys, select mykey
- Leave server hostname and server label blank
- Select Deploy
SSH into root & run Tech Independence script again
I just wanted to use the script to set up SSH access & disable root account. One day I'll learn how to do this on my own.
ssh root@4gb-server-ip
yes
ftp https://sive.rs/ti.sh
sh ti.sh
- Enter my domain name
- Enter a username
y
- Enter a full name
- Enter a password
- Retype password
- Open a new PowerShell window
ssh $user@4gb-server-ip
- And then again per the instructions,
ssh $user@4gb-server-ip
- Close out of both PowerShell windows
Mirror files from original server to new server
I ended up doing this because I set up the 4gb-server the next night, and one of the node modules updated 14 hours before my second install. That module update caused new errors that I didn't want to spend time troubleshooting. Since I knew my first install was building just fine, I decided to clone it over.
- Open FreeFileSync (because I didn't know how to use Rsync yet)
- On the left, add 1gb-server's SFTP login info
- On the right, add 4gb-server's SFTP login info
- Select the green gear icon
- Go to Comparison tab and Performance improvements on the right
- Update Parallel file operations to 4 for each server (requires donation edition of app; I didn't know about this setting when I started my sync and things were painfully slow since there were so many tiny files — I ran the sync overnight)
- Go to Synchronization tab
- Under Delete and overwrite:, select Permanent (because there is no Recycle Bin on the server)
- Use FreeFileSync to mirror
/filebrowser
directory from original server to new server- Under Drag & drop for both sides, ensure the path is
/home/$user/filebrowser
- Select Synchronize
- Wait for it to complete
- Under Drag & drop for both sides, ensure the path is
Build File Browser on new server (and succeed)
ssh $user@4gb-server-ip
- Install these things to build from source (same as before)
doas pkg_add gmake node go
doas pkg_add git
doas pkg_add bash
doas npm install -g pnpm
cd filebrowser
gmake build
- See new error:
failed to load config from /home/$user/filebrowser/frontend/vite.config.ts
error during build:
Error: The service was stopped: spawn /home/$user/filebrowser/frontend/node_modules/.pnpm/@esbuild+openbsd-x64@0.24.2/node_modules/@esbuild/openbsd-x64/bin/esbuild EACCES
at /home/$user/filebrowser/frontend/node_modules/.pnpm/esbuild@0.24.2/node_modules/esbuild/lib/main.js:968:34
at responseCallbacks.<computed> (/home/$user/filebrowser/frontend/node_modules/.pnpm/esbuild@0.24.2/node_modules/esbuild/lib/main.js:622:9)
at ChildProcess.afterClose (/home/$user/filebrowser/frontend/node_modules/.pnpm/esbuild@0.24.2/node_modules/esbuild/lib/main.js:613:28)
at ChildProcess.emit (node:events:518:28)
at ChildProcess._handle.onexit (node:internal/child_process:291:12)
at onErrorNT (node:internal/child_process:483:16)
at process.processTicksAndRejections (node:internal/process/task_queues:82:21)
 ELIFECYCLE  Command failed with exit code 1.
gmake: *** [Makefile:13: build-frontend] Error 1
- Apparently a permission issue. Grant it permission to execute:
chmod +x /home/$user/filebrowser/frontend/node_modules/.pnpm/@esbuild+openbsd-x64@0.24.2/node_modules/@esbuild/openbsd-x64/bin/esbuild
gmake build
- See new error:
✓ 1140 modules transformed.
rendering chunks (1)...Segmentation fault
 ELIFECYCLE  Command failed with exit code 139.
gmake: *** [Makefile:13: build-frontend] Error 139
- Okay, so error 139 is another memory issue. Check my user's memory limits:
ulimit -a
- See that
data(kbytes)
is set to1572864 KB
(1.5 GB) - Increase to 3 GB
ulimit -d 3145728
gmake build
- IT'S BUILT! I never thought I'd get here
Temporarily run File Browser
~/filebrowser/filebrowser -r ~/files
- It's running on local host:
Listening on 127.0.0.1:8080
- Leave this window open (closing it will stop running File Browser)
- Open a new PowerShell window and SSH into 4gb-server
- Confirm that it's running
netstat -an | grep 8080
- Should see:
- Confirm that it's running
tcp 0 0 127.0.0.1.8080 *.* LISTEN
- Since it's running on localhost (127.0.0.1:8080) and I want to access it from my local computer
- Open a new PowerShell window
ssh -L 8080:127.0.0.1:8080 $user@4gb-server-ip
- Open my Firefox browser
- Go to http://127.0.0.1:8080
- IT WORKS! happy dance
- Close all the windows
Now I just need to get this back to my 1 GB server.
Set up Rsync & copy the binary back to original server
- PowerShell window 1: Generate a new SSH key for 4gb-server
ssh-keygen -t ed25519 -C "your-email@example.com"
- enter
- enter
- Confirm key was generated
ls -la ~/.ssh/id_*
- View SSH key
cat ~/.ssh/id_ed25519.pub
- New PowerShell window 2: Give 4gb-server access to 1gb-server
ssh $user@1gb-server-ip
doas mg ~/.ssh/authorized_keys
- Copy the public key from PowerShell window 1
- Starts with
ssh-ed25519
- Starts with
- Paste it it into the
authorized_keys
file and save the file
- PowerShell window 1: Log into 1gb-server from 4gb-server
ssh $user@1gb-server-ip
yes
- We're in!
exit
- Copy the binary from 4gb-server to 1gb-server
rsync -avz ~/filebrowser/filebrowser $user@1gb-server-ip:/home/$user/filebrowser/
- PowerShell window 2: Remove 4gb-server's access
doas mg ~/.ssh/authorized_keys
- Delete the key we added earlier and save the file
- Close window 1; done with the 4gb-server now and no longer need to log into it again
Get the binary
I'm sharing the binary for those who would prefer not to build from source. (Please be cautious about such things.)
- Pull the binary file
ftp https://eeeee.lol/pub/filebrowser2.32-openbsd7.6.tar.gz
- Rename the file and remove the tar.gz extension
mv filebrowser2.32-openbsd7.6.tar.gz filebrowser
- Make the filebrowser directory & move the binary in
cd
mkdir ~/filebrowser
mv filebrowser ~/filebrowser/
Run File Browser on original server
- Update permissions so that the binary can be executed
chmod +x ~/filebrowser/filebrowser
- Confirm that it runs properly after the copy
~/filebrowser/filebrowser
- Should see this error since port 8080 is already taken:
2025/03/21 05:01:33 Warning: filebrowser.db can't be found. Initialing in /home/$user/filebrowser/
2025/03/21 05:01:33 Using database: /home/$user/filebrowser/filebrowser.db
2025/03/21 05:01:33 No config file used
2025/03/21 05:01:33 listen tcp 127.0.0.1:8080: bind: address already in use
- Assign IP and set port to 8082
cd ~/filebrowser
rm ~/filebrowser/filebrowser.db
~/filebrowser/filebrowser config init -a '127.0.0.1'
~/filebrowser/filebrowser users add username password --perm.admin
- Since we just deleted the database, we need to recreate an admin — this is for logging into the File Browser web GUI
username
should be a new unique usernamepassword
should be a new unique generated password
~/filebrowser/filebrowser config set -p '8082'
- Should see:
Server:
...
Port: 8082
...
Address: 127.0.0.1
- Run File Browser and confirm it works as expected
~/filebrowser/filebrowser
- Should see:
Listening on 127.0.0.1:8082
- Close the window
Set up File Browser as a service
- Create a new
_filebrowser
daemon userdoas useradd -g =uid -s /sbin/nologin -d /nonexistent _filebrowser
- Move File Browser to /usr/local/bin with the other executable binaries and make root the owner
doas mv ~/filebrowser/filebrowser /usr/local/bin/
doas chown root:wheel /usr/local/bin/filebrowser
- Move filebrowser.db to /var/db with the other dbs
doas mkdir -p /var/db/filebrowser
doas mv ~/filebrowser/filebrowser.db /var/db/filebrowser/
- Update permissions to db
doas chown _filebrowser:$user /var/db/filebrowser/filebrowser.db
doas chmod 660 /var/db/filebrowser/filebrowser.db
doas chown _filebrowser:$user /var/db/filebrowser/
doas chmod 770 /var/db/filebrowser
- Give File Browser access to /mnt (same group as in Syncthing setup)
doas usermod -G storagegroup _filebrowser
id _filebrowser
- See the group on the list:
9999(storagegroup)
- Confirm File Browser still runs after all these changes
filebrowser -d /var/db/filebrowser/filebrowser.db -r /mnt
-d
specifies which db it should use-r
specifies what folder it should display
- Create a rc.d service script (modeled after the Miniflux and Syncthing versions of the same file)
doas mg /etc/rc.d/filebrowser
- Add the following info and save:
#!/bin/ksh
daemon="/usr/local/bin/filebrowser"
daemon_user="_filebrowser"
daemon_flags="-d /var/db/filebrowser/filebrowser.db -r /mnt"
. /etc/rc.d/rc.subr
rc_bg=YES
rc_reload=NO
rc_cmd $1
- Update permissions so it's executable
doas chmod +x /etc/rc.d/filebrowser
Run File Browser as a service
doas rcctl enable filebrowser
doas rcctl start filebrowser
- Should see:
filebrowser(ok)
- IT'S ALIVE
- Update permissions now that the daemon will run things moving forward
doas chown _filebrowser:_filebrowser /var/db/filebrowser/filebrowser.db
doas chown _filebrowser:_filebrowser /var/db/filebrowser/
doas chmod 740 /var/db/filebrowser/filebrowser.db
doas chmod 750 /var/db/filebrowser
- Update permissions for sync directory, so File Browser has access to /mnt in the ways it needs
doas chmod 770 /mnt
doas chmod g+s /mnt
find /mnt -type d -exec doas chmod 770 {} +
find /mnt -type d -exec doas chmod g+s {} +
find /mnt -type f -exec doas chmod 660 {} +
- Check permissions for folders and files
ls -ld directoryname
- Should see:
drwxrws---
- Read + write + execute
- Should see:
ls -l filename
- Should see:
-rw-rw----
- Read + write
- Should see:
Set up subdomain + security certificate
- Set up filebrowser.example.com subdomain
doas mg /etc/relayd.conf
- Add
table <filebrowser> { 127.0.0.1 }
to the existing list - Add
pass request quick header "Host" value "filebrowser.example.com" forward to <filebrowser>
to the existing list - Add
forward to <filebrowser> port 8082
to the existing list
- Add
- Set up security certificate for filebrowser.example.com subdomain
doas mg /etc/acme-client.conf
- Add
filebrowser.example.com
- Add
- Update and apply new security certificate
doas su
domain=yourdomain.com
acme-client -v $domain
rcctl restart relayd
Access File Browser via web GUI
- Go to filebrowser.example.com
- Log in with the username and password we created earlier
- Create new test folder and test file
- Confirm new folder and file are in /mnt
find /mnt
- See them listed
Clean up my servers
tl;dr — Delete everything!
- Destroy 4gb-server on Vultr.com
- Remove all those things I installed to build from source on 1gb-server
doas pkg_delete gmake node go git bash
- See note:
--- -git-2.46.1 -------------------
You should also run /usr/sbin/userdel _gitdaemon
You should also run /usr/sbin/groupdel _gitdaemon
--- -node-20.18.2v0 -------------------
You may wish to remove /usr/local/lib/node_modules/npm/man from man.conf
Error deleting directory /usr/local/lib/node_modules: Directory not empty
- Remove lingering files
doas /usr/sbin/userdel _gitdaemon
doas /usr/sbin/groupdel _gitdaemon
doas rm -rf /usr/local/lib/node_modules
rm -rf ~/.npm
rm -rf ~/.cache/pnpm
doas rm -rf ~/go
rm -rf ~/.local/share/pnpm
rm -rf ~/.local/state/pnpm
rm -rf ~/.cache/pnpm
rm -rf ~/.cache/go-build
rm -rf ~/filebrowser
References
I ran a lot of searches on error codes; I haven't included all the pages I looked at, only the ones that helped with errors or provided other useful information. I also used ChatGPT as a "thought partner" to better understand the errors I was seeing and my options for troubleshooting.
- File Browser stuff
- Researching errors
- OpenBSD stuff
- Rsync stuff
Notes
- Updated
m
andm-x
shortcuts to include starting/stopping File Browser (along with Syncthing) - I don't like this: Content Security Policy