首頁 > 後端開發 > Golang > Golang testcontainers,無法讓網路運作

Golang testcontainers,無法讓網路運作

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
發布: 2024-02-14 13:30:09
轉載
1118 人瀏覽過

Golang testcontainers,无法使网络工作

php小編草莓在使用Golang testcontainers時遇到了一個問題,即無法使網路正常工作。 Golang testcontainers是一個用於在測試中執行容器的工具,它可以幫助開發人員在測試環境中快速啟動和銷毀容器。然而,在實際使用中,php小編草莓發現無法在容器中正常使用網路功能,這給測試帶來了一定的困擾。接下來,我們將一起探討這個問題的解決方案。

問題內容

我正在嘗試在我的微服務中創建一些測試,我想創建一個網絡,將我的資料庫測試容器(postgres)和我的微服務測試容器附加到該網路。無論我嘗試什麼,我都無法讓我的微服務連接到資料庫。我的微服務是使用 Fiber 和 Gorm 的 golang。我嘗試連接到 db.go 設定檔中的資料庫,如下所示:

func SetupDB(port string, host string) *gorm.DB {

    dsn := "host=" + host + " user=postgres password=password dbname=prescription port=" + port + " sslmode=disable"

    db, err := gorm.Open(postgres.Open(dsn), &gorm.Config{})

    if err != nil {
        panic("error connecting to database")
    }

    db.AutoMigrate(&model.Prescription{})

    return db
}
登入後複製

這就是我的測試容器的樣子:

prescriptionDBContainer, err := testcontainers.GenericContainer(context.Background(), testcontainers.GenericContainerRequest{
        ContainerRequest: testcontainers.ContainerRequest{
            Image:        "postgres",
            ExposedPorts: []string{postgresPort.Port()},
            Env: map[string]string{
                "POSTGRES_USER":     "postgres",
                "POSTGRES_PASSWORD": "password",
                "POSTGRES_DB":       "prescription",
            },
            Networks: []string{network.Name},
            NetworkAliases: map[string][]string{
                network.Name: {"db-network"},
            },
            WaitingFor: wait.ForAll(
                wait.ForLog("database system is ready to accept connections"),
                wait.ForListeningPort(postgresPort),
            ),
        },
        Started: true,
    })
登入後複製
prescriptionContainer, err := testcontainers.GenericContainer(context.Background(), testcontainers.GenericContainerRequest{
        ContainerRequest: testcontainers.ContainerRequest{
            FromDockerfile: testcontainers.FromDockerfile{Context: "../../../../prescription"},
            Networks:       []string{network.Name},
            NetworkAliases: map[string][]string{
                network.Name: {"db-network"},
            },
            Env: map[string]string{
                "POSTGRES_USER":     "postgres",
                "POSTGRES_PASSWORD": "password",
                "POSTGRES_DB":       "prescription",
                "HOST":              prescriptionDBHost,
                "DB_PORT":           prescriptionDBPort.Port(),
            },
            ExposedPorts: []string{pMicroPort.Port()},
            WaitingFor:   wait.ForListeningPort("8080"),
        },
        Started: true,
    })
登入後複製

也許是因為我根本不了解docker中網路連接過程中發生的事情,但我真的迷失了,當我為HOST和DB_PORT設定環境時(我已經嘗試了陽光下的每一種組合),它拒絕連接資料庫的微服務

在我嘗試過的微服務的測試容器中:

"HOST":              prescriptionDBHost,
"DB_PORT":           prescriptionDBPort.Port(),
登入後複製

prescriptionDBHost 的擷取方式為:

prescriptionDBHost, err := prescriptionDBContainer.Name(context.Background())
登入後複製

導致錯誤訊息:

failed to initialize database, got error failed to connect to `host=/stoic_heyrovsky user=postgres database=prescription`: dial error (dial unix /stoic_heyrovsky/.s.PGSQL.53802: connect: no such file or directory)
panic: error connecting to database
登入後複製

然後我嘗試從主機名稱中刪除“/”,例如:

"HOST":              strings.Trim(prescriptionDBHost,"/"),
"DB_PORT":           prescriptionDBPort.Port(),
登入後複製

我也嘗試過:

"HOST":              "localhost",
"DB_PORT":           prescriptionDBPort.Port(),
登入後複製
"HOST":              "127.0.0.1",
"DB_PORT":           prescriptionDBPort.Port(),
登入後複製
prescriptionDBHost, err := prescriptionDBContainer.ContainerIP(context.Background())

"HOST":              prescriptionDBHost,
"DB_PORT":           prescriptionDBPort.Port(),
登入後複製

這裡的最後 4 個範例都會導致某種撥號 TCP 錯誤,例如:

failed to initialize database, got error failed to connect to `host=localhost user=postgres database=prescription`: dial error (dial tcp [::1]:53921: connect: cannot assign requested address)
登入後複製

我還在testcontainer 創建資料庫容器後進行了調試和停止,然後轉到我的微服務並使用DB_HOST=localhost 和port= 硬編碼連接到該容器,並且成功了,所以我真的迷失了正在發生的事情錯誤的。我唯一能想到的是,微服務容器在嘗試連接資料庫之前沒有連接到網路?我做了一個 docker 網路檢查,我可以看到資料庫容器已附加,但微服務從未附加(但也許這只是因為其他原因?)。

解決方法

你可以這樣做:

prescriptionDBContainer, err := testcontainers.GenericContainer(context.Background(), testcontainers.GenericContainerRequest{
    ContainerRequest: testcontainers.ContainerRequest{
        Image:        "postgres",
        ExposedPorts: []string{"5432/tcp"},
        Env: map[string]string{
            "POSTGRES_USER":     "postgres",
            "POSTGRES_PASSWORD": "password",
            "POSTGRES_DB":       "prescription",
        },
        Networks:       []string{networkName},
        NetworkAliases: map[string][]string{networkName: []string{"postgres"}},
        WaitingFor: wait.ForAll(
            wait.ForLog("database system is ready to accept connections"),
            wait.ForListeningPort("5432/tcp"),
        ),
    },
    Started: true,
})
if err != nil {
    t.Fatal(err)
}

prescriptionContainer, err := testcontainers.GenericContainer(context.Background(), testcontainers.GenericContainerRequest{
    ContainerRequest: testcontainers.ContainerRequest{
        FromDockerfile: testcontainers.FromDockerfile{Context: "./testapp"},
        ExposedPorts:   []string{"8080/tcp"},
        Networks:       []string{networkName},
        NetworkAliases: map[string][]string{networkName: []string{"blah"}},
        Env: map[string]string{
            "DATABASE_URL": "postgres://postgres:password@postgres:5432/prescription",
        },
        WaitingFor: wait.ForListeningPort("8080/tcp"),
    },
    Started: true,
})
登入後複製

注意NetworkAliases的配置方式;在您的程式碼中,您將兩者都設定為 db-network 但是,我想,這是由於誤解。該設定配置了一個可以引用容器的別名(在本例中,我使用postgres 作為postgres 容器;這表示當連接HOST 時,根據上面範例中使用的URL ,postgres 將是postgres)。

作為替代方案,您可以使用port, err :=處方DBContainer.MappedPort(context.Background(), "5432/tcp") 取得主機上公開的端口,然後連接到連接埠上的host.docker.internal port.Port().當被測試的應用程式在主機上而不是在容器中運行時,經常使用此方法(但在這種情況下,您將連接到localhost 並使用從MappedPort() 傳回的報告)。

以上是Golang testcontainers,無法讓網路運作的詳細內容。更多資訊請關注PHP中文網其他相關文章!

來源:stackoverflow.com
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板