Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Unset NOTIFY_SOCKET when running the temporary server. #1325

Merged
merged 2 commits into from
Feb 27, 2025

Conversation

plietar
Copy link
Contributor

@plietar plietar commented Feb 25, 2025

Some container runtimes, such as podman, can expose a notify socket when used in combination with systemd. The socket's path is available in the NOTIFY_SOCKET envirionment variable.

Postgres has native support for this notification socket, and will write a READY=1 message once it is ready and accepting connections. Unfortunately, the temporary server used by the docker-entrypoint.sh to populate the database also sends a message on the socket, making it appear as though the container is ready and serving connections when it is not.

To avoid this issue we need to unset the NOTIFY_SOCKET envirionment variable before starting the temporary server. The real server will eventually find the socket and send the appropriate message.

The easiest way to test this is through the systemd-run command, which creates a transient service:

systemd-run --user --service-type=notify \
  podman run -e POSTGRES_PASSWORD=password postgres

When looking at the logs for this command, prior to this change systemd prints a "Started" message mid-way through the container startup (I've omitted a lot of irrelevant lines for the sake of brevity):

systemd[2970]: Starting run-r916adefbc19c4465a1b0afc892980bd6.service - /usr/bin/podman run --log-driver none -e POSTGRES_PASSWORD=password postgres...
[...]
podman[851405]: waiting for server to start....2025-02-25 17:37:39.378 UTC [48] LOG:  starting PostgreSQL 17.4 (Debian 17.4-1.pgdg120+2) on x86_64-pc-linux-gnu, compiled by gcc (Debian 12.2.0-14) 12.2.0, 64-bit
podman[851405]: 2025-02-25 17:37:39.381 UTC [48] LOG:  listening on Unix socket "/var/run/postgresql/.s.PGSQL.5432"
podman[851405]: 2025-02-25 17:37:39.393 UTC [51] LOG:  database system was shut down at 2025-02-25 17:37:39 UTC
podman[851405]: 2025-02-25 17:37:39.400 UTC [48] LOG:  database system is ready to accept connections
systemd[2970]: Started run-r916adefbc19c4465a1b0afc892980bd6.service - /usr/bin/podman run --log-driver none -e POSTGRES_PASSWORD=password postgres.
podman[851405]:  done
podman[851405]: server started
podman[851405]: /usr/local/bin/docker-entrypoint.sh: ignoring /docker-entrypoint-initdb.d/*
podman[851405]: waiting for server to shut down...2025-02-25 17:37:39.520 UTC [48] LOG:  received fast shutdown request
[...]
podman[851405]: 2025-02-25 17:37:39.556 UTC [48] LOG:  database system is shut down
podman[851405]:  done
podman[851405]: server stopped
podman[851405]: PostgreSQL init process complete; ready for start up.
podman[851405]: 2025-02-25 17:37:39.677 UTC [1] LOG:  starting PostgreSQL 17.4 (Debian 17.4-1.pgdg120+2) on x86_64-pc-linux-gnu, compiled by gcc (Debian 12.2.0-14) 12.2.0, 64-bit
podman[851405]: 2025-02-25 17:37:39.677 UTC [1] LOG:  listening on IPv4 address "0.0.0.0", port 5432
podman[851405]: 2025-02-25 17:37:39.677 UTC [1] LOG:  listening on IPv6 address "::", port 5432
podman[851405]: 2025-02-25 17:37:39.684 UTC [1] LOG:  listening on Unix socket "/var/run/postgresql/.s.PGSQL.5432"
podman[851405]: 2025-02-25 17:37:39.693 UTC [62] LOG:  database system was shut down at 2025-02-25 17:37:39 UTC
podman[851405]: 2025-02-25 17:37:39.700 UTC [1] LOG:  database system is ready to accept connections

After applying this change, the log output is very similar except the log line from systemd is moved to the very end, after the database server has started listening on the IPv4 and IPv6 addresses.

Some container runtimes, such as podman, can expose a notify socket when
used in combination with systemd. The socket's path is available in the
`NOTIFY_SOCKET` envirionment variable.

Postgres has native support for this notification socket, and will write
a `READY=1` message once it is ready and accepting connections.
Unfortunately, the temporary server used by the `docker-entrypoint.sh`
to populate the database also sends a message on the socket, making it
appear as though the container is ready and serving connections when it
is not.

To avoid this issue we need to unset the NOTIFY_SOCKET envirionment
variable before starting the temporary server. The real server will
eventually find the socket and send the appropriate message.

The easiest way to test this is through the systemd-run command, which
creates a transient service:

```sh
systemd-run --user --service-type=notify \
  podman run -e POSTGRES_PASSWORD=password postgres
```

When looking at the logs for this command, prior to this change systemd
prints a "Started" message mid-way through the container startup (I've
omitted a lot of irrelevant lines for the sake of brevity):

```
systemd[2970]: Starting run-r916adefbc19c4465a1b0afc892980bd6.service - /usr/bin/podman run --log-driver none -e POSTGRES_PASSWORD=password postgres...
[...]
podman[851405]: waiting for server to start....2025-02-25 17:37:39.378 UTC [48] LOG:  starting PostgreSQL 17.4 (Debian 17.4-1.pgdg120+2) on x86_64-pc-linux-gnu, compiled by gcc (Debian 12.2.0-14) 12.2.0, 64-bit
podman[851405]: 2025-02-25 17:37:39.381 UTC [48] LOG:  listening on Unix socket "/var/run/postgresql/.s.PGSQL.5432"
podman[851405]: 2025-02-25 17:37:39.393 UTC [51] LOG:  database system was shut down at 2025-02-25 17:37:39 UTC
podman[851405]: 2025-02-25 17:37:39.400 UTC [48] LOG:  database system is ready to accept connections
systemd[2970]: Started run-r916adefbc19c4465a1b0afc892980bd6.service - /usr/bin/podman run --log-driver none -e POSTGRES_PASSWORD=password postgres.
podman[851405]:  done
podman[851405]: server started
podman[851405]: /usr/local/bin/docker-entrypoint.sh: ignoring /docker-entrypoint-initdb.d/*
podman[851405]: waiting for server to shut down...2025-02-25 17:37:39.520 UTC [48] LOG:  received fast shutdown request
[...]
podman[851405]: 2025-02-25 17:37:39.556 UTC [48] LOG:  database system is shut down
podman[851405]:  done
podman[851405]: server stopped
podman[851405]: PostgreSQL init process complete; ready for start up.
podman[851405]: 2025-02-25 17:37:39.677 UTC [1] LOG:  starting PostgreSQL 17.4 (Debian 17.4-1.pgdg120+2) on x86_64-pc-linux-gnu, compiled by gcc (Debian 12.2.0-14) 12.2.0, 64-bit
podman[851405]: 2025-02-25 17:37:39.677 UTC [1] LOG:  listening on IPv4 address "0.0.0.0", port 5432
podman[851405]: 2025-02-25 17:37:39.677 UTC [1] LOG:  listening on IPv6 address "::", port 5432
podman[851405]: 2025-02-25 17:37:39.684 UTC [1] LOG:  listening on Unix socket "/var/run/postgresql/.s.PGSQL.5432"
podman[851405]: 2025-02-25 17:37:39.693 UTC [62] LOG:  database system was shut down at 2025-02-25 17:37:39 UTC
podman[851405]: 2025-02-25 17:37:39.700 UTC [1] LOG:  database system is ready to accept connections
```

After applying this change, the log output is very similar except the
log line from systemd is moved to the very end, after the database
server has started listening on the IPv4 and IPv6 addresses.
Comment on lines 272 to 279
# Unset NOTIFY_SOCKET so the temporary server doesn't prematurely notify
# any process supervisor.
env \
--unset NOTIFY_SOCKET \
PGUSER="${PGUSER:-$POSTGRES_USER}" \
pg_ctl -D "$PGDATA" \
-o "$(printf '%q ' "$@")" \
-w start
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just setting it to the empty value should be sufficient, right?

Suggested change
# Unset NOTIFY_SOCKET so the temporary server doesn't prematurely notify
# any process supervisor.
env \
--unset NOTIFY_SOCKET \
PGUSER="${PGUSER:-$POSTGRES_USER}" \
pg_ctl -D "$PGDATA" \
-o "$(printf '%q ' "$@")" \
-w start
# unset NOTIFY_SOCKET so the temporary server doesn't prematurely notify any process supervisor
NOTIFY_SOCKET= \
PGUSER="${PGUSER:-$POSTGRES_USER}" \
pg_ctl -D "$PGDATA" \
-o "$(printf '%q ' "$@")" \
-w start

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes you're right. It is more concise this way. I've applied the suggested change and regenerated from the template.

@yosifkit yosifkit merged commit cc254e8 into docker-library:master Feb 27, 2025
32 checks passed
docker-library-bot added a commit to docker-library-bot/official-images that referenced this pull request Feb 27, 2025
Changes:

- docker-library/postgres@cc254e8: Unset NOTIFY_SOCKET when running the temporary server. (docker-library/postgres#1325)
@plietar plietar deleted the unset-notify branch February 27, 2025 01:56
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants