flyway 无法连接到 docker-entrypoint-initdb.d 脚本中的 postgres 容器

Posted

技术标签:

【中文标题】flyway 无法连接到 docker-entrypoint-initdb.d 脚本中的 postgres 容器【英文标题】:flyway unable to connect to postgres container within docker-entrypoint-initdb.d script 【发布时间】:2019-01-12 22:53:48 【问题描述】:

我正在尝试扩展 postgres Docker 图像以潜在地(通过环境变量标志)在 DB init 上执行 flyway DB 迁移。我的 Dockerfile 在这里:

FROM postgres:9.6

# Install curl and java (for Flyway)
RUN set -x \
    && apt-get update \
    && apt-get install -y --no-install-recommends ca-certificates curl openjdk-8-jre 

# Install Flyway
ENV FLYWAY_VERSION 4.2.0
ENV FLYWAY_INSTALL_DIR /usr/src/flyway
ENV FLYWAY_CONF $FLYWAY_INSTALL_DIR/flyway-$FLYWAY_VERSION/conf/flyway.conf
ENV FLYWAY_EXE $FLYWAY_INSTALL_DIR/flyway-$FLYWAY_VERSION/flyway
RUN mkdir -p $FLYWAY_INSTALL_DIR && \
    curl -L https://repo1.maven.org/maven2/org/flywaydb/flyway-commandline/$FLYWAY_VERSION/flyway-commandline-$FLYWAY_VERSION.tar.gz | \
    tar -xzC $FLYWAY_INSTALL_DIR && \
    chmod +x $FLYWAY_EXE

# Copy migration scripts
ENV MIGRATIONS_LOCATION /flyway/migrations
COPY migrations $MIGRATIONS_LOCATION

COPY init_db.sh /docker-entrypoint-initdb.d/init_db.sh

使用我的init_db.sh 启动脚本:

#!/bin/bash
set -e

RUN_MIGRATIONS="$RUN_MIGRATIONS:-false"
DB_URL="jdbc:postgresql://localhost:5432/$DB_NAME"

psql -v ON_ERROR_STOP=1 --username "$POSTGRES_USER" <<-EOSQL
    CREATE DATABASE $DB_NAME;
EOSQL

if [ "$RUN_MIGRATIONS" == "true" ]; then
        echo "running migrations ..."
        $FLYWAY_EXE -user=$POSTGRES_USER -password=$POSTGRES_PASSWORD -url=$DB_URL -locations="filesystem:$MIGRATIONS_LOCATION" migrate
fi

但是,当使用RUN_MIGRATIONS=true 运行容器时,flyway 无法连接到 postgres:

docker build . -t postgres-flyway && docker run -e DB_NAME=db -e RUN_MIGRATIONS=true -e POSTGRES_USER=postgres -e POSTGRES_PASSWORD=postgres postgres-flyway
The files belonging to this database system will be owned by user "postgres".
This user must also own the server process.

The database cluster will be initialized with locale "en_US.utf8".
The default database encoding has accordingly been set to "UTF8".
The default text search configuration will be set to "english".

Data page checksums are disabled.

fixing permissions on existing directory /var/lib/postgresql/data ... ok
creating subdirectories ... ok
selecting default max_connections ... 100
selecting default shared_buffers ... 128MB
selecting dynamic shared memory implementation ... posix
creating configuration files ... ok
running bootstrap script ... ok
performing post-bootstrap initialization ... ok
syncing data to disk ... ok

WARNING: enabling "trust" authentication for local connections
You can change this by editing pg_hba.conf or using the option -A, or
--auth-local and --auth-host, the next time you run initdb.

Success. You can now start the database server using:

    pg_ctl -D /var/lib/postgresql/data -l logfile start

waiting for server to start....LOG:  database system was shut down at 2018-08-06 02:19:32 UTC
LOG:  MultiXact member wraparound protections are now enabled
LOG:  autovacuum launcher started
LOG:  database system is ready to accept connections
 done
server started
ALTER ROLE


/usr/local/bin/docker-entrypoint.sh: sourcing /docker-entrypoint-initdb.d/init_db.sh
CREATE DATABASE
running migrations ...
Flyway 4.2.0 by Boxfuse

