在带有 Postgresql 数据库的 docker 容器中运行简单的 Kotlin Ktor 应用程序

Posted

技术标签:

【中文标题】在带有 Postgresql 数据库的 docker 容器中运行简单的 Kotlin Ktor 应用程序【英文标题】:Running simple Kotlin Ktor app in docker container with PostgresSql database 【发布时间】:2021-04-10 17:40:47 【问题描述】:

我想创建使用 PostgresSqlKotlin Ktor 的简单 Kotlin 应用程序,所有内容都应嵌入到 docker 容器中。

到目前为止,我设法分别运行成功地相互连接的 PostgresSqlPgAdmin,并为此创建了 docker-compose.yml 文件,这对我来说效果很好。当我想向其中添加我的 Kotlin 应用程序时,问题就开始了。

这是我的docker-compose.yml 文件


version: "3.9"
networks:
  m8network:
    ipam:
      config:
        - subnet: 172.20.0.0/24
services:
  postgres:
    image: postgres
    environment:
      - "POSTGRES_USER=SomeFancyUser"
      - "POSTGRES_PASSWORD=pwd"
      - "POSTGRES_DB=MSC8"
    ports:
      - "5432:5432"
    volumes:
      #           - postgres-data:/var/lib/postgresql/data
      - D:\docker\myApp\data:/var/lib/postgresql/data
    networks:
      m8network:
        ipv4_address: 172.20.0.6
  pgadmin:
    image: dpage/pgadmin4
    depends_on:
      - postgres
    environment:
      - "PGADMIN_DEFAULT_EMAIL=SomeFancyUser@domain.com"
      - "PGADMIN_DEFAULT_PASSWORD=pwd"
    #           - "PGADMIN_ENABLE_TLS=False"
    ports:
      - "5001:80"
    networks:
      m8network:
  app:
    build: .
    ports:
      - "5000:8080"
    links:
      - postgres
    depends_on:
      - postgres
    restart: on-failure
    networks:
      m8network:
#volumes:
#    postgres-data:
#        driver: local

这是我的应用程序源代码。


package com.something.m8

import com.squareup.sqldelight.db.SqlDriver
import com.squareup.sqldelight.sqlite.driver.asJdbcDriver
import com.zaxxer.hikari.HikariConfig
import com.zaxxer.hikari.HikariDataSource
import io.ktor.application.*
import io.ktor.html.*
import io.ktor.http.*
import io.ktor.response.*
import io.ktor.routing.*
import io.ktor.server.engine.*
import io.ktor.server.netty.*
import kotlinx.html.*
import java.io.PrintWriter
import java.util.*


fun HTML.index() 
    head 
        title("Hello from Ktor!")
    
    body 
        div 
            +"Hello from Ktor"
        
    


fun main() 
    println("starting app")
    val props = Properties()
    props.setProperty("dataSourceClassName", "org.postgresql.ds.PGSimpleDataSource")
    props.setProperty("dataSource.user", "SomeFancyUser")
    props.setProperty("dataSource.password", "pwd")
    props.setProperty("dataSource.databaseName", "M8")
    props.setProperty("dataSource.portNumber", "5432")
    props.setProperty("dataSource.serverName", "172.20.0.6")
    props["dataSource.logWriter"] = PrintWriter(System.out)
    println("a")
    val config = HikariConfig(props)
    println("b")

    val ds = HikariDataSource(config)
    println("c")
    val driver: SqlDriver = ds.asJdbcDriver()
    println("d")
    MSC8.Schema.create(driver)
    println("e")
    embeddedServer(Netty, port = 8080,
       // host = "127.0.0.1"
    ) 
        routing 
            get("/") 
                call.respondHtml(HttpStatusCode.OK, HTML::index)
            
            get("/m8/code") 
                val code = call.parameters["code"]
                println("code $code")
                call.respondRedirect("https://google.com")
            
        
    .start(wait = true)


还有应用程序的Dockerfile


#FROM openjdk:8
FROM gradle:6.7-jdk8

WORKDIR /var/www/html
RUN mkdir -p ./app/
WORKDIR /var/www/html/app
COPY build.gradle.kts .
COPY gradle.properties .
COPY settings.gradle.kts .
COPY Redirect/src ./Redirect/src
COPY Redirect/build.gradle.kts ./Redirect/build.gradle.kts
COPY gradlew .
COPY gradle ./gradle
EXPOSE 8080

USER root
WORKDIR /var/www/html
RUN pwd
RUN ls
RUN chown -R gradle ./app
USER gradle
WORKDIR /var/www/html/app
RUN ./gradlew run

使用这个设置我有两个问题

第一个问题:

当我运行docker-compose.exe up --build 时,我在val ds = HikariDataSource(config) 上收到异常HikariPool$PoolInitializationException: Failed to initialize pool: The connection attempt failed. 我为 postgres (172.20.0.6) 设置了静态 ip,当我在 PGAdmin 中使用这个 ip 时它可以工作,为什么我的应用无法连接到 postgres?

第二个问题:

我尝试测试应用程序是否正常启动,并且在大多数情况下一切正常。所以我注释了所有与数据库连接相关的源代码,因为当我运行docker-compose.exe up --build 时,我的应用程序只显示来自println("e") 行的字母e,此时一切似乎都被冻结了, postgres 和 PGAdming 没有启动,之后容器似乎没有响应并且应用程序在端口 5000 或 8080 上没有响应。有什么方法可以运行应用程序,这样它就不会阻止其他部分的执行?

【问题讨论】:

【参考方案1】:

第一个问题: 我开始使用主机名而不是 IP 地址,所以现在我使用 postgres 而不是 172.20.0.6。其余的都与第二个问题有关

第二个问题:

问题是我在容器的构建阶段启动了应用程序。

我用的是RUN ./gradlew run

RUN gradle build
ENTRYPOINT ["gradle","run"]

我还注意到我在使用 FROM gradle:6.7-jdk8 时不必使用 gradle 包装器

现在一切正常。

【讨论】:

以上是关于在带有 Postgresql 数据库的 docker 容器中运行简单的 Kotlin Ktor 应用程序的主要内容,如果未能解决你的问题,请参考以下文章

在 PostgreSQL 数据库中插入带有 jOOQ 的 SQL 枚举

PostgreSQL:查询在带有变量的查询中没有结果数据的目的地

如何在带有 PostgreSQL 数据库的 Web 应用程序中拥有完整的离线功能?

限制 PostgreSQL 最小时间间隔的结果行

带有 Postgresql 的 Grails:无法在适当的模式中创建表

带有 PostgreSQL 和 Pentaho 的 JDBC 驱动程序