mirror of
https://github.com/prometheus-community/postgres_exporter
synced 2025-04-11 03:31:26 +00:00
222 lines
4.8 KiB
Go
222 lines
4.8 KiB
Go
// Copyright 2021 The Prometheus Authors
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"math"
|
|
"net/url"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/go-kit/log/level"
|
|
"github.com/lib/pq"
|
|
)
|
|
|
|
func contains(a []string, x string) bool {
|
|
for _, n := range a {
|
|
if x == n {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// convert a string to the corresponding ColumnUsage
|
|
func stringToColumnUsage(s string) (ColumnUsage, error) {
|
|
var u ColumnUsage
|
|
var err error
|
|
switch s {
|
|
case "DISCARD":
|
|
u = DISCARD
|
|
|
|
case "LABEL":
|
|
u = LABEL
|
|
|
|
case "COUNTER":
|
|
u = COUNTER
|
|
|
|
case "GAUGE":
|
|
u = GAUGE
|
|
|
|
case "HISTOGRAM":
|
|
u = HISTOGRAM
|
|
|
|
case "MAPPEDMETRIC":
|
|
u = MAPPEDMETRIC
|
|
|
|
case "DURATION":
|
|
u = DURATION
|
|
|
|
default:
|
|
err = fmt.Errorf("wrong ColumnUsage given : %s", s)
|
|
}
|
|
|
|
return u, err
|
|
}
|
|
|
|
// Convert database.sql types to float64s for Prometheus consumption. Null types are mapped to NaN. string and []byte
|
|
// types are mapped as NaN and !ok
|
|
func dbToFloat64(t interface{}) (float64, bool) {
|
|
switch v := t.(type) {
|
|
case int64:
|
|
return float64(v), true
|
|
case float64:
|
|
return v, true
|
|
case time.Time:
|
|
return float64(v.Unix()), true
|
|
case []byte:
|
|
// Try and convert to string and then parse to a float64
|
|
strV := string(v)
|
|
result, err := strconv.ParseFloat(strV, 64)
|
|
if err != nil {
|
|
level.Info(logger).Log("msg", "Could not parse []byte", "err", err)
|
|
return math.NaN(), false
|
|
}
|
|
return result, true
|
|
case string:
|
|
result, err := strconv.ParseFloat(v, 64)
|
|
if err != nil {
|
|
level.Info(logger).Log("msg", "Could not parse string", "err", err)
|
|
return math.NaN(), false
|
|
}
|
|
return result, true
|
|
case bool:
|
|
if v {
|
|
return 1.0, true
|
|
}
|
|
return 0.0, true
|
|
case nil:
|
|
return math.NaN(), true
|
|
default:
|
|
return math.NaN(), false
|
|
}
|
|
}
|
|
|
|
// Convert database.sql types to uint64 for Prometheus consumption. Null types are mapped to 0. string and []byte
|
|
// types are mapped as 0 and !ok
|
|
func dbToUint64(t interface{}) (uint64, bool) {
|
|
switch v := t.(type) {
|
|
case uint64:
|
|
return v, true
|
|
case int64:
|
|
return uint64(v), true
|
|
case float64:
|
|
return uint64(v), true
|
|
case time.Time:
|
|
return uint64(v.Unix()), true
|
|
case []byte:
|
|
// Try and convert to string and then parse to a uint64
|
|
strV := string(v)
|
|
result, err := strconv.ParseUint(strV, 10, 64)
|
|
if err != nil {
|
|
level.Info(logger).Log("msg", "Could not parse []byte", "err", err)
|
|
return 0, false
|
|
}
|
|
return result, true
|
|
case string:
|
|
result, err := strconv.ParseUint(v, 10, 64)
|
|
if err != nil {
|
|
level.Info(logger).Log("msg", "Could not parse string", "err", err)
|
|
return 0, false
|
|
}
|
|
return result, true
|
|
case bool:
|
|
if v {
|
|
return 1, true
|
|
}
|
|
return 0, true
|
|
case nil:
|
|
return 0, true
|
|
default:
|
|
return 0, false
|
|
}
|
|
}
|
|
|
|
// Convert database.sql to string for Prometheus labels. Null types are mapped to empty strings.
|
|
func dbToString(t interface{}) (string, bool) {
|
|
switch v := t.(type) {
|
|
case int64:
|
|
return fmt.Sprintf("%v", v), true
|
|
case float64:
|
|
return fmt.Sprintf("%v", v), true
|
|
case time.Time:
|
|
return fmt.Sprintf("%v", v.Unix()), true
|
|
case nil:
|
|
return "", true
|
|
case []byte:
|
|
// Try and convert to string
|
|
return string(v), true
|
|
case string:
|
|
return v, true
|
|
case bool:
|
|
if v {
|
|
return "true", true
|
|
}
|
|
return "false", true
|
|
default:
|
|
return "", false
|
|
}
|
|
}
|
|
|
|
func parseFingerprint(url string) (string, error) {
|
|
dsn, err := pq.ParseURL(url)
|
|
if err != nil {
|
|
dsn = url
|
|
}
|
|
|
|
pairs := strings.Split(dsn, " ")
|
|
kv := make(map[string]string, len(pairs))
|
|
for _, pair := range pairs {
|
|
splitted := strings.SplitN(pair, "=", 2)
|
|
if len(splitted) != 2 {
|
|
return "", fmt.Errorf("malformed dsn %q", dsn)
|
|
}
|
|
// Newer versions of pq.ParseURL quote values so trim them off if they exist
|
|
key := strings.Trim(splitted[0], "'\"")
|
|
value := strings.Trim(splitted[1], "'\"")
|
|
kv[key] = value
|
|
}
|
|
|
|
var fingerprint string
|
|
|
|
if host, ok := kv["host"]; ok {
|
|
fingerprint += host
|
|
} else {
|
|
fingerprint += "localhost"
|
|
}
|
|
|
|
if port, ok := kv["port"]; ok {
|
|
fingerprint += ":" + port
|
|
} else {
|
|
fingerprint += ":5432"
|
|
}
|
|
|
|
return fingerprint, nil
|
|
}
|
|
|
|
func loggableDSN(dsn string) string {
|
|
pDSN, err := url.Parse(dsn)
|
|
if err != nil {
|
|
return "could not parse DATA_SOURCE_NAME"
|
|
}
|
|
// Blank user info if not nil
|
|
if pDSN.User != nil {
|
|
pDSN.User = url.UserPassword(pDSN.User.Username(), "PASSWORD_REMOVED")
|
|
}
|
|
|
|
return pDSN.String()
|
|
}
|