diff --git a/.gitignore b/.gitignore index 1c2d52b..00d0a07 100644 --- a/.gitignore +++ b/.gitignore @@ -1 +1,2 @@ .idea/* +.DS_store diff --git a/Makefile b/Makefile index 3664c77..327f2ea 100644 --- a/Makefile +++ b/Makefile @@ -11,13 +11,9 @@ test: $(GOBIN) test -v ./... .PHONY: build -build: $(BINDIR)/codegen +build: make -C tscli -$(BINDIR)/codegen: $(SCRIPTDIR)/codegen.go - @mkdir -p $(BINDIR) - GOOS=linux GOARCH=amd64 $(GOBIN) build -o $(BINDIR)/codegen $< - .PHONY: install install: build cp $(BINDIR)/tscli /usr/local/bin/ @@ -26,11 +22,3 @@ install: build clean: rm -rf $(BINDIR) -CODEGENFILE="set_aggregation_opts.go" -.PHONY: codegen -codegen: - $(BINDIR)/codegen -output ./tsplot/$(CODEGENFILE) - -.PHONY: codegendiff -codegendiff: - diff ./tsplot/$(CODEGENFILE) <($(BINDIR)/codegen -stdout) diff --git a/README.md b/README.md index 203290a..9a275a9 100644 --- a/README.md +++ b/README.md @@ -2,76 +2,40 @@ [![Go Report Card](https://goreportcard.com/badge/github.com/bitly/tsplot)](https://goreportcard.com/report/github.com/bitly/tsplot) [![Go Reference](https://pkg.go.dev/badge/github.com/bitly/tsplot.svg)](https://pkg.go.dev/github.com/bitly/tsplot) -This package provides a method of querying for raw time series data from the GCM APIs and additionally plotting that data for use in other applications. - -This came to be due to what we consider a small limitation in the Google APIs which require us to re-draw graphs to include them in other applications such as -Slack bots. There is no facility in the Google API that provides a PNG of already graphed data. - ## Authentication This package makes no effort to assist in authentication to the Google APIs. Instead, it will expect the caller to supply an authenticated client. More information on authentication can be found in the official [Google Cloud documentation](https://cloud.google.com/docs/authentication). - -## Query -tsplot helps to facilitate easy querying of the Google Cloud Monitoring API for time series matching the supplied criteria. -In addition it provides methods of overriding certain aspects of the query. - -For example, the following code snippet will return a single time series for the following metric descriptor: `custom.googleapis.com/opencensus/fishnet/queuereader_fishnet/messages_total`. ``` func main() { - ... snip ... +... snip ... start := time.Now().Add(-1 * time.Hour) end := time.Now() - mq := &tsplot.NewMetricQuery( - "bitly-gcp-prod", // GCP project - "custom.googleapis.com/opencensus/fishent/queuereader_fishnet/messages_total", // metric descriptor - &start, // start of time window - &end, // end of time window - ) - - // disable cross series reducer (MEAN reduction is default) - query.Set_REDUCE_NONE() - - // set different alignment window. (Default is 1 minute) - query.SetAlignmentPeriod(time.Minute * 2) - - tsi, err := mq.PerformWithClient(client) // client is provided by user - if err != nil { - fmt.Printf("error performing query: %v\n", err) - } -} -``` - -## Plotting -To plot the data, tsplot leverages the open source package [gonum/plot](github.com/gonum/plot) to create a graph and plot the data for a given time series. - -The example below creates a new graph containing a singular time series, plots it, and saves the resulting plot to disk. -``` -func main() { - - ... snip ... - - ts := tsplot.TimeSeries{} - - // optionally iterate over returned time series - timeSeries, _ := tsi.Next() - ts[metric] = ts.GetPoints() - - // create the plot with some formatting options - p, err := ts.Plot([]tsplot.PlotOption{ - tsplot.WithXAxisName("UTC"), - tsplot.WIthGrid(colornames.Darkgrey), - tsplot.WithTitle(metric)}...) - if err != nil { - return err + // create new request + request := monitoringpb.ListTimeSeriesRequest{ + Name: fmt.Sprintf("projects/%s", project), + Filter: query, + Interval: &monitoringpb.TimeInterval{ + EndTime: timestamppb.New(et), + StartTime: timestamppb.New(st), + }, + Aggregation: nil, + SecondaryAggregation: nil, + View: monitoringpb.ListTimeSeriesRequest_FULL, } - - // optionally save the plot to disk - p.Save(8*vg.Inch, 4*vg.Inch, "./my-graph.png") + + // execute the request and get the response from Google APIs + tsi := GoogleCloudMonitoringClient.ListTimeSeries(context.Background(), request) + + // Create the plot from the GAPI TimeSeries + plot, _ := tsplot.NewPlotFromTimeSeriesIterator(tsi) + + // Save the new plot to disk. + plot.Save(8*vg.Inch, 4*vg.Inch, "my_plot.png") } ``` @@ -79,7 +43,3 @@ func main() { Query across multiple time series with mean reducer: ![graph1](sample/1.png) -### Graph Color Scheme -I'm not a UX designer, but I have selected colors that I find higher contrast -and easier to see. I am basing this completely off my colorblindness which is -unique to me. Improvements to the color palette used are welcome. diff --git a/scripts/codegen.go b/scripts/codegen.go deleted file mode 100644 index 508fd0b..0000000 --- a/scripts/codegen.go +++ /dev/null @@ -1,67 +0,0 @@ -package main -/* -codegen.go creates Set_* API methods for Google Cloud Monitoring aggregation alignment and reduction options. -Leverages the exported variables in the Google monitoring/v3 package (common.pb.go) to ensure that all -alignment and reduction options are covered. -*/ - -import ( - "flag" - "log" - "os" - "text/template" - - monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3" -) - -var rawTpl = `package tsplot -/* -DO NOT EDIT -Generated by codegen.go -https://github.com/googleapis/go-genproto/blob/0135a39c27378c1b903c75204eff61a060be5eb7/googleapis/monitoring/v3/common.pb.go -*/ -{{range $name, $value := .Aligners}} -func (mq *MetricQuery) Set_{{$name}}() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner({{$value}})) -} -{{end -}} -{{range $name, $value := .Reducers}} -func (mq *MetricQuery) Set_{{$name}}() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer({{$value}})) -} -{{end}} -` - -func main() { - outFileName := flag.String("output", "./tsplot/set_aggregation_opts.go", "Output path of generated file.") - toStdout := flag.Bool("stdout", false, "Toggle output to STDOUT.") - flag.Parse() - - var outFile *os.File - var fileErr error - - if *toStdout { - outFile = os.Stdout - } else { - outFile, fileErr = os.OpenFile(*outFileName, os.O_CREATE|os.O_RDWR|os.O_TRUNC, 0644) - if fileErr != nil { - log.Print(fileErr) - } - defer outFile.Close() - } - - values := struct { - Aligners map[string]int32 - Reducers map[string]int32 - }{ - Aligners: monitoringpb.Aggregation_Aligner_value, - Reducers: monitoringpb.Aggregation_Reducer_value, - } - - t := template.New("codegen") - pt := template.Must(t.Parse(rawTpl)) - err := pt.Execute(outFile, values) - if err != nil { - log.Fatal(err) - } -} diff --git a/tscli/Makefile b/tscli/Makefile index 37fcc60..a70ff11 100644 --- a/tscli/Makefile +++ b/tscli/Makefile @@ -10,4 +10,4 @@ default: build .PHONY: build build: @mkdir -p $(BINDIR) - GOOS=linux GOARCH=amd64 $(GOBIN) build -o $(BINDIR)/tscli + $(GOBIN) build -o $(BINDIR)/tscli diff --git a/tscli/main.go b/tscli/main.go index f0c07ac..ebbcdf1 100644 --- a/tscli/main.go +++ b/tscli/main.go @@ -4,6 +4,11 @@ import ( "context" "errors" "fmt" + "github.com/bitly/tsplot/tsplot" + "gonum.org/v1/plot/vg" + "google.golang.org/protobuf/types/known/durationpb" + "google.golang.org/protobuf/types/known/timestamppb" + "log" "os" "regexp" "strconv" @@ -11,12 +16,9 @@ import ( "time" monitoring "cloud.google.com/go/monitoring/apiv3/v2" - "github.com/bitly/tsplot/tsplot" "github.com/spf13/cobra" - "golang.org/x/image/colornames" - "gonum.org/v1/plot/vg" - "google.golang.org/api/iterator" "google.golang.org/api/option" + monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3" ) const GAP = "GOOGLE_APPLICATION_CREDENTIALS" @@ -48,33 +50,21 @@ from Google Cloud Monitoring (formerly StackDriver). RunE: executeQuery, } - project string - app string - service string - metric string - startTime string - endTime string - queryOverride string - outDir string - reduce bool - justPrint bool + project string + query string + startTime string + endTime string + outDir string ) func init() { rootCmd.Flags().StringVarP(&project, "project", "p", "", "GCP Project.") - rootCmd.Flags().StringVarP(&app, "app", "a", "", "The (Bitly) application. Usually top level directory") - rootCmd.Flags().StringVarP(&service, "service", "s", "", "The (Bitly) service. Service directory found under application directory.") - rootCmd.Flags().StringVarP(&metric, "metric", "m", "", "The metric.") + rootCmd.Flags().StringVarP(&query, "query", "m", "", "The query filter.") rootCmd.Flags().StringVar(&startTime, "start", "", "Start time of window for which the query returns time series data for. Hours or minutes accepted, i.e: -5h or -5m.") rootCmd.Flags().StringVar(&endTime, "end", "now", "End of the time window for which the query returns time series data for. Hours or minutes accepted, i.e: -5h or -5m or now.") - rootCmd.Flags().BoolVar(&justPrint, "print-raw", false, "Only print time series data and exit.") - rootCmd.Flags().BoolVar(&reduce, "reduce", false, "Use a time series reducer to return a single averaged result.") - rootCmd.Flags().StringVar(&queryOverride, "query-override", "", "Override the default query. Must be a full valid query. Metric flag is not used.") rootCmd.Flags().StringVarP(&outDir, "output", "o", "", "Specify output directory for resulting plot. Defaults to current working directory.") rootCmd.MarkFlagRequired("project") - rootCmd.MarkFlagRequired("app") - rootCmd.MarkFlagRequired("service") - rootCmd.MarkFlagRequired("metric") + //rootCmd.MarkFlagRequired("query") rootCmd.MarkFlagRequired("start") } @@ -95,10 +85,6 @@ func auth(cmd *cobra.Command, args []string) error { func executeQuery(cmd *cobra.Command, args []string) error { - if metric != "" && queryOverride != "" { - fmt.Println("warn: both --metric and --query-override flag used. Favoring --query-override.") - } - if !timeFormatOK(startTime) { return errors.New("err validating start time format") } @@ -114,69 +100,34 @@ func executeQuery(cmd *cobra.Command, args []string) error { st := parseTime(startTime) et := parseTime(endTime) - query := tsplot.NewMetricQuery( - project, - fmt.Sprintf("custom.googleapis.com/opencensus/%s/%s/%s", app, service, metric), - &st, - &et, - ) - - if queryOverride != "" { - query.SetQueryFilter(queryOverride) - } - - if !reduce { - query.Set_REDUCE_NONE() - } - - tsi, err := query.PerformWithClient(GoogleCloudMonitoringClient) + request := &monitoringpb.ListTimeSeriesRequest{ + Name: fmt.Sprintf("projects/%s", project), + Filter: `resource.type = "global" AND metric.type = "custom.googleapis.com/opencensus/fishnet/queuereader_fishnet/messages_total"`, + Interval: &monitoringpb.TimeInterval{ + EndTime: timestamppb.New(et), + StartTime: timestamppb.New(st), + }, + Aggregation: &monitoringpb.Aggregation{ + AlignmentPeriod: durationpb.New(time.Minute * 1), + PerSeriesAligner: monitoringpb.Aggregation_ALIGN_RATE, + CrossSeriesReducer: monitoringpb.Aggregation_REDUCE_MEAN, + }, + View: monitoringpb.ListTimeSeriesRequest_FULL, + } + + tsi := GoogleCloudMonitoringClient.ListTimeSeries(context.Background(), request) + // todo: handle multiple time series + ts, err := tsi.Next() if err != nil { - return err + log.Fatal(err) } - ts := tsplot.TimeSeries{} - for { - timeSeries, err := tsi.Next() - if err != nil { - if err == iterator.Done { - break - } - return err - } - - // todo: implement "just-print" mode for multiple time series - //if justPrint { - // fmt.Printf("%v", timeSeries) - // return nil - //} - - // key helps to fill out legend. - // Here we are grabbing the pod name. - key := timeSeries.GetMetric().GetLabels()["opencensus_task"] - if key == "" { - // Labels we want to use don't necessarily exist when a cross series reducer has been used. - // So we can just use "mean" in the legend. - key = "mean" - } - ts[key] = timeSeries.GetPoints() - } - - p, err := ts.Plot([]tsplot.PlotOption{tsplot.WithXAxisName("UTC"), - tsplot.WithXTimeTicks(time.Kitchen), - tsplot.WithFontSize(float64(12)), - tsplot.WithGrid(colornames.Darkgrey), - tsplot.WithTitle(metric)}...) + plot, err := tsplot.NewPlotFromTimeSeries(*ts) if err != nil { - return err + log.Fatal(err) } - - if outDir == "" { - outDir, _ = os.Getwd() - } - saveFile := fmt.Sprintf("%s/%s-%s.png", outDir, service, metric) - p.Save(8*vg.Inch, 4*vg.Inch, saveFile) - - return nil + saveFile := fmt.Sprintf("%s/test.png", "/tmp/") + return plot.Save(8*vg.Inch, 4*vg.Inch, saveFile) } func timeFormatOK(s string) bool { diff --git a/tsplot/options.go b/tsplot/options.go index 7410bcf..f930eb6 100644 --- a/tsplot/options.go +++ b/tsplot/options.go @@ -2,13 +2,12 @@ package tsplot import ( "image/color" - "time" "gonum.org/v1/plot" "gonum.org/v1/plot/font" "gonum.org/v1/plot/plotter" + "gonum.org/v1/plot/vg" monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3" - "google.golang.org/protobuf/types/known/durationpb" ) // PlotOption defines the type used to configure the underlying *plot.Plot. @@ -107,27 +106,12 @@ func ApplyDefaultHighContrast(p *plot.Plot) { } } -// aggregationOption defines the type used to configure the underlying *monitoringpb.Aggregation. -// A function that returns aggregationOption can be used to set options on the *monitoringpb.Aggregation. -type aggregationOption func(agg *monitoringpb.Aggregation) - -// withAlignmentPeriod sets the duration of the aggregation's alignment period. -func withAlignmentPeriod(d time.Duration) aggregationOption { - return func(agg *monitoringpb.Aggregation) { - agg.AlignmentPeriod = durationpb.New(d) - } -} - -// withPerSeriesAligner sets the alignment method used for the time series. -func withPerSeriesAligner(aligner monitoringpb.Aggregation_Aligner) aggregationOption { - return func(agg *monitoringpb.Aggregation) { - agg.PerSeriesAligner = aligner - } -} - -// withCrossSeriesReducer sets the reduction method used for the time series. -func withCrossSeriesReducer(reducer monitoringpb.Aggregation_Reducer) aggregationOption { - return func(agg *monitoringpb.Aggregation) { - agg.CrossSeriesReducer = reducer +// WithLineFromPoints creates a *plotter.Line from the passed in data points and adds +// it to the plot. +func WithLineFromPoints(pts []*monitoringpb.Point) PlotOption { + return func(p *plot.Plot) { + line, _ := createLine(pts) + line.Width = vg.Points(1) + p.Add(line) } } diff --git a/tsplot/plot.go b/tsplot/plot.go index 099549f..873c795 100644 --- a/tsplot/plot.go +++ b/tsplot/plot.go @@ -1,69 +1,49 @@ package tsplot import ( - "errors" - "gonum.org/v1/plot" "gonum.org/v1/plot/plotter" - "gonum.org/v1/plot/vg" monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3" ) -// TimeSeries is a map representation of unique names to time series. -type TimeSeries map[string][]*monitoringpb.Point - -// Plot builds and returns a *plot.Plot. If the TimeSeriesGroup field -// in TimeSeriesPlot is empty, a nil plot is returned as well as an error. -// This is also the case if an error is encountered building the line from the XY coordinates. -func (ts TimeSeries) Plot(opts ...PlotOption) (*plot.Plot, error) { - - if len(ts) == 0 { - return nil, errors.New("no data to plot") +func NewPlotFromTimeSeries(ts monitoringpb.TimeSeries, opts ...PlotOption) (*plot.Plot, error) { + var maximum float64 + points := ts.GetPoints() + for _, v := range points { + cur := v.GetValue().GetDoubleValue() + if cur > maximum { + maximum = cur + } } - + opts = append(opts, WithLineFromPoints(points)) p := plot.New() - - // default high contrast - ApplyDefaultHighContrast(p) - - // user overrides for _, opt := range opts { opt(p) } - lineColors := DefaultColors_HighContrast.LineColors - - // create a unique line for each time series - // each line should have a unique color and entry - // in the legend. - // current limit = 4 - limit := len(lineColors) - 1 - for name, series := range ts { - if limit < 0 { - break - } - line, err := createLine(series) - if err != nil { - return nil, err - } - - // width of the line - line.Width = vg.Points(2) - - // color the line - line.Color = lineColors[limit] - - // add to legend - p.Legend.Add(name, line) - - // add to chart - p.Add(line) - limit-- - } - + p.Y.Max = maximum + 200 return p, nil } +/* +func NewPlotFromTimeSeriesIterator(ts monitoring.TimeSeriesIterator) (*plot.Plot, error) { + + var ts *monitoringpb.TimeSeries + for { + timeSeries, err := tsi.Next() + if err != nil { + if err == iterator.Done { + break + } + return err + } + ts = timeSeries + } + + return nil, nil +} +*/ + // createLine creates a line from data points. func createLine(dataPoints []*monitoringpb.Point) (*plotter.Line, error) { var XYs plotter.XYs diff --git a/tsplot/query.go b/tsplot/query.go deleted file mode 100644 index cc5ae41..0000000 --- a/tsplot/query.go +++ /dev/null @@ -1,118 +0,0 @@ -package tsplot - -import ( - "context" - "errors" - "fmt" - "time" - - monitoring "cloud.google.com/go/monitoring/apiv3/v2" - monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3" - "google.golang.org/protobuf/types/known/durationpb" - "google.golang.org/protobuf/types/known/timestamppb" -) - -// DefaultQueryFilter the query filter used if no overrides are given. -const DefaultQueryFilter = "resource.type = \"global\" AND metric.type = \"%s\"" - -// MetricQuery is a type that encapsulates the various parts that are used to form a ListTimeSeriesRequest. -// If EndTime is not provided, it will be defaulted to the current time. -// -// Required Fields: -// Project -// MetricDescriptor -// StartTime -type MetricQuery struct { - Project string - MetricDescriptor string - StartTime *time.Time - EndTime *time.Time - - queryFilter string - aggregation *[]aggregationOption -} - -// NewMetricQuery creates a new MetricQuery type with the aggregation opts initialized. -func NewMetricQuery(project, metric string, startTime, endTime *time.Time) *MetricQuery { - return &MetricQuery{ - Project: project, - MetricDescriptor: metric, - StartTime: startTime, - EndTime: endTime, - aggregation: &[]aggregationOption{}, - } -} - -// SetQueryFilter provides a hook to modify the metric query filter. -func (mq *MetricQuery) SetQueryFilter(queryFilter string) { - mq.queryFilter = queryFilter -} - -// SetAlignmentPeriod sets the alignment duration. -func (mq *MetricQuery) SetAlignmentPeriod(d time.Duration) { - *mq.aggregation = append(*mq.aggregation, withAlignmentPeriod(d)) -} - -// request builds and returns a *monitoringpb.ListTimeSeriesRequest. -// If there is not enough information to build the request an error is returned. -func (mq *MetricQuery) request() (*monitoringpb.ListTimeSeriesRequest, error) { - - var tsreq monitoringpb.ListTimeSeriesRequest - - if mq.Project == "" { - return nil, errors.New("MetricQuery missing GCE Project") - } - - if mq.MetricDescriptor == "" && mq.queryFilter == "" { - return nil, errors.New("MetricQuery missing MetricDescriptor") - } - - if mq.StartTime == nil { - return nil, errors.New("start time has not been provided") - } - - now := time.Now() - if mq.EndTime == nil { - mq.EndTime = &now - } - - // Complete override of timeSeriesRequestFilter. Use verbatim. - // Resolves: https://github.com/bitly/tsplot/issues/9 - timeSeriesRequestFilter := fmt.Sprintf(DefaultQueryFilter, mq.MetricDescriptor) - if mq.queryFilter != "" { - timeSeriesRequestFilter = mq.queryFilter - } - - tsreq = monitoringpb.ListTimeSeriesRequest{ - Name: fmt.Sprintf("projects/%s", mq.Project), - Filter: timeSeriesRequestFilter, - Interval: &monitoringpb.TimeInterval{ - EndTime: timestamppb.New(*mq.EndTime), - StartTime: timestamppb.New(*mq.StartTime), - }, - Aggregation: &monitoringpb.Aggregation{ - AlignmentPeriod: durationpb.New(time.Minute * 1), - PerSeriesAligner: monitoringpb.Aggregation_ALIGN_RATE, - CrossSeriesReducer: monitoringpb.Aggregation_REDUCE_MEAN, - }, - View: monitoringpb.ListTimeSeriesRequest_FULL, - } - - for _, opt := range *mq.aggregation { - opt(tsreq.Aggregation) - } - - return &tsreq, nil -} - -// PerformWithClient sends the MetricQuery.ListTimeSeriesRequest to the Google Cloud Monitoring API. -// If the request has not been built yet, i.e: BuildRequest() has not been called on the MetricQuery, -// an error will be returned. A Google Cloud Monitoring client is required to be passed in as a parameter -// if authentication has not been set up on the client, an error will result from the call. -func (mq *MetricQuery) PerformWithClient(client *monitoring.MetricClient) (*monitoring.TimeSeriesIterator, error) { - request, err := mq.request() - if err != nil { - return nil, err - } - return client.ListTimeSeries(context.Background(), request), nil -} diff --git a/tsplot/query_test.go b/tsplot/query_test.go deleted file mode 100644 index 2ef438b..0000000 --- a/tsplot/query_test.go +++ /dev/null @@ -1,98 +0,0 @@ -package tsplot - -import ( - "testing" - "time" - - monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3" -) - -func TestMetricQuery_BuildRequest(t *testing.T) { - tests := []struct { - desc string - in *MetricQuery - out *MetricQuery - expectErr bool - }{ - { - desc: "fail to build request, missing Project", - in: &MetricQuery{ - MetricDescriptor: "compute.googleapis.com/instance/cpu/usage_time", - }, - out: &MetricQuery{}, - expectErr: true, - }, - { - desc: "fail to build request, missing MetricDescriptor", - in: &MetricQuery{ - Project: "my-project", - }, - out: &MetricQuery{}, - expectErr: true, - }, - } - - for _, test := range tests { - _, err := test.in.request() - if err != nil && !test.expectErr { - t.Errorf("got unexpected err: %v\n", err) - } - } -} - -func TestMetricQuery_SetQueryFilter(t *testing.T) { - expectedFilter := "some advanced query" - st := time.Now().Add(-1 * time.Hour) - et := time.Now() - query := NewMetricQuery("bitly-gcp-prod", "", &st, &et) - query.SetQueryFilter(expectedFilter) - - tsr, err := query.request() - if err != nil { - t.Error(err) - } - - filter := tsr.GetFilter() - if filter != expectedFilter { - t.Fatalf("query filter not overriden. got: %s, expected: %s", filter, expectedFilter) - } -} - -func TestMetricQuery_SetAlignmentPeriod(t *testing.T) { - expectedAlignmentPeriod := time.Minute * 10 - st := time.Now().Add(-1 * time.Hour) - et := time.Now() - query := NewMetricQuery("bitly-gcp-prod", "some metric", &st, &et) - query.SetAlignmentPeriod(expectedAlignmentPeriod) - - req, err := query.request() - if err != nil { - t.Error(err) - } - - alignmentPeriod := req.GetAggregation().GetAlignmentPeriod().GetSeconds() - if req.GetAggregation().GetAlignmentPeriod().GetSeconds() != int64(expectedAlignmentPeriod.Seconds()) { - t.Fatalf("alignment period not overriden. got %d, expected: %s", alignmentPeriod, expectedAlignmentPeriod) - } -} - -func TestMetricQuery_AggregationOptions(t *testing.T) { - st := time.Now().Add(-1 * time.Hour) - et := time.Now() - query := NewMetricQuery("bitly-gcp-prod", "some metric", &st, &et) - query.Set_ALIGN_NONE() - query.Set_REDUCE_NONE() - - req, err := query.request() - if err != nil { - t.Error(err) - } - - aggregation := req.GetAggregation() - if aggregation.GetPerSeriesAligner() != monitoringpb.Aggregation_ALIGN_NONE { - t.Fatal("aligner not overridden") - } - if aggregation.GetCrossSeriesReducer() != monitoringpb.Aggregation_REDUCE_NONE { - t.Fatal("reducer not overridden") - } -} diff --git a/tsplot/set_aggregation_opts.go b/tsplot/set_aggregation_opts.go deleted file mode 100644 index fe298a3..0000000 --- a/tsplot/set_aggregation_opts.go +++ /dev/null @@ -1,139 +0,0 @@ -package tsplot -/* -DO NOT EDIT -Generated by codegen.go -https://github.com/googleapis/go-genproto/blob/0135a39c27378c1b903c75204eff61a060be5eb7/googleapis/monitoring/v3/common.pb.go -*/ - -func (mq *MetricQuery) Set_ALIGN_COUNT() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(13)) -} - -func (mq *MetricQuery) Set_ALIGN_COUNT_FALSE() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(24)) -} - -func (mq *MetricQuery) Set_ALIGN_COUNT_TRUE() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(16)) -} - -func (mq *MetricQuery) Set_ALIGN_DELTA() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(1)) -} - -func (mq *MetricQuery) Set_ALIGN_FRACTION_TRUE() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(17)) -} - -func (mq *MetricQuery) Set_ALIGN_INTERPOLATE() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(3)) -} - -func (mq *MetricQuery) Set_ALIGN_MAX() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(11)) -} - -func (mq *MetricQuery) Set_ALIGN_MEAN() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(12)) -} - -func (mq *MetricQuery) Set_ALIGN_MIN() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(10)) -} - -func (mq *MetricQuery) Set_ALIGN_NEXT_OLDER() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(4)) -} - -func (mq *MetricQuery) Set_ALIGN_NONE() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(0)) -} - -func (mq *MetricQuery) Set_ALIGN_PERCENTILE_05() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(21)) -} - -func (mq *MetricQuery) Set_ALIGN_PERCENTILE_50() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(20)) -} - -func (mq *MetricQuery) Set_ALIGN_PERCENTILE_95() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(19)) -} - -func (mq *MetricQuery) Set_ALIGN_PERCENTILE_99() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(18)) -} - -func (mq *MetricQuery) Set_ALIGN_PERCENT_CHANGE() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(23)) -} - -func (mq *MetricQuery) Set_ALIGN_RATE() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(2)) -} - -func (mq *MetricQuery) Set_ALIGN_STDDEV() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(15)) -} - -func (mq *MetricQuery) Set_ALIGN_SUM() { - *mq.aggregation = append(*mq.aggregation, withPerSeriesAligner(14)) -} - -func (mq *MetricQuery) Set_REDUCE_COUNT() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(6)) -} - -func (mq *MetricQuery) Set_REDUCE_COUNT_FALSE() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(15)) -} - -func (mq *MetricQuery) Set_REDUCE_COUNT_TRUE() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(7)) -} - -func (mq *MetricQuery) Set_REDUCE_FRACTION_TRUE() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(8)) -} - -func (mq *MetricQuery) Set_REDUCE_MAX() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(3)) -} - -func (mq *MetricQuery) Set_REDUCE_MEAN() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(1)) -} - -func (mq *MetricQuery) Set_REDUCE_MIN() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(2)) -} - -func (mq *MetricQuery) Set_REDUCE_NONE() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(0)) -} - -func (mq *MetricQuery) Set_REDUCE_PERCENTILE_05() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(12)) -} - -func (mq *MetricQuery) Set_REDUCE_PERCENTILE_50() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(11)) -} - -func (mq *MetricQuery) Set_REDUCE_PERCENTILE_95() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(10)) -} - -func (mq *MetricQuery) Set_REDUCE_PERCENTILE_99() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(9)) -} - -func (mq *MetricQuery) Set_REDUCE_STDDEV() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(5)) -} - -func (mq *MetricQuery) Set_REDUCE_SUM() { - *mq.aggregation = append(*mq.aggregation, withCrossSeriesReducer(4)) -} -