mirror of https://gitlab.com/curben/blog
158 lines
5.5 KiB
Markdown
158 lines
5.5 KiB
Markdown
---
|
|
title: rsync is surprisingly simple to setup
|
|
excerpt: configure rsync on NixOS, no daemon required
|
|
date: 2021-03-15
|
|
tags:
|
|
- linux
|
|
- nixos
|
|
---
|
|
|
|
When I first tried to figure out how to deploy this website, I thought of using [rsync](https://en.wikipedia.org/wiki/Rsync) to sync pipeline-generated static files to the web server. FTP was never viable because it is inefficient to upload everything, whereas rsync uploads modified files only. rsync's `--delete` option also can automatically remove files in destination folder that are no longer exist in source folder.
|
|
|
|
At that time, I thought rsync needs to run as a server daemon on the web server and I was reluctant to open another port. That turned out to be incorrect after I read the wiki.
|
|
|
|
{% blockquote rsync https://en.wikipedia.org/wiki/Rsync %}
|
|
For example, if the command `rsync local-file user@remote-host:remote-file` is run, rsync will use SSH to connect as user to remote-host...
|
|
|
|
Rsync can also operate in a daemon mode (rsyncd), serving and receiving files in the native rsync protocol (using the "rsync://" syntax).
|
|
{% endblockquote %}
|
|
|
|
This means it's _optional_ for rsync to operate in a daemon mode. When operating over SSH, rsync first establishes an SSH connection, execute rsync on the remote server and starts syncing files from local to remote. In the above example, the direction is to mirror local changes to remote server. You could also reverse the direction, `rsync user@remote-host:remote-file local-file`, where remote changes is reflected locally; this mode is usually used by Linux distribution mirrors to sync with the primary server.
|
|
|
|
The way rsync works by default--by piggybacking on SSH--is similar to how [Mosh](https://mosh.org/) operates, except that Mosh needs to listen on a single UDP port between 60000 and 61000. rsync utilises existing SSH connection, so there is no need to open another port.
|
|
|
|
## SSH key and user setup
|
|
|
|
Generate a new SSH key. If you're planning to use rsync on CI/CD pipeline, leave the password empty.
|
|
|
|
```
|
|
ssh-keygen -t ed25519 -C "www-data@nixos-server"
|
|
```
|
|
|
|
Create a separate user with home folder set to where web server will be deployed. I use the convention of `www-data` user with `/var/www` home folder. Create
|
|
|
|
``` nix /etc/nixos/configuration.nix
|
|
users = {
|
|
users = {
|
|
www-data = {
|
|
openssh.authorizedKeys.keys = [ "ssh-ed25519 ..." ];
|
|
home = "/var/www";
|
|
createHome = true;
|
|
## Required for rsync
|
|
useDefaultShell = true;
|
|
};
|
|
};
|
|
};
|
|
```
|
|
|
|
`useDefaultShell` is required to execute rsync on the remote server. This has a security implication and requires a minor tweak to the web server; more on this in the next section. Execute `nixos-rebuild switch` as root to create `www-data` user and its home folder.
|
|
|
|
Home folder is not world-readable by default, so if you start a web server using different user, it can't access the `/var/www`. To fix this,
|
|
|
|
```
|
|
chmod +xr /var/www
|
|
```
|
|
|
|
### Hide dotfiles in web server
|
|
|
|
`useDefaultShell` grants a shell to the user and the shell may generate dotfiles to home folder (e.g. `~/.bash_history`/`~/.bashrc`). In practice, those files will be removed automatically every time rsync runs. As a precaution, you should configure the web server not to expose those dotfiles.
|
|
|
|
Example Caddy config:
|
|
|
|
``` plain Caddyfile
|
|
example.com www.example.com {
|
|
root * /var/www
|
|
file_server {
|
|
hide /var/www/.*
|
|
}
|
|
}
|
|
```
|
|
|
|
The example assumes there is no existing dotfiles that are intended to be public, like `.well_known/`; in that case, adjust the regex accordingly.
|
|
|
|
## SSH config
|
|
|
|
Add the following lines to `~/.ssh/config`:
|
|
|
|
```
|
|
Host rsync-remote
|
|
HostName x.x.x.x
|
|
User www-data
|
|
## Uncomment if using custom port
|
|
#Port 1234
|
|
IdentityFile /path/to/private/key
|
|
IdentitiesOnly yes
|
|
```
|
|
|
|
The config creates an alias `rsync-remote` and specify the private key, so that ssh/rsync destination is specified with `rsync-remote`, instead of `user@remote-host`.
|
|
|
|
## Upload test file
|
|
|
|
> rsync is included in most Linux distributions (e.g. Ubuntu, NixOS, Arch/Manjaro), but not in Alpine
|
|
|
|
Test this setup by uploading a small test file:
|
|
|
|
```
|
|
echo "test content" > test.txt
|
|
rsync -zvh test.txt rsync-remote:/var/www/
|
|
```
|
|
|
|
SSH/login into the remote server, the test file should exists in `/var/www` folder.
|
|
|
|
```
|
|
cat /var/www/test.txt
|
|
```
|
|
|
|
Once the test pass, we can move on to uploading the whole static sites. In my case, I use [Hexo](https://hexo.io/) static site generator and it generates into `public/` folder.
|
|
|
|
```
|
|
hexo generate
|
|
# Do a dry run
|
|
rsync -azvh --delete --dry-run public/ rsync-remote:/var/www/
|
|
# Actual upload
|
|
rsync -azvh --delete public/ rsync-remote:/var/www/
|
|
```
|
|
|
|
## Rsync in CI/CD pipeline
|
|
|
|
Add `SSH_KEY`, `SSH_CONFIG` and `SSH_KNOWN_HOSTS` to CI/CD variables. `SSH_KNOWN_HOSTS` value can be generated by:
|
|
|
|
```
|
|
ssh-keyscan x.x.x.x
|
|
|
|
# Custom port
|
|
ssh-keyscan -p 1234 x.x.x.x
|
|
```
|
|
|
|
``` yml .gitlab-ci.yml
|
|
build:
|
|
stage: build
|
|
|
|
before_script:
|
|
- npm install
|
|
|
|
script:
|
|
- npm run build
|
|
|
|
deploy:
|
|
stage: deploy
|
|
|
|
before_script:
|
|
- apk update && apk add openssh-client rsync
|
|
- mkdir -p ~/.ssh
|
|
- chmod 700 ~/.ssh
|
|
- echo "$SSH_KNOWN_HOSTS" > ~/.ssh/known_hosts
|
|
- chmod 644 ~/.ssh/known_hosts
|
|
## Adjust the private key path in ssh config accordingly
|
|
- echo "$SSH_KEY" > ~/.ssh/id_remote_rsync
|
|
- chmod 600 ~/.ssh/id_remote_rsync
|
|
- echo "$SSH_CONFIG" > ~/.ssh/config
|
|
- chmod 600 ~/.ssh/config
|
|
|
|
script:
|
|
## Dry run
|
|
- rsync -azvh --delete --dry-run public/ rsync-remote:/var/www/
|
|
## Remove above & uncomment below if no issue
|
|
#- rsync -azvh --delete public/ rsync-remote:/var/www/
|
|
```
|