mirror of
https://github.com/prometheus-community/postgres_exporter
synced 2025-04-20 22:15:28 +00:00
It's possible that incoming labels will contain invalid UTF-8 characters. This results in a panic. This fix sanitizes the label's string to ensure only valid UTF-8 characters are included, by replacing invalid characters with � (REPLACEMENT CHARACTER) Signed-off-by: Cooper Worobetz <cooper@worobetz.ca>
221 lines
4.7 KiB
Go
221 lines
4.7 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/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 {
|
||
logger.Info("Could not parse []byte", "err", err)
|
||
return math.NaN(), false
|
||
}
|
||
return result, true
|
||
case string:
|
||
result, err := strconv.ParseFloat(v, 64)
|
||
if err != nil {
|
||
logger.Info("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 {
|
||
logger.Info("Could not parse []byte", "err", err)
|
||
return 0, false
|
||
}
|
||
return result, true
|
||
case string:
|
||
result, err := strconv.ParseUint(v, 10, 64)
|
||
if err != nil {
|
||
logger.Info("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 strings.ToValidUTF8(v, "<22>"), 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()
|
||
}
|