Skip to content

Commit

Permalink
Added the regex functionality
Browse files Browse the repository at this point in the history
  • Loading branch information
Mathieu committed Jan 2, 2018
1 parent 4e92259 commit 880ea50
Show file tree
Hide file tree
Showing 3 changed files with 108 additions and 29 deletions.
123 changes: 95 additions & 28 deletions aws.go
Original file line number Diff line number Diff line change
@@ -1,12 +1,13 @@
package main

import (
"fmt"
"github.com/aws/aws-sdk-go/aws"
"github.com/aws/aws-sdk-go/aws/session"
"github.com/aws/aws-sdk-go/service/cloudwatch"
"github.com/prometheus/client_golang/prometheus"
"time"
"fmt"
"regexp"
)

func getLatestDatapoint(datapoints []*cloudwatch.Datapoint) *cloudwatch.Datapoint {
Expand All @@ -29,7 +30,6 @@ func scrape(collector *cwCollector, ch chan<- prometheus.Metric) {
}))

svc := cloudwatch.New(session)

for m := range collector.Template.Metrics {
metric := &collector.Template.Metrics[m]

Expand All @@ -48,6 +48,9 @@ func scrape(collector *cwCollector, ch chan<- prometheus.Metric) {
Unit: nil,
}

dimensions:=[]*cloudwatch.Dimension{}


for _, stat := range metric.ConfMetric.Statistics {
params.Statistics = append(params.Statistics, aws.String(stat))
}
Expand All @@ -64,7 +67,7 @@ func scrape(collector *cwCollector, ch chan<- prometheus.Metric) {
dimValue = collector.Target
}

params.Dimensions = append(params.Dimensions, &cloudwatch.Dimension{
dimensions = append(dimensions, &cloudwatch.Dimension{
Name: aws.String(dim),
Value: aws.String(dimValue),
})
Expand All @@ -73,43 +76,107 @@ func scrape(collector *cwCollector, ch chan<- prometheus.Metric) {
}
}

labels = append(labels, collector.Template.Task.Name)
if (len(dimensions)>0){
labels = append(labels, collector.Template.Task.Name)
params.Dimensions=dimensions
scrapeSingleDataPoint(collector,ch,params,metric,labels,svc)
continue
}

// Call CloudWatch to gather the datapoints
resp, err := svc.GetMetricStatistics(params)
totalRequests.Inc()
// Get all the metric to select the ones who'll match the regex
result, err := svc.ListMetrics(&cloudwatch.ListMetricsInput{
MetricName: aws.String(metric.ConfMetric.Name),
Namespace: aws.String(metric.ConfMetric.Namespace),
})

if err != nil {
collector.ErroneousRequests.Inc()
fmt.Println(err)
continue
}

//This map will hold dimensions name which has been already collected
valueCollected := map[string]bool{}

// There's nothing in there, don't publish the metric
if len(resp.Datapoints) == 0 {
continue
}
for dimensions := range metric.ConfMetric.DimensionsSelectRegex {
dimRegex := metric.ConfMetric.DimensionsSelectRegex[dimensions]

// Pick the latest datapoint
dp := getLatestDatapoint(resp.Datapoints)
if dp.Sum != nil {
ch <- prometheus.MustNewConstMetric(metric.Desc, metric.ValType, float64(*dp.Sum), labels...)
}
// Replace $_target token by the actual URL target
if dimRegex == "$_target" {
dimRegex = collector.Target
}

if dp.Average != nil {
ch <- prometheus.MustNewConstMetric(metric.Desc, metric.ValType, float64(*dp.Average), labels...)
//Loop through all the dimensions for the metric given
for _,met := range result.Metrics {
for _,dim := range met.Dimensions {

//Select the one which match the regex
match,_:=regexp.MatchString(dimRegex,*dim.Value)
if _, ok := valueCollected[*dim.Value]; match && !ok {
//Create the request and send it to the prometheus lib
labels := make([]string, 0, len(metric.LabelNames))

params.Dimensions = []*cloudwatch.Dimension{}
params.Dimensions = append(params.Dimensions, &cloudwatch.Dimension{
Name: aws.String(*dim.Name),
Value: aws.String(*dim.Value),
})

labels = append(labels, *dim.Value)

labels = append(labels, collector.Template.Task.Name)
scrapeSingleDataPoint(collector,ch,params,metric,labels,svc)

valueCollected[*dim.Value]=true
}
}


}
}

if dp.Maximum != nil {
ch <- prometheus.MustNewConstMetric(metric.Desc, metric.ValType, float64(*dp.Maximum), labels...)
}

if dp.Minimum != nil {
ch <- prometheus.MustNewConstMetric(metric.Desc, metric.ValType, float64(*dp.Minimum), labels...)
}

}
}

if dp.SampleCount != nil {
ch <- prometheus.MustNewConstMetric(metric.Desc, metric.ValType, float64(*dp.SampleCount), labels...)
}
//Send a single dataPoint to the Prometheus lib
func scrapeSingleDataPoint(collector *cwCollector, ch chan<- prometheus.Metric,params *cloudwatch.GetMetricStatisticsInput,metric *cwMetric,labels []string,svc *cloudwatch.CloudWatch) error {

resp, err := svc.GetMetricStatistics(params)
totalRequests.Inc()

if err != nil {
collector.ErroneousRequests.Inc()
fmt.Println(err)
return err
}

// There's nothing in there, don't publish the metric
if len(resp.Datapoints) == 0 {
return nil
}
// Pick the latest datapoint
dp := getLatestDatapoint(resp.Datapoints)


if dp.Sum != nil {
ch <- prometheus.MustNewConstMetric(metric.Desc, metric.ValType, float64(*dp.Sum), labels...)
}

if dp.Average != nil {
ch <- prometheus.MustNewConstMetric(metric.Desc, metric.ValType, float64(*dp.Average), labels...)
}

if dp.Maximum != nil {
ch <- prometheus.MustNewConstMetric(metric.Desc, metric.ValType, float64(*dp.Maximum), labels...)
}

if dp.Minimum != nil {
ch <- prometheus.MustNewConstMetric(metric.Desc, metric.ValType, float64(*dp.Minimum), labels...)
}

if dp.SampleCount != nil {
ch <- prometheus.MustNewConstMetric(metric.Desc, metric.ValType, float64(*dp.SampleCount), labels...)
}
return nil
}
13 changes: 12 additions & 1 deletion config.yml
Original file line number Diff line number Diff line change
Expand Up @@ -50,4 +50,15 @@ tasks:
VpnId: [$_target]
aws_metric_name: TunnelDataOut
aws_statistics: [Average]
range_seconds: 3600
range_seconds: 3600
- name: lambda_duration
default_region: eu-west-1
metrics:
- aws_namespace: "AWS/Lambda"
aws_dimensions: [FunctionName]
aws_dimensions_select_regex:
FunctionName: .*
aws_metric_name: Duration
aws_statistics: [Maximum]
range_seconds: 1728000
period_seconds: 3000
1 change: 1 addition & 0 deletions config/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ type Metric struct {
Statistics []string `yaml:"aws_statistics"`
Dimensions []string `yaml:"aws_dimensions,omitempty"`
DimensionsSelect map[string][]string `yaml:"aws_dimensions_select,omitempty"`
DimensionsSelectRegex map[string]string `yaml:"aws_dimensions_select_regex,omitempty"`
DimensionsSelectParam map[string][]string `yaml:"aws_dimensions_select_param,omitempty"`

RangeSeconds int `yaml:"range_seconds,omitempty"`
Expand Down

0 comments on commit 880ea50

Please sign in to comment.