This is a PoC for using pg_upgrade
inside Docker -- learn from it, adapt it for your needs; don't expect it to work as-is!
(Source for this image is available at https://github.com/nobidev/docker-postgres-upgrade.)
Tags of this image are of the format OLD-to-NEW
, where OLD
represents the version of PostgreSQL you are currently running, and NEW
represents the version of PostgreSQL you would like to upgrade to.
In order to get good performance, it is recommended to run this image with docker run image --link
(see pg_upgrade
documentation for more details).
For this to be feasible, your directory structure should look something like this: (if yours does not, either adjust it, or scroll down to see the alternate pattern for running this image)
$ find DIR -mindepth 2 -maxdepth 2
DIR/OLD/data
DIR/NEW/data
$ docker run --rm \
-v DIR:/var/lib/postgresql \
nobidev/postgres-upgrade:OLD-to-NEW \
--link
...
More concretely, assuming OLD
of 9.4
, NEW
of 9.5
, and DIR
of /mnt/bigdrive/postgresql
:
$ find /mnt/bigdrive/postgresql -mindepth 2 -maxdepth 2
/mnt/bigdrive/postgresql/9.4/data
/mnt/bigdrive/postgresql/9.5/data
$ docker run --rm \
-v /mnt/bigdrive/postgresql:/var/lib/postgresql \
nobidev/postgres-upgrade:9.4-to-9.5 \
--link
...
(which assumes that your previous postgres:9.4
instance was running with -v /mnt/bigdrive/postgresql/9.4/data:/var/lib/postgresql/data
, and your new postgres:9.5
instance will run with -v /mnt/bigdrive/postgresql/9.5/data:/var/lib/postgresql/data
)
If your two directories (denoted below as PGDATAOLD
and PGDATANEW
) do not follow this structure, then the following may also be used (but will be slower):
$ docker run --rm \
-v PGDATAOLD:/var/lib/postgresql/OLD/data \
-v PGDATANEW:/var/lib/postgresql/NEW/data \
nobidev/postgres-upgrade:OLD-to-NEW
...
More concretely, assuming OLD
of 9.4
, NEW
of 9.5
, PGDATAOLD
of /mnt/bigdrive/postgresql-9.4
, and PGDATANEW
of /mnt/bigdrive/postgresql-9.5
:
$ docker run --rm \
-v /mnt/bigdrive/postgresql-9.4:/var/lib/postgresql/9.4/data \
-v /mnt/bigdrive/postgresql-9.5:/var/lib/postgresql/9.5/data \
nobidev/postgres-upgrade:9.4-to-9.5
...
(which assumes that your previous postgres:9.4
instance was running with -v /mnt/bigdrive/postgresql-9.4:/var/lib/postgresql/data
, and your new postgres:9.5
instance will run with -v /mnt/bigdrive/postgresql-9.5:/var/lib/postgresql/data
)
Putting it all together:
$ mkdir -p postgres-upgrade-testing
$ cd postgres-upgrade-testing
$ OLD='9.4'
$ NEW='9.5'
$ docker pull "postgres:$OLD"
$ docker run -dit \
--name postgres-upgrade-testing \
-e POSTGRES_PASSWORD=password \
-v "$PWD/$OLD/data":/var/lib/postgresql/data \
"postgres:$OLD"
$ sleep 5
$ docker logs --tail 100 postgres-upgrade-testing
$ # let's get some testing data in there
$ docker exec -it \
-u postgres \
postgres-upgrade-testing \
pgbench -i -s 10
$ docker stop postgres-upgrade-testing
$ docker rm postgres-upgrade-testing
$ docker run --rm \
-v "$PWD":/var/lib/postgresql \
"nobidev/postgres-upgrade:$OLD-to-$NEW" \
--link
$ docker pull "postgres:$NEW"
$ docker run -dit \
--name postgres-upgrade-testing \
-e POSTGRES_PASSWORD=password \
-v "$PWD/$NEW/data":/var/lib/postgresql/data \
"postgres:$NEW"
$ sleep 5
$ docker logs --tail 100 postgres-upgrade-testing
$ # can now safely remove "$OLD"
$ sudo rm -rf "$OLD"