如何在ENTRYPOINT等待postgres启动?

时间:2017-02-13 14:12:08

标签: bash postgresql docker

在继续执行postgres之前,等待ENTRYPOINTnosetests内完全启动的最佳方法是什么?

现在,我已经将我的机器上的启动时间定为大约50秒。所以我只是睡了60秒。这感觉不太好,因为在另一台机器上运行时可能无法正常工作。

ENTRYPOINT \
           runuser -l postgres -c '/usr/lib/postgresql/9.3/bin/postgres -D /var/lib/postgresql/9.3/main -c config_file=/etc/postgresql/9.3/main/postgresql.conf & ' && \
           sleep 60 && \
           nosetests --verbose --cover-erase --with-coverage --cover-package=stalker

这是启动postgres

的输出
2017-02-13 13:46:49.541 UTC [9] LOG:  database system was interrupted; last known up at 2017-02-13 12:53:23 UTC
2017-02-13 13:47:37.951 UTC [9] LOG:  database system was not properly shut down; automatic recovery in progress
2017-02-13 13:47:37.994 UTC [9] LOG:  redo starts at 0/1783EA0
2017-02-13 13:47:37.995 UTC [9] LOG:  record with zero length at 0/17841E8
2017-02-13 13:47:37.995 UTC [9] LOG:  redo done at 0/17841B8
2017-02-13 13:47:37.995 UTC [9] LOG:  last completed transaction was at log time 2017-02-13 12:53:23.731984+00
2017-02-13 13:47:38.384 UTC [9] LOG:  MultiXact member wraparound protections are now enabled
2017-02-13 13:47:38.387 UTC [7] LOG:  database system is ready to accept connections
2017-02-13 13:47:38.387 UTC [13] LOG:  autovacuum launcher started

请注意,这违反了在ENTRYPOINT中运行多个命令的惯例。在这种情况下,我有充分的理由这样做。

3 个答案:

答案 0 :(得分:11)

感谢@zeppelin建议pg_isready。我最终使用了这个:

#!/bin/bash
# wait-for-postgres.sh

set -e

cmd="$@"
timer="5"

until runuser -l postgres -c 'pg_isready' 2>/dev/null; do
  >&2 echo "Postgres is unavailable - sleeping for $timer seconds"
  sleep $timer
done

>&2 echo "Postgres is up - executing command"
exec $cmd

我在ENTRYPOINT

中使用此功能
ENTRYPOINT \

            # Start PostgreSQL
            runuser -l postgres -c '/usr/lib/postgresql/9.3/bin/postgres -D /var/lib/postgresql/9.3/main -c config_file=/etc/postgresql/9.3/main/postgresql.conf & ' && \

            # Exectute tests when db is up
            ./wait-for-postgres.sh nosetests --verbose --cover-erase --with-coverage --cover-package=stalker

答案 1 :(得分:4)

我通常会使用一个小的“tcp-port-wait”脚本,如下所示:

#!/bin/bash
set -e 

if [ -z "$1" -o -z "$2" ]
then
    echo "tcp-port-wait - block until specified TCP port becomes available"
    echo "Usage: ntcp-port-wait HOST PORT"
    exit 1
fi
echo Waiting for port $1:$2 to become available...
while ! nc -z $1 $2 2>/dev/null
do
    let elapsed=elapsed+1
    if [ "$elapsed" -gt 90 ] 
    then
        echo "TIMED OUT !"
        exit 1
    fi  
    sleep 1;
done

echo "READY !"

等到某项服务启动并准备就绪。

对于Postgresql,它将侦听的默认端口是 5432 ,因此命令为:

tcp-port-wait localhost 5432

这将阻塞,直到Postgresql服务准备好提供连接,在容器内的loopback接口上的5432上(当在ENTRYPOINT脚本的上下文中运行时)。

您当然必须将此脚本复制到容器中, 通过在 Dockerfile 中添加类似的行:

COPY tcp-port-wait /usr/local/bin/

然后才能使用它。

并安装 netcat 实用程序。

您也可以将此用于其他类型的服务,例如Tomcat或Mysql。

而且,如果您需要,您也可以在“容器外”等待,如下所示:

docker exec my_container tcp-port-wait localhost 5432

注意,肯定有其他方法可以做到这一点,例如:使用一些编排工具,例如 docker-compose healthcheck directive,或者容器内部的一些流程管理器。

答案 2 :(得分:-1)

最好不要在一个容器中运行多个东西。最好的方法是让你postgres在一个容器中运行,并为你的nosetests使用另一个容器。

您可以创建一个docker-compose文件,其中nosetest"取决于"在数据库上。