Migrate Nextcloud to a new server with the same Fully Qualified Domain Name
Introduction
In this document is described a migration path of Nextcloud instance from one VPS to another. The migration path described here relates to a Nextcloud instance running in docker-compose along with a few other services which are not relevant for this tutorial.
The starting docker-compose.yml
is:
$ cat nextcloud/docker-compose.yml
version: '3'
services:
proxy:
image: nginxproxy/nginx-proxy:alpine
labels:
- "com.github.jrcs.letsencrypt_nginx_proxy_companion.nginx_proxy=true"
container_name: nextcloud-proxy
networks:
- nextcloud_network
ports:
- 80:80
- 443:443
volumes:
- ./proxy/conf.d:/etc/nginx/conf.d:rw
- ./proxy/vhost.d:/etc/nginx/vhost.d:rw
- ./proxy/html:/usr/share/nginx/html:rw
- ./proxy/certs:/etc/nginx/certs:ro
- /etc/localtime:/etc/localtime:ro
- /var/run/docker.sock:/tmp/docker.sock:ro
restart: unless-stopped
letsencrypt:
image: jrcs/letsencrypt-nginx-proxy-companion
container_name: nextcloud-letsencrypt
depends_on:
- proxy
networks:
- nextcloud_network
volumes:
- ./proxy/certs:/etc/nginx/certs:rw
- ./proxy/vhost.d:/etc/nginx/vhost.d:rw
- ./proxy/html:/usr/share/nginx/html:rw
- /etc/localtime:/etc/localtime:ro
- /var/run/docker.sock:/var/run/docker.sock:ro
restart: unless-stopped
db:
image: mariadb:10.6
command:
- "--innodb-read-only-compressed=OFF"
container_name: nextcloud-mariadb
networks:
- nextcloud_network
volumes:
- db:/var/lib/mysql
- /etc/localtime:/etc/localtime:ro
environment:
- MYSQL_ROOT_PASSWORD=<SOME_GREAT_PASS>
- MYSQL_PASSWORD=<ANOTHER_GREAT_PASS>
- MYSQL_DATABASE=nextcloud
- MYSQL_USER=nextcloud
restart: unless-stopped
app:
image: nextcloud:27-apache
container_name: nextcloud-app
networks:
- nextcloud_network
depends_on:
- letsencrypt
- proxy
- db
volumes:
- nextcloud:/var/www/html
- ./app/config:/var/www/html/config
- ./app/custom_apps:/var/www/html/custom_apps
- ./app/data:/var/www/html/data
- ./app/themes:/var/www/html/themes
- /etc/localtime:/etc/localtime:ro
environment:
- VIRTUAL_HOST=<FQDN>
- LETSENCRYPT_HOST=<FQDN>
- LETSENCRYPT_EMAIL=<EMAIL>
- OVERWRITEPROTOCOL=https
restart: unless-stopped
volumes:
nextcloud:
db:
networks:
nextcloud_network:
Planning ahead the migration
To do a successful migration we should do some planning ahead. The rough sketch of the steps is:
- Switch the current instance of the Nextcloud to the maintenance mode
- Perform backup of the current instance
- Install prerequisites on the new VPS
- Create a necessary folder structure on the new server
- Copy backup to the new VPS
- List all exact version of images of the running instance
- Set up a docker-compose.yml on the new instance to create all necessary docker volumes
- Restore the content from the backup to the newly created docker volumes taking care that ownership and permissions are preserved
- Start the services with the fixed versions without
letsencrypt
- Check the logs of all services and verify that all of them started without issues
- Switch the DNS record to point to the new VPS
- Wait till old TTL expires and verify that you can access the new instance
- When TTL is expired start the remaining
letsencrypt
service - If everything OK switch off the maintenance mode on the new instance
- The old instance can be safely removed now
Execution of the migration plan
1. Switch the current instance to the maintenance mode
On the old VPS execute the following command:
docker exec -it -u www-data nextcloud-app /var/www/html/occ maintenance:mode --on
Wait some time and verify that the instance is in the maintenance mode.
$ curl https://<FQDN>
...
<div class="icon-big icon-error"></div>
<h2>Maintenance mode</h2>
<p>This Nextcloud instance is currently in maintenance mode, which may take a while. This page will refresh itself when the instance is available again.</p>
<p>Contact your system administrator if this message persists or appeared unexpectedly.</p>
</div>
...
2. Backup of the current instance
$ cat >>backup_nextcloud.sh<<EOT
#!/bin/bash
set -e
cd ~/nextcloud/
docker-compose down
[ -d backup/backup_$(date +"%Y%m%d") ] || mkdir -p backup/backup_$(date +"%Y%m%d")
docker run --rm --volume nextcloud_db:/db --volume $(pwd)/backup/backup_$(date +"%Y%m%d"):/backup ubuntu tar cvfz /backup/db.tar.gz /db
docker run --rm --volume nextcloud_nextcloud:/nextcloud --volume $(pwd)/backup/backup_$(date +"%Y%m%d"):/backup ubuntu tar cvfz /backup/nextcloud.tar.gz /nextcloud
sudo tar -I pigz -cvf $(pwd)/backup/backup_$(date +"%Y%m%d")/app_data.tar.gz app/ # This can take a lot of time
sudo tar -czvf $(pwd)/backup/backup_$(date +"%Y%m%d")/proxy.tar.gz proxy/
cd ~/nextcloud/
docker-compose up -d
EOT
3. Setting up the prerequisites on the new instance
Since docker-compose
will be used to run the nextcloud on the new instance too we need to install it. To do so we will follow the official documentation from docs.docker.com.
Besides the required software we need to enable the communication between the new and old VPS. The easiest will be setting up ssh with keys.
3.1 Setup the hostname
sudo hostnamectl set-hostname <FQDN>
3.2 Setup communication from new to the old VPS
First let's generate some ssh-keys.
ssh-keygen -t ed25519
Now that the ssh keys were created, we can add it to the authorized_keys
on the old VPS.
$ ssh-copy-id -i $HOME/.ssh/id_ed25519.pub tihomir@<OLD_VPS_IP>
/usr/bin/ssh-copy-id: INFO: Source of key(s) to be installed: "/home/tihomir/.ssh/id_ed25519.pub"
The authenticity of host '<OLD_VPS_IP> (<OLD_VPS_IP>)' can't be established.
ED25519 key fingerprint is SHA256:<MASKED>.
This key is not known by any other names
Are you sure you want to continue connecting (yes/no/[fingerprint])? yes
/usr/bin/ssh-copy-id: INFO: attempting to log in with the new key(s), to filter out any that are already installed
/usr/bin/ssh-copy-id: INFO: 1 key(s) remain to be installed -- if you are prompted now it is to install the new keys
tihomir@<OLD_VPS_IP>'s password:
Number of key(s) added: 1
Now try logging into the machine, with: "ssh 'tihomir@<OLD_VPS_IP>'"
and check to make sure that only the key(s) you wanted were added.
Add ssh-agent
to startup
cat >>.bashrc<<EOT
if [ ! -S ~/.ssh/ssh_auth_sock ]; then
eval `ssh-agent`
ln -sf "$SSH_AUTH_SOCK" ~/.ssh/ssh_auth_sock
fi
export SSH_AUTH_SOCK=~/.ssh/ssh_auth_sock
EOT
Start the ssh-agent
$ eval `ssh-agent`
Agent pid 21835
Add the private ssh key to the ssh-agent
$ ssh-add /home/tihomir/.ssh/id_ed25519
Identity added: /home/tihomir/.ssh/id_ed25519 (tihomir@<FQDN>)
Test the connection to the old VPS.
ssh tihomir@<OLD_VPS_IP>
3.3 Install the prerequisites
As a start we should upgrade ubuntu to the lates version.
sudo apt update
sudo apt upgrade
Restart if necessary.
Install everyday's utilities
sudo apt install vim git screen unzip jq gcc npm python3-venv
Install the docker engine on the new VPS(running ubuntu). The commands are copied from the original site just to make the things more observable.
# Add Docker's official GPG key:
sudo apt-get update
sudo apt-get install ca-certificates curl gnupg
sudo install -m 0755 -d /etc/apt/keyrings
curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo gpg --dearmor -o /etc/apt/keyrings/docker.gpg
sudo chmod a+r /etc/apt/keyrings/docker.gpg
# Add the repository to Apt sources:
echo \
"deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/docker.gpg] https://download.docker.com/linux/ubuntu \
$(. /etc/os-release && echo "$VERSION_CODENAME") stable" | \
sudo tee /etc/apt/sources.list.d/docker.list > /dev/null
sudo apt-get update
sudo apt-get install docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-**plugin**
Install docker-compose
standalone.
sudo curl -SL https://github.com/docker/compose/releases/download/v2.23.3/docker-compose-linux-x86_64 -o /usr/local/bin/docker-compose
sudo chmod a+x /usr/local/bin/docker-compose
sudo ln -s /usr/local/bin/docker-compose /usr/bin/docker-compose
To avoid using sudo when running docker command add the user to the docker group.
sudo usermod -aG docker tihomir
4. Crate necessary directory structure on the new VPS
mkdir -p nextcloud/backup
5. Copy the backup from the old VPS the the new one
Start a new screen session
screen
rsync -a tihomir@<OLD_VPS_IP>:/home/tihomir/nextcloud/backup/backup_20231126 /home/tihomir/nextcloud/backup/
If using IPv6 enable it and make sure that you have connectivity.
6. Find out exact versions running on the current instance
Nextcloud
$ docker exec -it -u www-data nextcloud-app /var/www/html/occ -V
Nextcloud is in maintenance mode, no apps are loaded.
Commands provided by apps are unavailable.
Nextcloud 27.1.2
Based on our docker-compose.yml
and the version of the running Nextcloud we need the docker image nextcloud:27.1.2-apache
for the app service.
MariaDB
$ docker exec -it nextcloud-mariadb mariadb --version
mariadb Ver 15.1 Distrib 10.6.15-MariaDB, for debian-linux-gnu (x86_64) using readline 5.2
Based on our docker-compose.yml
and the version of the running MariaDB we need the docker image mariadb:10.6.15
for the db service.
7. Setup docker-compose.yml to create necessary volumes
Switch to the working dir
cd $HOME/nextcloud
cat >>docker-compose.yml<<EOT
version: '3'
services:
dummy:
image: alpine:latest
container_name: dummy
restart: unless-stopped
networks:
- nextcloud_network
volumes:
- nextcloud:/nc
- db:/db
volumes:
nextcloud:
db:
networks:
nextcloud_network:
EOT
Fire docker-compose to create the necessary volumes
$ docker-compose up -d
[+] Running 3/3
✔ Volume "nextcloud_db" Created 0.0s
✔ Volume "nextcloud_nextcloud" Created 0.0s
✔ Container dummy Started 0.0s
8. Restore backup
Now that we have backups transferred from the old VPS to the new one and the necessary docker volumes are created we can start restoring files from the backup.
One small remark: the username under which we work and path to docker-compose.yml
must be the same on both VMs otherwise we have to do some manual moves of the directories created below.
cd $HONE/nextcloud
sudo tar -C / -xpvzf backup/backup_20231126/proxy.tar.gz
docker run --rm --volume nextcloud_db:/db --volume $(pwd)/backup:/backup ubuntu tar -C / -xpvzf /backup/backup_20241101/db.tar.gz
docker run --rm --volume nextcloud_nextcloud:/nextcloud --volume $(pwd)/backup:/backup ubuntu tar -C / -xpvzf /backup/backup_20241101/nextcloud.tar.gz
sudo tar -C / -xpvzf backup/backup_20241101/app_data.tar.gz # This can take a lot of time, depending on the size of the backup and on the performance of the VPS
9. Start the services with fixed versions and without letsencrypt
The docker-compose.yml used in this step is:
$ cat docker-compose.yml
version: '3'
services:
proxy:
image: nginxproxy/nginx-proxy:alpine
labels:
- "com.github.jrcs.letsencrypt_nginx_proxy_companion.nginx_proxy=true"
container_name: nextcloud-proxy
networks:
- nextcloud_network
ports:
- 80:80
- 443:443
volumes:
- ./proxy/conf.d:/etc/nginx/conf.d:rw
- ./proxy/vhost.d:/etc/nginx/vhost.d:rw
- ./proxy/html:/usr/share/nginx/html:rw
- ./proxy/certs:/etc/nginx/certs:ro
- /etc/localtime:/etc/localtime:ro
- /var/run/docker.sock:/tmp/docker.sock:ro
restart: unless-stopped
# letsencrypt:
# image: jrcs/letsencrypt-nginx-proxy-companion
# container_name: nextcloud-letsencrypt
# depends_on:
# - proxy
# networks:
# - nextcloud_network
# volumes:
# - ./proxy/certs:/etc/nginx/certs:rw
# - ./proxy/vhost.d:/etc/nginx/vhost.d:rw
# - ./proxy/html:/usr/share/nginx/html:rw
# - /etc/localtime:/etc/localtime:ro
# - /var/run/docker.sock:/var/run/docker.sock:ro
# restart: unless-stopped
#
db:
image: mariadb:10.6.15
command:
- "--innodb-read-only-compressed=OFF"
container_name: nextcloud-mariadb
networks:
- nextcloud_network
volumes:
- db:/var/lib/mysql
- /etc/localtime:/etc/localtime:ro
environment:
- MYSQL_ROOT_PASSWORD=<SOME_GREAT_PASS>
- MYSQL_PASSWORD=<ANOTHER_GREAT_PASS>
- MYSQL_DATABASE=nextcloud
- MYSQL_USER=nextcloud
restart: unless-stopped
app:
image: nextcloud:27.1.2-apache
container_name: nextcloud-app
networks:
- nextcloud_network
depends_on:
# - letsencrypt
- proxy
- db
volumes:
- nextcloud:/var/www/html
- ./app/config:/var/www/html/config
- ./app/custom_apps:/var/www/html/custom_apps
- ./app/data:/var/www/html/data
- ./app/themes:/var/www/html/themes
- /etc/localtime:/etc/localtime:ro
environment:
- VIRTUAL_HOST=<FQDN>
- LETSENCRYPT_HOST=<FQDN>
- LETSENCRYPT_EMAIL=<EMAIL>
- OVERWRITEPROTOCOL=https
restart: unless-stopped
volumes:
nextcloud:
db:
networks:
nextcloud_network:
Start the docker-compose
$ docker-compose up -d
...
✔ b49eecef6de7 Pull complete 10.1s
[+] Running 4/4
✔ Network nextcloud_nextcloud_network Created 0.1s
✔ Container nextcloud-mariadb Started 1.0s
✔ Container nextcloud-proxy Started 1.0s
✔ Container nextcloud-app Started
10. Check the logs of all services and verify that all of them started without issues
docker-compose logs --tail 30 app
docker-compose logs --tail 30 db
Check if the server is reachable and in maintenance mode:
$ curl -k -H 'host: <FQDN>' https://<NEW_VPS_IP>
...
<main>
<h1 class="hidden-visually">
Nextcloud </h1>
<div class="guest-box">
<div class="icon-big icon-error"></div>
<h2>Maintenance mode</h2>
<p>This Nextcloud instance is currently in maintenance mode, which may take a while. This page will refresh itself when the instance is available again.</p>
<p>Contact your system administrator if this message persists or appeared unexpectedly.</p>
</div>
</main>
...
11. Switch the DNS record to point to the new VPS
This is out of scope of this tutorial. Please check with your DNS provider how to do it.
12. Wait till old TTL expires and verify that you can access the new instance
Again this si out of hte scope of this tutorial. DNS TTL can be set to anything between 1 minute and a few hours.
13. If everything OK switch off the maintenance mode on the new instance
After uncommenting the letsencrypt
service section in the docker-compose.yml
stop all running service in docker-compose
and start all with the service letsencrypt
enabled.
docker-compose stop
docker-compose up -d
Now the new instance should be accessible and with a valid TLS certificate. Verify the connection and if everything is OK, proceed to the last step that involves the configuration of the new instance.
14. If everything so far was OK, switch off the maintenance mode on the new instance
docker exec -it -u www-data nextcloud-app /var/www/html/occ maintenance:mode --off
Give it a bit of time and connect to the new instance.
15. The old instance can be safely removed now
Yeah, the old instance can be recycled now. As usual perform a good backup before decommissioning anything.
Conclusion
Migration path required a bit of work but it was not to troublesome. The good point was that the FQDN didn't change so we didn't have to reconfigure the clients and we cloud just restore the backup from the old instance without touching the configuration.