【问题标题】:Docker Compose + Spring Boot + Postgres connectionDocker Compose + Spring Boot + Postgres 连接
【发布时间】:2023-05-09 15:43:02
【问题描述】:

我有一个与 Postgres 数据库配合使用的 Java Spring Boot 应用程序。我想对它们都使用 Docker。我最初只将 Postgres 放在 Docker 中,并且我有一个 docker-compose.yml 文件定义如下:

version: '2'
services:
    db:
        container_name: sample_db
        image: postgres:9.5
        volumes:
            - sample_db:/var/lib/postgresql/data
        environment:
            - POSTGRES_PASSWORD=sample
            - POSTGRES_USER=sample
            - POSTGRES_DB=sample
            - PGDATA=/var/lib/postgresql/data/pgdata
        ports:
            - 5432:5432

volumes:
    sample_db: {}

然后,当我发出命令sudo dockerdsudo docker-compose -f docker-compose.yml up 时,它正在启动数据库。例如,我可以使用pgAdmin 连接,使用localhost 作为服务器和端口5432。然后,在我的 Spring Boot 应用程序中,在 application.properties 文件中,我定义了以下属性。

spring.datasource.url=jdbc:postgresql://localhost:5432/sample
spring.datasource.username=sample
spring.datasource.password=sample
spring.jpa.generate-ddl=true

此时我可以通过 Spring Suite 在本地运行我的 Spring Boot 应用程序,并且一切正常。然后,我还想将我的 Spring Boot 应用程序添加为 Docker 映像。我首先在我的项目目录中创建了一个 Dockerfile,如下所示:

FROM java:8
EXPOSE 8080
ADD /target/manager.jar manager.jar
ENTRYPOINT ["java","-jar","manager.jar"]

然后,我进入mvn clean发布的项目目录,然后是mvn install。接下来,发出docker build -f Dockerfile -t manager .,后跟docker tag 9c6b1e3f1d5e myuser/manager:latest(id 正确)。最后,我将现有的docker-compose.yml 文件编辑为如下所示:

version: '2'
services:
    web:
      image: myuser/manager:latest
      ports: 
          - 8080:8080
      depends_on:
          - db
    db:
        container_name: sample_db
        image: postgres:9.5
        volumes:
            - sample_db:/var/lib/postgresql/data
        environment:
            - POSTGRES_PASSWORD=sample
            - POSTGRES_USER=sample
            - POSTGRES_DB=sample
            - PGDATA=/var/lib/postgresql/data/pgdata
        ports:
            - 5432:5432

volumes:
    sample_db: {}

但是,现在如果我发出 sudo docker-compose -f docker-compose.yml up 命令,数据库将再次正确启动,但我收到错误并退出 Web 应用程序部分的代码 1。问题是连接字符串。我相信我必须将其更改为其他内容,但我不知道应该是什么。我收到以下错误消息:

web_1  | 2017-06-27 22:11:54.418 ERROR 1 --- [           main] o.a.tomcat.jdbc.pool.ConnectionPool      : Unable to create initial connections of pool.
web_1  | 
web_1  | org.postgresql.util.PSQLException: Connection to localhost:5432 refused. Check that the hostname and port are correct and that the postmaster is accepting TCP/IP connections

有什么想法吗?

【问题讨论】:

标签: spring postgresql docker docker-compose dockerfile


【解决方案1】:

每个容器都有自己的网络接口和自己的本地主机。所以改变 Java 指向 Postgres 的方式:

spring.datasource.url=jdbc:postgresql://localhost:5432/sample

收件人:

spring.datasource.url=jdbc:postgresql://db:5432/sample

db 将解析到正确的 Postgres IP。


奖金。使用 docker-compose,您无需手动构建图像。所以改变:

web:
  image: myuser/manager:latest

收件人:

web:
  build: .

【讨论】:

  • 我不明白的一件事是,我在我的应用程序中使用 JPA,并且为了构建它需要连接到我的数据库的 spring-boot 应用程序,如果我的 postgre 我无法做到这一点码头工人没有运行,对吗?因此,基本上使用配置为连接到 dockerized postgre db 的类似应用程序属性,我无法首先构建我的应用程序...
【解决方案2】:

我遇到了同样的问题,我浪费了一些时间来理解和解决这个问题:

org.postgresql.util.PSQLException: Connection to localhost:5432 refused. Check that the hostname and port are correct and that the postmaster is accepting TCP/IP connections.

我把所有的属性都展示出来让大家看懂。
application.properties:

spring.datasource.url=jdbc:postgresql://localhost:5432/testdb
spring.datasource.driver-class-name=org.postgresql.Driver
spring.datasource.username=postgres
spring.datasource.password=postgres
spring.jpa.properties.hibernate.dialect=org.hibernate.dialect.PostgreSQL82Dialect
spring.jpa.hibernate.ddl-auto=update

docker-compose.yml:

  version: "3"
  services:
    springapp:
      build: .
      container_name: springapp
      environment:
        SPRING_DATASOURCE_URL: jdbc:postgresql://db:5432/testdb
      ports:
        - 8000:8080
      restart: always
      depends_on:
        - db
    db:
      image: postgres
      container_name: db
      environment:
        - POSTGRES_USER=postgres
        - POSTGRES_PASSWORD=postgres
        - POSTGRES_DB=testdb
        - PGDATA=/var/lib/postgresql/data/pgdata
      ports:
        - 5000:5432
      volumes:
        - pgdata:/var/lib/postgresql/data
      restart: always
  volumes:
    pgdata:

为了使用本地数据库启动 spring 应用程序,我们使用 url localhost。
要使用数据库连接到容器,我们需要将您的数据库服务上的“localhost”更改为“localhost”,在我的情况下为“db”。

解决方案:在docker-compose.yml 中添加SPRING_DATASOURCE_URL 环境,重写spring.datasource.url 连接值:

  environment:
    SPRING_DATASOURCE_URL: jdbc:postgresql://db:5432/testdb

我希望这可以帮助某人节省时间。

【讨论】:

  • 你能分享你的dockerfile吗
  • 我在 app.properties 中没有t know that env variables override whats。谢谢亚历山大!
  • 谢谢,我花了几个小时才找到解决方案。
【解决方案3】:

你可以用这个。

version: "2"
services:
    sample_db-postgresql:
        image: postgres:9.5
        ports:
            - 5432:5432
        environment:
            - POSTGRES_PASSWORD=sample
            - POSTGRES_USER=sample
            - POSTGRES_DB=sample
         volumes:
            - sample_db:/var/lib/postgresql/data

volumes:
    sample_db:

【讨论】: