Docker does not provide the rootfs path through docker inspect or statefile and the path is dependent on the storage driver being used. Instead of enumerating the storage drivers, we pick a pid from the container and get the config from /proc/pid/root. Although a bit expensive, this method works for non-docker containers too.
158 lines
4.7 KiB
Go
158 lines
4.7 KiB
Go
// Copyright 2015 Google Inc. All Rights Reserved.
|
|
//
|
|
// 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 collector
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/google/cadvisor/info/v1"
|
|
)
|
|
|
|
type GenericCollector struct {
|
|
//name of the collector
|
|
name string
|
|
|
|
//holds information extracted from the config file for a collector
|
|
configFile Config
|
|
|
|
//holds information necessary to extract metrics
|
|
info *collectorInfo
|
|
}
|
|
|
|
type collectorInfo struct {
|
|
//minimum polling frequency among all metrics
|
|
minPollingFrequency time.Duration
|
|
|
|
//regular expresssions for all metrics
|
|
regexps []*regexp.Regexp
|
|
}
|
|
|
|
//Returns a new collector using the information extracted from the configfile
|
|
func NewCollector(collectorName string, configFile []byte) (*GenericCollector, error) {
|
|
var configInJSON Config
|
|
err := json.Unmarshal(configFile, &configInJSON)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
//TODO : Add checks for validity of config file (eg : Accurate JSON fields)
|
|
|
|
if len(configInJSON.MetricsConfig) == 0 {
|
|
return nil, fmt.Errorf("No metrics provided in config")
|
|
}
|
|
|
|
minPollFrequency := time.Duration(0)
|
|
regexprs := make([]*regexp.Regexp, len(configInJSON.MetricsConfig))
|
|
|
|
for ind, metricConfig := range configInJSON.MetricsConfig {
|
|
// Find the minimum specified polling frequency in metric config.
|
|
if metricConfig.PollingFrequency != 0 {
|
|
if minPollFrequency == 0 || metricConfig.PollingFrequency < minPollFrequency {
|
|
minPollFrequency = metricConfig.PollingFrequency
|
|
}
|
|
}
|
|
|
|
regexprs[ind], err = regexp.Compile(metricConfig.Regex)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Invalid regexp %v for metric %v", metricConfig.Regex, metricConfig.Name)
|
|
}
|
|
}
|
|
|
|
// Minimum supported polling frequency is 1s.
|
|
minSupportedFrequency := 1 * time.Second
|
|
if minPollFrequency < minSupportedFrequency {
|
|
minPollFrequency = minSupportedFrequency
|
|
}
|
|
|
|
return &GenericCollector{
|
|
name: collectorName,
|
|
configFile: configInJSON,
|
|
info: &collectorInfo{
|
|
minPollingFrequency: minPollFrequency,
|
|
regexps: regexprs},
|
|
}, nil
|
|
}
|
|
|
|
//Returns name of the collector
|
|
func (collector *GenericCollector) Name() string {
|
|
return collector.name
|
|
}
|
|
|
|
//Returns collected metrics and the next collection time of the collector
|
|
func (collector *GenericCollector) Collect() (time.Time, []v1.Metric, error) {
|
|
currentTime := time.Now()
|
|
nextCollectionTime := currentTime.Add(time.Duration(collector.info.minPollingFrequency * time.Second))
|
|
|
|
uri := collector.configFile.Endpoint
|
|
response, err := http.Get(uri)
|
|
if err != nil {
|
|
return nextCollectionTime, nil, err
|
|
}
|
|
|
|
defer response.Body.Close()
|
|
|
|
pageContent, err := ioutil.ReadAll(response.Body)
|
|
if err != nil {
|
|
return nextCollectionTime, nil, err
|
|
}
|
|
|
|
metrics := make([]v1.Metric, len(collector.configFile.MetricsConfig))
|
|
var errorSlice []error
|
|
|
|
for ind, metricConfig := range collector.configFile.MetricsConfig {
|
|
matchString := collector.info.regexps[ind].FindStringSubmatch(string(pageContent))
|
|
if matchString != nil {
|
|
if metricConfig.Units == "float" {
|
|
regVal, err := strconv.ParseFloat(strings.TrimSpace(matchString[1]), 64)
|
|
if err != nil {
|
|
errorSlice = append(errorSlice, err)
|
|
}
|
|
metrics[ind].FloatPoints = []v1.FloatPoint{
|
|
{Value: regVal, Timestamp: currentTime},
|
|
}
|
|
} else if metricConfig.Units == "integer" || metricConfig.Units == "int" {
|
|
regVal, err := strconv.ParseInt(strings.TrimSpace(matchString[1]), 10, 64)
|
|
if err != nil {
|
|
errorSlice = append(errorSlice, err)
|
|
}
|
|
metrics[ind].IntPoints = []v1.IntPoint{
|
|
{Value: regVal, Timestamp: currentTime},
|
|
}
|
|
|
|
} else {
|
|
errorSlice = append(errorSlice, fmt.Errorf("Unexpected value of 'units' for metric '%v' in config ", metricConfig.Name))
|
|
}
|
|
} else {
|
|
errorSlice = append(errorSlice, fmt.Errorf("No match found for regexp: %v for metric '%v' in config", metricConfig.Regex, metricConfig.Name))
|
|
}
|
|
|
|
metrics[ind].Name = metricConfig.Name
|
|
if metricConfig.MetricType == "gauge" {
|
|
metrics[ind].Type = v1.MetricGauge
|
|
} else if metricConfig.MetricType == "counter" {
|
|
metrics[ind].Type = v1.MetricCumulative
|
|
}
|
|
}
|
|
|
|
return nextCollectionTime, metrics, compileErrors(errorSlice)
|
|
}
|