Skip to content

Commit 173632a

Browse files
authoredApr 10, 2020
[Feature] Add new API exporter (#19)
1 parent 612ffa0 commit 173632a

7 files changed

+164
-18
lines changed
 

‎.gitignore

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
11
.gobuild
22
bin
33
arangodb-exporter
4-
4+
vendor

‎Dockerfile.scratch

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
1-
FROM scratch
1+
ARG BASE_IMAGE=scratch
2+
FROM ${BASE_IMAGE}
23

34
ARG VERSION
45
LABEL name="arangodb-exporter" \

‎Dockerfile.ubi

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
ARG IMAGE=registry.access.redhat.com/ubi8/ubi-minimal:8.0
2+
FROM ${IMAGE}
3+
4+
RUN microdnf update && microdnf clean all

‎Makefile

+13-6
Original file line numberDiff line numberDiff line change
@@ -87,17 +87,24 @@ $(GOBUILDDIR):
8787
run-tests:
8888
go test $(REPOPATH)
8989

90+
docker-ubi-base: check-vars
91+
docker build --no-cache -t $(DOCKERIMAGE)-base-image-ubi -f Dockerfile.ubi .
92+
93+
docker-ubi: docker-ubi-base build
94+
for arch in amd64; do \
95+
docker build --build-arg "GOARCH=$$arch" --build-arg "VERSION=$(VERSION_MAJOR_MINOR_PATCH)" -t $(DOCKERIMAGE)-ubi --build-arg "BASE_IMAGE=$(DOCKERIMAGE)-base-image-ubi" -f Dockerfile.scratch . ; \
96+
docker push $(DOCKERIMAGE)-ubi ; \
97+
done
98+
99+
ifndef IGNORE_UBI
100+
docker: docker-ubi
101+
endif
102+
90103
docker: check-vars build
91104
for arch in $(ARCHS); do \
92105
docker build --build-arg "GOARCH=$$arch" --build-arg "VERSION=$(VERSION_MAJOR_MINOR_PATCH)" -t $(DOCKERIMAGE)-$$arch -f Dockerfile.scratch . ; \
93106
docker push $(DOCKERIMAGE)-$$arch ; \
94107
done
95-
for arch in amd64; do \
96-
sed -e 's|FROM scratch|FROM $(UBI)|' Dockerfile.scratch > Dockerfile.ubi ; \
97-
docker build --build-arg "GOARCH=$$arch" --build-arg "VERSION=$(VERSION_MAJOR_MINOR_PATCH)" -t $(DOCKERIMAGE)-ubi -f Dockerfile.ubi . ; \
98-
rm -f Dockerfile.ubi ; \
99-
docker push $(DOCKERIMAGE)-ubi ; \
100-
done
101108
DOCKER_CLI_EXPERIMENTAL=enabled docker manifest create --amend $(DOCKERIMAGE) $(foreach arch,$(ARCHS),$(DOCKERIMAGE)-$(arch)) $(DOCKERIMAGE)-ubi
102109
DOCKER_CLI_EXPERIMENTAL=enabled docker manifest push $(DOCKERIMAGE)
103110

‎README.md

+14
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,20 @@ This results in an ArangoDB Exporter exposing all statistics of
1818
the ArangoDB server (running at `http://<your-database-host>:8529`)
1919
at `http://<your-host-ip>:9101/metrics`.
2020

21+
## Exporter modes
22+
23+
### internal
24+
25+
Use internal metrics exporter mode for ArangoDB < 3.6.0
26+
27+
In this mode metrics are calculated on ArangoDB Exporter side
28+
29+
### passthru
30+
31+
Expose ArangoDB metrics for ArangoDB >= 3.6.0
32+
33+
In this mode metrics provided by ArangoDB `_admin/metrics` are exposed on Exporter port.
34+
2135
## Running in Docker
2236

2337
To run the ArangoDB Exporter in docker, use an image such as

‎main.go

+28-10
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,13 @@ import (
3737
"github.com/spf13/cobra"
3838
)
3939

40+
type ExporterMode string
41+
42+
const (
43+
ModeInternal ExporterMode = "internal"
44+
ModePassthru ExporterMode = "passthru"
45+
)
46+
4047
var (
4148
projectVersion = "dev"
4249
projectBuild = "dev"
@@ -50,6 +57,7 @@ var (
5057
serverOptions ServerConfig
5158
arangodbOptions struct {
5259
endpoint string
60+
mode string
5361
jwtSecret string
5462
jwtFile string
5563
timeout time.Duration
@@ -67,6 +75,8 @@ func init() {
6775
f.StringVar(&arangodbOptions.jwtFile, "arangodb.jwt-file", "", "File containing the JWT for authentication with ArangoDB server")
6876
f.DurationVar(&arangodbOptions.timeout, "arangodb.timeout", time.Second*15, "Timeout of statistics requests for ArangoDB")
6977

78+
f.StringVar(&arangodbOptions.mode, "mode", "internal", "Mode for ArangoDB exporter. Internal - use internal, old mode of metrics calculation (default). Passthru - expose ArangoD metrics directly, using proper authentication.")
79+
7080
f.MarkDeprecated("arangodb.jwtsecret", "please use --arangodb.jwt-file instead")
7181
}
7282

@@ -91,20 +101,28 @@ func cmdMainRun(cmd *cobra.Command, args []string) {
91101
log.Fatal(err)
92102
}
93103
}
94-
95-
exporter, err := NewExporter(arangodbOptions.endpoint, token, false, arangodbOptions.timeout)
96-
if err != nil {
97-
log.Fatal(err)
104+
mux := http.NewServeMux()
105+
switch ExporterMode(arangodbOptions.mode) {
106+
case ModePassthru:
107+
passthru, err := NewPassthru(arangodbOptions.endpoint, token, false, arangodbOptions.timeout)
108+
if err != nil {
109+
log.Fatal(err)
110+
}
111+
mux.Handle("/metrics", passthru)
112+
default:
113+
exporter, err := NewExporter(arangodbOptions.endpoint, token, false, arangodbOptions.timeout)
114+
if err != nil {
115+
log.Fatal(err)
116+
}
117+
prometheus.MustRegister(exporter)
118+
version.Version = projectVersion
119+
version.Revision = projectBuild
120+
prometheus.MustRegister(version.NewCollector("arangodb_exporter"))
121+
mux.Handle("/metrics", prometheus.Handler())
98122
}
99-
prometheus.MustRegister(exporter)
100-
version.Version = projectVersion
101-
version.Revision = projectBuild
102-
prometheus.MustRegister(version.NewCollector("arangodb_exporter"))
103123

104124
log.Infoln("Listening on", serverOptions.Address)
105125

106-
mux := http.NewServeMux()
107-
mux.Handle("/metrics", prometheus.Handler())
108126
mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
109127
w.Write([]byte(`<html>
110128
<head><title>ArangoDB Exporter</title></head>

‎passthru.go

+102
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
//
2+
// DISCLAIMER
3+
//
4+
// Copyright 2018 ArangoDB GmbH, Cologne, Germany
5+
//
6+
// Licensed under the Apache License, Version 2.0 (the "License");
7+
// you may not use this file except in compliance with the License.
8+
// You may obtain a copy of the License at
9+
//
10+
// http://www.apache.org/licenses/LICENSE-2.0
11+
//
12+
// Unless required by applicable law or agreed to in writing, software
13+
// distributed under the License is distributed on an "AS IS" BASIS,
14+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15+
// See the License for the specific language governing permissions and
16+
// limitations under the License.
17+
//
18+
// Copyright holder is ArangoDB GmbH, Cologne, Germany
19+
//
20+
// Author Adam Janikowski
21+
//
22+
23+
package main
24+
25+
import (
26+
"crypto/tls"
27+
"fmt"
28+
"io"
29+
"net/http"
30+
"time"
31+
)
32+
33+
var _ http.Handler = &passthru{}
34+
35+
func NewPassthru(arangodbEndpoint, jwt string, sslVerify bool, timeout time.Duration) (http.Handler, error) {
36+
transport := &http.Transport{}
37+
38+
req, err := http.NewRequest("GET", fmt.Sprintf("%s/_admin/metrics", arangodbEndpoint), nil)
39+
if err != nil {
40+
return nil, maskAny(err)
41+
}
42+
43+
if !sslVerify {
44+
transport.TLSClientConfig = &tls.Config{InsecureSkipVerify: true}
45+
}
46+
47+
if jwt != "" {
48+
hdr, err := CreateArangodJwtAuthorizationHeader(jwt)
49+
if err != nil {
50+
return nil, maskAny(err)
51+
}
52+
req.Header.Add("Authorization", hdr)
53+
}
54+
55+
client := &http.Client{
56+
Transport: transport,
57+
Timeout: timeout,
58+
}
59+
60+
return &passthru{
61+
client: client,
62+
request: req,
63+
}, nil
64+
}
65+
66+
type passthru struct {
67+
request *http.Request
68+
client *http.Client
69+
}
70+
71+
func (p passthru) get() (*http.Response, error) {
72+
return p.client.Do(p.request)
73+
}
74+
75+
func (p passthru) ServeHTTP(resp http.ResponseWriter, req *http.Request) {
76+
data, err := p.get()
77+
78+
if err != nil {
79+
// Ignore error
80+
resp.WriteHeader(http.StatusInternalServerError)
81+
resp.Write([]byte(err.Error()))
82+
return
83+
}
84+
85+
if data.Body == nil {
86+
// Ignore error
87+
resp.WriteHeader(http.StatusInternalServerError)
88+
resp.Write([]byte("Body is empty"))
89+
return
90+
}
91+
92+
defer data.Body.Close()
93+
94+
_, err = io.Copy(resp, data.Body)
95+
96+
if err != nil {
97+
// Ignore error
98+
resp.WriteHeader(http.StatusInternalServerError)
99+
resp.Write([]byte("Unable to write body"))
100+
return
101+
}
102+
}

0 commit comments

Comments
 (0)
Please sign in to comment.