docker - 如何将 Docker 容器中运行的 Grafana 连接到主机上运行的 Prometheus 数据源(在 Docker for Mac 上)?

标签 docker go prometheus grafana docker-networking

我在 localhost:57475/metrics 启动了 Prometheus 服务器:

enter image description here

使用以下 Go 代码:

package main

import (
    "net/http"

    "contrib.go.opencensus.io/exporter/prometheus"
    log "github.com/sirupsen/logrus"
    "go.opencensus.io/plugin/ochttp"
    "go.opencensus.io/stats/view"
)

func main() {
    stop := make(chan struct{})

    server := &http.Server{Addr: ":8080"}

    statsMux := http.NewServeMux()
    statsServer := &http.Server{Handler: statsMux, Addr: ":57475"}

    if err := view.Register(ochttp.DefaultServerViews...); err != nil {
        log.WithError(err).Fatal("register HTTP metrics view")
    }

    exporter, err := prometheus.NewExporter(prometheus.Options{
        Namespace: "default",
    })
    if err != nil {
        log.WithError(err).Fatal("create Prometheus exporter")
    }

    view.RegisterExporter(exporter)

    statsMux.Handle("/metrics", exporter)

    originalHandler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        w.Write([]byte("Hello, World!"))
    })
    och := &ochttp.Handler{
        Handler: originalHandler,
    }

    server.Handler = och

    go func() {
        log.Info("Starting stats server...")
        if err := statsServer.ListenAndServe(); err != nil {
            log.WithError(err).Fatal("listen and serve stats")
        }
    }()

    go func() {
        log.Info("Starting server...")
        if err := server.ListenAndServe(); err != nil {
            log.WithError(err).Fatal("listen and serve service endpoints")
        }
    }()

    <-stop
}

我还使用以下命令(在 https://grafana.com/docs/installation/docker/ 之后)在容器中启动了 Grafana:
> docker run --detach --publish 3000:3000  --env "GF_SECURITY_ADMIN_PASSWORD=secret" grafana/grafana
2a17ad2bcc05e2955190129b981b3329cd2c5e89098e28b443337ac79ed607f2

在 Grafana 中,我想连接到在我的 Mac 上运行的 Prometheus 导出器 localhost .关注 https://docs.docker.com/docker-for-mac/networking/ ,我尝试使用特殊的 DNS 名称 host.docker.internal ,因此将 URL 指定为 http://host.docker.internal:57475/metrics .但是,这会导致 HTTP Error Not Found当我单击“保存并测试”时:

enter image description here

知道为什么这不起作用吗?

更新

而不是使用特殊的 DNS 名称 host.docker.internal ,我尝试将服务器和 Grafana 组合成一个 Docker Compose 多容器应用程序,并使用 Docker 桥连接它们,但这也不起作用。使用以下目录结构:
.
├── docker-compose.yml
└── server
    ├── Dockerfile
    ├── go.mod
    ├── go.sum
    └── main.go

以及以下 docker-compose.yml :
version: '3'
services:
  server:
    build: ./server
    ports:
      - "8080:8080"
      - "57475:57475"
  grafana:
    image: grafana/grafana
    ports:
      - "3000:3000"

以及以下 Dockerfile :
FROM golang:1.12

WORKDIR /go/src/app
COPY . .

RUN go get -d -v ./...
RUN go install -v ./...

CMD ["app"]

如果我连接到 Grafana 容器并 curl server:8080server:57475/metrics ,我得到一个回应:
useprometheus> docker exec -it --user root useprometheus_grafana_1 /bin/ash
/usr/share/grafana # apk add curl
fetch http://dl-cdn.alpinelinux.org/alpine/v3.10/main/x86_64/APKINDEX.tar.gz
fetch http://dl-cdn.alpinelinux.org/alpine/v3.10/community/x86_64/APKINDEX.tar.gz
(1/3) Installing nghttp2-libs (1.39.2-r0)
(2/3) Installing libcurl (7.66.0-r0)
(3/3) Installing curl (7.66.0-r0)
Executing busybox-1.30.1-r2.trigger
OK: 17 MiB in 24 packages
/usr/share/grafana # curl server:8080
Hello, World!/usr/share/grafana # curl server:57475/metrics
# HELP default_opencensus_io_http_server_latency Latency distribution of HTTP requests
# TYPE default_opencensus_io_http_server_latency histogram
default_opencensus_io_http_server_latency_bucket{le="1"} 1
default_opencensus_io_http_server_latency_bucket{le="2"} 1
default_opencensus_io_http_server_latency_bucket{le="3"} 1
default_opencensus_io_http_server_latency_bucket{le="4"} 1

但是,如果我尝试在 Grafana 管理员中添加相同的 URL,则会收到相同的错误:

enter image description here

换句话说,虽然我能够curl这个端点来自 Grafana 容器,我不能将它用作 Grafana 仪表板中的数据源 URL,让我认为这是 Grafana 特定的。知道为什么这不起作用吗?

最佳答案

原来我不太明白普罗米修斯是如何工作的。示例中的 Go 应用程序创建了一个 Prometheus 导出器,它仍然需要由 Prometheus 服务器抓取,而 Prometheus 服务器又是 Grafana 的数据源。

我把目录结构修改成这样:

.
├── docker-compose.yml
├── prometheus
│   ├── Dockerfile
│   └── prometheus.yml
└── server
    ├── Dockerfile
    ├── go.mod
    ├── go.sum
    └── main.go

哪里docker-compose.yml
version: '3'
services:
  server:
    build: ./server
    ports:
      - "8080:8080"
      - "57475:57475"
  grafana:
    image: grafana/grafana
    ports:
      - "3000:3000"
  prometheus:
    build: ./prometheus
    ports:
      - "9090:9090"

prometheus/Dockerfile
FROM prom/prometheus

ADD prometheus.yml /etc/prometheus

prometheus/prometheus.yml
global:
  scrape_interval:     15s

scrape_configs:
  - job_name: 'server'

    static_configs:
      - targets: ['server:57475']

现在我可以添加 prometheus:9090作为数据源 URL:

enter image description here

关于docker - 如何将 Docker 容器中运行的 Grafana 连接到主机上运行的 Prometheus 数据源(在 Docker for Mac 上)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58699648/

相关文章:

json - 使用jsonnet覆盖嵌套列表元素

prometheus - 如何在普罗米修斯中删除除少数列入白名单的指标之外的所有指标?

go - 在 golang 中收集 Kubernetes 指标

Docker swarm 无法访问工作节点

docker - CodeBuild 环境需要我自己的 docker 镜像吗?

go - 为什么从数据库中删除一些数据后它没有插入到同一个表中

Heroku go-getting-started 不在本地运行

Docker卷: specifying permissions using mount options

c# - centos 8 中 docker build net core 5.0 报错 NET SDK is not installed

sqlite - gosqlite(golang)中出现奇怪的 sqlite 错误