Docker Compose + Spring Boot + Postgres 连接

30

我有一个使用Java Spring Boot开发的应用程序,它与Postgres数据库配合工作。我想要将它们都放入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 cleanmvn 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 命令,数据库会再次正确启动,但是网页应用程序部分会出现错误和退出代码 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 Boot 2、Docker Compose以及PostgreSQL的演示项目,链接为https://muzir.github.io/spring/docker/docker-compose/postgres/2019/03/24/Spring-Boot-Docker.html。 - erhun
4个回答

42
每个容器都有自己的网络接口和本地主机。因此更改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: .

1
我不理解的一件事是,我在我的应用程序中使用JPA,并且为了构建Spring-boot应用程序,它需要连接到我的数据库,但如果我的PostgreSQL Docker未运行,则无法连接,对吗?因此,基本上,如果配置了类似于连接到Docker化的PostgreSQL数据库的应用程序属性,那么我就无法首先构建我的应用程序... - solujic

12

我曾经遇到过同样的问题,花费了一些时间才理解和解决了这个问题:

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

希望这能帮助大家节省时间。


1
你能分享一下你的Dockerfile吗? - aswzen
我不知道环境变量会覆盖app.properties文件中的内容。感谢Alexander! - Dimo Georgiev
谢谢,我花了几个小时才找到解决方案。 - xmen-5

0

您可以使用环境变量在docker-compose中更改数据库地址。 Dockerfile:

FROM java:8
EXPOSE 8080
ENV POSTGRES localhost
ADD /target/manager.jar manager.jar
ENTRYPOINT exec java $JAVA_OPTS -jar manager.jar --spring.datasource.url=jdbc:postgresql://$POSTGRES:5432/sample

docker-compose:

  container_name: springapp
  environment:
   - POSTGRES=db`

0
你可以使用这个。
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:

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接