ERROR:
Unable to obtain Jdbc connection from DataSource (jdbc:postgresql://localhost:5432/db) for user 'postgres': Connection to localhost:5432 refused. Check that the hostname and port are correct and that the postmaster is accepting TCP/IP connections.
-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
SQL State  : 08001
Error Code : 0
Message    : Connection to localhost:5432 refused. Check that the hostname and port are correct and that the postmaster is accepting TCP/IP connections.

postgres 映像在端口 5432 上运行 postgres(像往常一样),所以我不知道为什么 flyway 无法通过 localhost:5432 连接到 postgres。

我还注意到,在这种情况下,pg_isready 声明 postgres 正在接受连接,但是当将主机名指定为 localhost127.0.0.1 时,它也无法访问 postgres。也就是说,通过在我的init_db.sh 脚本中插入一些pg_isready 命令:

...
pg_isready
pg_isready -p 5432
pg_isready -h localhost -p 5432
...

我在 postgres init 上看到以下日志输出:

...
/var/run/postgresql:5432 - accepting connections
/var/run/postgresql:5432 - accepting connections
localhost:5432 - no response
...

我怀疑我已经达到了 postgres 初始化上下文的限制,但我想了解为什么 postgres 在初始化时无法通过 localhost/127.0.0.1:5432 访问。

【问题讨论】:

【参考方案1】:

我在挖掘图像entry point 脚本时发现了问题。最近对图像的更改将 postgres 限制为仅在内部初始化期间侦听 unix 域套接字上的连接:https://github.com/docker-library/postgres/pull/440

【讨论】:

【参考方案2】:

在基于 postgres:10.5 映像为我的数据库创建 docker 映像时,我在运行 flyway 时遇到了同样的问题。在运行 flyway 之前,我将以下内容添加到我的 entrypoint.sh 中,以确认我看到的问题是由@Nick Maraston 在他的回答中发布的 docker-entrypoint.sh 更改引起的:

echo "$(date) - waiting for database to start"
while ! pg_isready -h localhost -p 5432 -d $POSTGRES_DB
do
    echo "$(date) - waiting for database to start"
    sleep 10
done

结果是上面的代码永远循环。然后我将其替换为以下代码,以重新启动数据库以侦听 localhost 上的 TCP/IP 连接:

pg_ctl -D "$PGDATA" -m fast -w stop
pg_ctl -D "$PGDATA" \
            -o "-c listen_addresses='localhost'" \
            -w start

与其像这样重新启动数据库,更简洁的解决方案是使用 JDBC -socketFactory 选项解释 here。

【讨论】:

【参考方案3】:

postgres docker 确实在监听 unix 套接字,例如。 /var/run/postgresql/.s.PGSQL.5432。但不必强制服务器切换其监听地址。 Postgres 数据库 URI 允许连接字符串指向一个 unix 套接字。

参考:Connect to a database over a unix socket using SQLAlchemy

提供的例子:

export DATABASE_URL=postgres://user:password@/dbname?host=/path/to/unix/socket

我能够省略主机,并决定在我的 /docker-entrypoint-initdb.d/*.sh 脚本中使用此环境变量。请注意,@ 符号后面没有字符串,这里也没有主机查询字符串。您可能需要根据您的应用程序显式定义主机。

解决方案:

export DATABASE_URL="postgres://$POSTGRES_USER:$POSTGRES_PASSWORD@/$POSTGRES_DB"

【讨论】:

以上是关于flyway 无法连接到 docker-entrypoint-initdb.d 脚本中的 postgres 容器的主要内容,如果未能解决你的问题,请参考以下文章

flyway 无法连接到 docker-entrypoint-initdb.d 脚本中的 postgres 容器

Flyway 与 Git 集成并连接到 Snowflake

Flyway 无法使用 Snowflake JDBC

如何将运行在 Docker 容器中的 Flyway 连接到运行在 Docker 容器中的数据库?

docker mysql initdb 脚本无法连接到数据库

Spring Boot FlywayException:无法连接到数据库。配置url、用户和密码