これは、PostgreSQL Advent Calendar 2022 の5日目の記事です。
下記のリリースノートの項目のとおり、PostgreSQL15でオンライン物理バックアップの排他モードが削除されました。排他モードが削除された理由として、リリースノートには「このモードでデータベース・サーバが突然停止すると、サーバの起動に失敗する可能性があります」とありますが、今回は具体的にどのようにその問題が発生するか確認します。
Remove long-deprecated exclusive backup mode (David Steele, Nathan Bossart)
《機械翻訳》長く非推奨排他バックアップモードを削除しました。 (David Steele, Nathan Bossart)If the database server stops abruptly while in this mode, the server could fail to start. The non-exclusive backup mode is considered superior for all purposes. Functions pg_start_backup()/pg_stop_backup() have been renamed to pg_backup_start()/pg_backup_stop(), and the functions pg_backup_start_time() and pg_is_in_backup() have been removed.
《機械翻訳》このモードでデータベース・サーバが突然停止すると、サーバの起動に失敗する可能性があります。 非排他バックアップ・モードは、すべての目的に対して優れていると考えられます。 関数pg_start_backup()/pg_stop_backup()はpg_backup_start()/pg_backup_stop()に名前が変更され、関数pg_backup_start_time()pg_is_in_backup()は削除されました。
この問題は、排他モードを利用できるPostgreSQL14以前で発生しますが、今回はあえて、「PostgreSQL15以降でも排他モードをサポートするpg_exclusive_backupエクステンション」を使ってPostgreSQL15上で再現してみます。
ちなみに、排他モードと非排他モードの違いや、排他モードが削除された背景などについては、PostgreSQLアンカンファレンスでの発表資料に説明がありますので、ぜひ参照してください。
PostgreSQL15上のpg_exclusive_backupの準備
PostgreSQL15が稼働する環境上で、こちらを参考にpg_exclusive_backup 1.0をインストールします。ソースアーカイブファイルはここから入手します。
$ tar zxf pg_exclusive_backup-1.0.tar.gz
$ cd pg_exclusive_backup-1.0/
$ make USE_PGXS=1
$ make USE_PGXS=1 install
インストールできたら、CREATE EXTENSIONコマンドでpg_exclusive_backupを登録します。
$ psql -c "CREATE EXTENSION pg_exclusive_backup"
排他モードによるバックアップ取得手順の確認
問題を再現させる前に、まずは排他モードによるバックアップ取得手順を確認します。以下の実行例のような手順になります。
- バックアップを開始する。
$ psql -c "SELECT pg_start_backup('test', true)"
- DBデータのバックアップを取得する。
$ cp -a $PGDATA /backup/
- バックアップを終了する。
$ psql -c "SELECT pg_stop_backup()"
排他モードによるサーバ起動失敗問題の再現
排他モードによる問題を再現するには、バックアップの開始から終了までの間に、WALファイルの切り替えとチェックポイントを複数回発生させるなどして、バックアップ開始時点のWALファイルが削除されるようにします。その状態で(バックアップ終了の前に)PostgreSQLを異常終了させ、PostgreSQLをクラッシュリカバリで起動させます。以下、実行例です。
- バックアップを開始する。
$ psql -c "SELECT pg_start_backup('test', true)"
- 1.のpg_start_backup()で作成されるbackup_labelファイルから、バックアップ開始時点のWALファイルを確認する。以下では、000000010000000000000002がバックアップ開始時のWALファイルと確認できる。
$ cat $PGDATA/backup_label START WAL LOCATION: 0/2000028 (file 000000010000000000000002) CHECKPOINT LOCATION: 0/2000060 BACKUP METHOD: pg_start_backup BACKUP FROM: primary START TIME: 2022-12-05 15:02:31 JST LABEL: test START TIMELINE: 1
- WALファイルの切り替えとチェックポイントを複数回実行する。
$ psql -c "SELECT pg_switch_wal(); CHECKPOINT" $ psql -c "SELECT pg_switch_wal(); CHECKPOINT" $ psql -c "SELECT pg_switch_wal(); CHECKPOINT" $ psql -c "SELECT pg_switch_wal(); CHECKPOINT"
- バックアップ開始時のWALファイルが削除されていることを確認する。まだ削除されていない場合は3.に戻る。
$ ls $PGDATA/pg_wal 000000010000000000000006 000000010000000000000007 archive_status
- (バックアップを終了する前に)postmasterプロセスにSIGKILLを送信して、PostgreSQLを異常終了させる。
$ kill -SIGKILL $(head -1 data/postmaster.pid)
- PostgreSQLを起動する。クラッシュリカバリがエラーとなり、起動が失敗する。
$ pg_ctl start ... LOG: database system was interrupted; last known up at 2022-12-05 15:04:30 JST LOG: invalid checkpoint record FATAL: could not locate required checkpoint record HINT: If you are restoring from a backup, touch "/System/Volumes/Data/opt/pgsql-15.1/data/recovery.signal" and add required recovery options. If you are not restoring from a backup, try removing the file "/System/Volumes/Data/opt/pgsql-15.1/data/backup_label". Be careful: removing "/System/Volumes/Data/opt/pgsql-15.1/data/backup_label" will result in a corrupt cluster if restoring from a backup. LOG: startup process (PID 65731) exited with exit code 1 LOG: aborting startup due to startup process failure LOG: database system is shut down
6.では、$PGDATA内にbackup_labelファイルが存在する状態でPostgreSQLを起動することになります。この状態では、backup_labelに記録されているバックアップ開始時点のWALファイル(上記では000000010000000000000002)からPostgreSQLはクラッシュリカバリを開始しようとしますが、そのWALファイルはチェックポイントにより削除済のため、リカバリを開始できず、起動は失敗します。
この起動失敗が排他モード特有の問題で、排他モードが非推奨とされてPostgreSQL15で削除された理由のひとつになります。一方で、非排他モードでは、バックアップ開始時にbackup_labelが$PGDATA配下に作成されないため、この問題は発生しません。
If you are not restoring from a backup, try removing the file "/System/Volumes/Data/opt/pgsql-15.1/data/backup_label".
ちなみに、6.の状態から起動を成功させるには、起動失敗時のHINTメッセージ(上記抜粋)に従ってbackup_labelを削除します。これにより、PostgreSQLは(バックアップ開始時点ではなく)最新のチェックポイント時点のWALファイルからクラッシュリカバリを開始でき、起動に成功します。
$ rm $PGDATA/backup_label
$ pg_ctl start
...
LOG: database system is ready to accept connections
さいごに
オンライン物理バックアップの排他モードは、今回確認したようなサーバ起動失敗のリスクがあることなどから、PostgreSQL15で削除されました。このため、PostgreSQL14以前で排他モードを使っていたユーザは、PostgreSQL15以降へのアップグレード時に非排他モードへの移行を検討する必要があります。
一方で、排他モードの方が便利なケースもあり、pg_exclusive_backupを使ってPostgreSQL15以降でも排他モードを使う場合は、サーバ起動失敗のリスクを十分に考慮した上で使いましょう。