| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | package stackdriver | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2018-09-10 23:49:13 +08:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2018-09-17 22:30:06 +08:00
										 |  |  | 	"math" | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	"net/http" | 
					
						
							|  |  |  | 	"net/url" | 
					
						
							|  |  |  | 	"path" | 
					
						
							|  |  |  | 	"regexp" | 
					
						
							| 
									
										
										
										
											2018-09-17 22:30:06 +08:00
										 |  |  | 	"strconv" | 
					
						
							| 
									
										
										
										
											2018-09-14 22:20:51 +08:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2018-09-10 20:49:41 +08:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"golang.org/x/net/context/ctxhttp" | 
					
						
							| 
									
										
										
										
											2018-10-10 17:55:45 +08:00
										 |  |  | 	"golang.org/x/oauth2/google" | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-10 04:52:12 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/api/pluginproxy" | 
					
						
							| 
									
										
										
										
											2018-09-10 17:31:53 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/components/null" | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/components/simplejson" | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/log" | 
					
						
							|  |  |  | 	"github.com/grafana/grafana/pkg/models" | 
					
						
							| 
									
										
										
										
											2018-09-10 23:49:13 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/plugins" | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/setting" | 
					
						
							|  |  |  | 	"github.com/grafana/grafana/pkg/tsdb" | 
					
						
							|  |  |  | 	"github.com/opentracing/opentracing-go" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | var ( | 
					
						
							| 
									
										
										
										
											2018-10-10 20:17:03 +08:00
										 |  |  | 	slog             log.Logger | 
					
						
							|  |  |  | 	legendKeyFormat  *regexp.Regexp | 
					
						
							|  |  |  | 	metricNameFormat *regexp.Regexp | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							|  |  |  | 	gceAuthentication string = "gce" | 
					
						
							|  |  |  | 	jwtAuthentication string = "jwt" | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2018-09-11 21:52:37 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // StackdriverExecutor executes queries for the Stackdriver datasource
 | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | type StackdriverExecutor struct { | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 	httpClient *http.Client | 
					
						
							|  |  |  | 	dsInfo     *models.DataSource | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 21:52:37 +08:00
										 |  |  | // NewStackdriverExecutor initializes a http client
 | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | func NewStackdriverExecutor(dsInfo *models.DataSource) (tsdb.TsdbQueryEndpoint, error) { | 
					
						
							|  |  |  | 	httpClient, err := dsInfo.GetHttpClient() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return &StackdriverExecutor{ | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 		httpClient: httpClient, | 
					
						
							|  |  |  | 		dsInfo:     dsInfo, | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 	}, nil | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func init() { | 
					
						
							| 
									
										
										
										
											2018-09-11 21:52:37 +08:00
										 |  |  | 	slog = log.New("tsdb.stackdriver") | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	tsdb.RegisterTsdbQueryEndpoint("stackdriver", NewStackdriverExecutor) | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 	legendKeyFormat = regexp.MustCompile(`\{\{\s*(.+?)\s*\}\}`) | 
					
						
							| 
									
										
										
										
											2018-09-28 23:15:55 +08:00
										 |  |  | 	metricNameFormat = regexp.MustCompile(`([\w\d_]+)\.googleapis\.com/(.+)`) | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | // Query takes in the frontend queries, parses them into the Stackdriver query format
 | 
					
						
							|  |  |  | // executes the queries against the Stackdriver API and parses the response into
 | 
					
						
							|  |  |  | // the time series or table format
 | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | func (e *StackdriverExecutor) Query(ctx context.Context, dsInfo *models.DataSource, tsdbQuery *tsdb.TsdbQuery) (*tsdb.Response, error) { | 
					
						
							| 
									
										
										
										
											2018-09-24 22:02:35 +08:00
										 |  |  | 	var result *tsdb.Response | 
					
						
							|  |  |  | 	var err error | 
					
						
							|  |  |  | 	queryType := tsdbQuery.Queries[0].Model.Get("type").MustString("") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch queryType { | 
					
						
							|  |  |  | 	case "annotationQuery": | 
					
						
							|  |  |  | 		result, err = e.executeAnnotationQuery(ctx, tsdbQuery) | 
					
						
							| 
									
										
										
										
											2018-10-11 19:44:15 +08:00
										 |  |  | 	case "ensureDefaultProjectQuery": | 
					
						
							|  |  |  | 		result, err = e.ensureDefaultProject(ctx, tsdbQuery) | 
					
						
							| 
									
										
										
										
											2018-09-24 22:02:35 +08:00
										 |  |  | 	case "timeSeriesQuery": | 
					
						
							|  |  |  | 		fallthrough | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		result, err = e.executeTimeSeriesQuery(ctx, tsdbQuery) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return result, err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (e *StackdriverExecutor) executeTimeSeriesQuery(ctx context.Context, tsdbQuery *tsdb.TsdbQuery) (*tsdb.Response, error) { | 
					
						
							| 
									
										
										
										
											2018-09-10 20:49:41 +08:00
										 |  |  | 	result := &tsdb.Response{ | 
					
						
							|  |  |  | 		Results: make(map[string]*tsdb.QueryResult), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-10 20:17:03 +08:00
										 |  |  | 	authenticationType := e.dsInfo.JsonData.Get("authenticationType").MustString(jwtAuthentication) | 
					
						
							| 
									
										
										
										
											2018-10-10 19:38:21 +08:00
										 |  |  | 	if authenticationType == gceAuthentication { | 
					
						
							| 
									
										
										
										
											2018-10-10 17:55:45 +08:00
										 |  |  | 		defaultProject, err := e.getDefaultProject(ctx) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, fmt.Errorf("Failed to retrieve default project from GCE metadata server. error: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		e.dsInfo.JsonData.Set("defaultProject", defaultProject) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-12 06:24:59 +08:00
										 |  |  | 	queries, err := e.buildQueries(tsdbQuery) | 
					
						
							| 
									
										
										
										
											2018-09-10 20:49:41 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 	for _, query := range queries { | 
					
						
							| 
									
										
										
										
											2018-09-27 21:17:35 +08:00
										 |  |  | 		queryRes, resp, err := e.executeQuery(ctx, query, tsdbQuery) | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-09-27 21:17:35 +08:00
										 |  |  | 		err = e.parseResponse(queryRes, resp, query) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			queryRes.Error = err | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 		result.Results[query.RefID] = queryRes | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return result, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-12 06:24:59 +08:00
										 |  |  | func (e *StackdriverExecutor) buildQueries(tsdbQuery *tsdb.TsdbQuery) ([]*StackdriverQuery, error) { | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 	stackdriverQueries := []*StackdriverQuery{} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	startTime, err := tsdbQuery.TimeRange.ParseFrom() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	endTime, err := tsdbQuery.TimeRange.ParseTo() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-26 23:50:08 +08:00
										 |  |  | 	durationSeconds := int(endTime.Sub(startTime).Seconds()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 	for _, query := range tsdbQuery.Queries { | 
					
						
							|  |  |  | 		var target string | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		metricType := query.Model.Get("metricType").MustString() | 
					
						
							| 
									
										
										
										
											2018-09-14 22:20:51 +08:00
										 |  |  | 		filterParts := query.Model.Get("filters").MustArray() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 		params := url.Values{} | 
					
						
							|  |  |  | 		params.Add("interval.startTime", startTime.UTC().Format(time.RFC3339)) | 
					
						
							|  |  |  | 		params.Add("interval.endTime", endTime.UTC().Format(time.RFC3339)) | 
					
						
							| 
									
										
										
										
											2018-09-27 16:13:32 +08:00
										 |  |  | 		params.Add("filter", buildFilterString(metricType, filterParts)) | 
					
						
							| 
									
										
										
										
											2018-09-30 05:45:28 +08:00
										 |  |  | 		params.Add("view", query.Model.Get("view").MustString("FULL")) | 
					
						
							| 
									
										
										
										
											2018-09-26 23:50:08 +08:00
										 |  |  | 		setAggParams(¶ms, query, durationSeconds) | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 16:13:32 +08:00
										 |  |  | 		target = params.Encode() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 		if setting.Env == setting.DEV { | 
					
						
							| 
									
										
										
										
											2018-09-11 21:52:37 +08:00
										 |  |  | 			slog.Debug("Stackdriver request", "params", params) | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-20 06:52:14 +08:00
										 |  |  | 		groupBys := query.Model.Get("groupBys").MustArray() | 
					
						
							|  |  |  | 		groupBysAsStrings := make([]string, 0) | 
					
						
							|  |  |  | 		for _, groupBy := range groupBys { | 
					
						
							|  |  |  | 			groupBysAsStrings = append(groupBysAsStrings, groupBy.(string)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 		aliasBy := query.Model.Get("aliasBy").MustString() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 		stackdriverQueries = append(stackdriverQueries, &StackdriverQuery{ | 
					
						
							| 
									
										
										
										
											2018-09-20 06:52:14 +08:00
										 |  |  | 			Target:   target, | 
					
						
							|  |  |  | 			Params:   params, | 
					
						
							|  |  |  | 			RefID:    query.RefId, | 
					
						
							|  |  |  | 			GroupBys: groupBysAsStrings, | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 			AliasBy:  aliasBy, | 
					
						
							| 
									
										
										
										
											2018-09-11 20:06:46 +08:00
										 |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return stackdriverQueries, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 23:07:46 +08:00
										 |  |  | func reverse(s string) string { | 
					
						
							|  |  |  | 	chars := []rune(s) | 
					
						
							|  |  |  | 	for i, j := 0, len(chars)-1; i < j; i, j = i+1, j-1 { | 
					
						
							|  |  |  | 		chars[i], chars[j] = chars[j], chars[i] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return string(chars) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func interpolateFilterWildcards(value string) string { | 
					
						
							| 
									
										
										
										
											2018-10-20 01:47:31 +08:00
										 |  |  | 	matches := strings.Count(value, "*") | 
					
						
							| 
									
										
										
										
											2018-10-02 23:52:26 +08:00
										 |  |  | 	if matches == 2 && strings.HasSuffix(value, "*") && strings.HasPrefix(value, "*") { | 
					
						
							| 
									
										
										
										
											2018-10-02 23:29:51 +08:00
										 |  |  | 		value = strings.Replace(value, "*", "", -1) | 
					
						
							| 
									
										
										
										
											2018-10-02 23:07:46 +08:00
										 |  |  | 		value = fmt.Sprintf(`has_substring("%s")`, value) | 
					
						
							| 
									
										
										
										
											2018-10-02 23:52:26 +08:00
										 |  |  | 	} else if matches == 1 && strings.HasPrefix(value, "*") { | 
					
						
							| 
									
										
										
										
											2018-10-02 23:07:46 +08:00
										 |  |  | 		value = strings.Replace(value, "*", "", 1) | 
					
						
							|  |  |  | 		value = fmt.Sprintf(`ends_with("%s")`, value) | 
					
						
							| 
									
										
										
										
											2018-10-02 23:52:26 +08:00
										 |  |  | 	} else if matches == 1 && strings.HasSuffix(value, "*") { | 
					
						
							| 
									
										
										
										
											2018-10-02 23:07:46 +08:00
										 |  |  | 		value = reverse(strings.Replace(reverse(value), "*", "", 1)) | 
					
						
							|  |  |  | 		value = fmt.Sprintf(`starts_with("%s")`, value) | 
					
						
							| 
									
										
										
										
											2018-10-02 23:58:31 +08:00
										 |  |  | 	} else if matches != 0 { | 
					
						
							| 
									
										
										
										
											2018-10-02 23:11:05 +08:00
										 |  |  | 		re := regexp.MustCompile(`[-\/^$+?.()|[\]{}]`) | 
					
						
							|  |  |  | 		value = string(re.ReplaceAllFunc([]byte(value), func(in []byte) []byte { | 
					
						
							|  |  |  | 			return []byte(strings.Replace(string(in), string(in), `\\`+string(in), 1)) | 
					
						
							|  |  |  | 		})) | 
					
						
							|  |  |  | 		value = strings.Replace(value, "*", ".*", -1) | 
					
						
							| 
									
										
										
										
											2018-10-02 23:07:46 +08:00
										 |  |  | 		value = strings.Replace(value, `"`, `\\"`, -1) | 
					
						
							|  |  |  | 		value = fmt.Sprintf(`monitoring.regex.full_match("^%s$")`, value) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return value | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 16:13:32 +08:00
										 |  |  | func buildFilterString(metricType string, filterParts []interface{}) string { | 
					
						
							|  |  |  | 	filterString := "" | 
					
						
							|  |  |  | 	for i, part := range filterParts { | 
					
						
							|  |  |  | 		mod := i % 4 | 
					
						
							|  |  |  | 		if part == "AND" { | 
					
						
							|  |  |  | 			filterString += " " | 
					
						
							|  |  |  | 		} else if mod == 2 { | 
					
						
							| 
									
										
										
										
											2018-10-08 17:12:26 +08:00
										 |  |  | 			operator := filterParts[i-1] | 
					
						
							|  |  |  | 			if operator == "=~" || operator == "!=~" { | 
					
						
							|  |  |  | 				filterString = reverse(strings.Replace(reverse(filterString), "~", "", 1)) | 
					
						
							|  |  |  | 				filterString += fmt.Sprintf(`monitoring.regex.full_match("%s")`, part) | 
					
						
							|  |  |  | 			} else if strings.Contains(part.(string), "*") { | 
					
						
							| 
									
										
										
										
											2018-10-02 23:07:46 +08:00
										 |  |  | 				filterString += interpolateFilterWildcards(part.(string)) | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				filterString += fmt.Sprintf(`"%s"`, part) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-09-27 16:13:32 +08:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			filterString += part.(string) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return strings.Trim(fmt.Sprintf(`metric.type="%s" %s`, metricType, filterString), " ") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-26 23:50:08 +08:00
										 |  |  | func setAggParams(params *url.Values, query *tsdb.Query, durationSeconds int) { | 
					
						
							| 
									
										
										
										
											2018-09-12 06:24:59 +08:00
										 |  |  | 	primaryAggregation := query.Model.Get("primaryAggregation").MustString() | 
					
						
							| 
									
										
										
										
											2018-09-17 20:34:18 +08:00
										 |  |  | 	perSeriesAligner := query.Model.Get("perSeriesAligner").MustString() | 
					
						
							| 
									
										
										
										
											2018-09-17 22:30:06 +08:00
										 |  |  | 	alignmentPeriod := query.Model.Get("alignmentPeriod").MustString() | 
					
						
							| 
									
										
										
										
											2018-09-17 20:34:18 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-12 06:24:59 +08:00
										 |  |  | 	if primaryAggregation == "" { | 
					
						
							|  |  |  | 		primaryAggregation = "REDUCE_NONE" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-17 20:34:18 +08:00
										 |  |  | 	if perSeriesAligner == "" { | 
					
						
							|  |  |  | 		perSeriesAligner = "ALIGN_MEAN" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-26 21:44:09 +08:00
										 |  |  | 	if alignmentPeriod == "grafana-auto" || alignmentPeriod == "" { | 
					
						
							| 
									
										
										
										
											2018-09-26 23:50:08 +08:00
										 |  |  | 		alignmentPeriodValue := int(math.Max(float64(query.IntervalMs)/1000, 60.0)) | 
					
						
							| 
									
										
										
										
											2018-09-17 22:30:06 +08:00
										 |  |  | 		alignmentPeriod = "+" + strconv.Itoa(alignmentPeriodValue) + "s" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-26 23:50:08 +08:00
										 |  |  | 	if alignmentPeriod == "stackdriver-auto" { | 
					
						
							|  |  |  | 		alignmentPeriodValue := int(math.Max(float64(durationSeconds), 60.0)) | 
					
						
							| 
									
										
										
										
											2018-09-27 16:20:01 +08:00
										 |  |  | 		if alignmentPeriodValue < 60*60*23 { | 
					
						
							| 
									
										
										
										
											2018-09-26 23:50:08 +08:00
										 |  |  | 			alignmentPeriod = "+60s" | 
					
						
							| 
									
										
										
										
											2018-09-27 16:20:01 +08:00
										 |  |  | 		} else if alignmentPeriodValue < 60*60*24*6 { | 
					
						
							| 
									
										
										
										
											2018-09-26 23:50:08 +08:00
										 |  |  | 			alignmentPeriod = "+300s" | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			alignmentPeriod = "+3600s" | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-17 23:13:06 +08:00
										 |  |  | 	re := regexp.MustCompile("[0-9]+") | 
					
						
							| 
									
										
										
										
											2018-09-26 21:37:29 +08:00
										 |  |  | 	seconds, err := strconv.ParseInt(re.FindString(alignmentPeriod), 10, 64) | 
					
						
							|  |  |  | 	if err != nil || seconds > 3600 { | 
					
						
							| 
									
										
										
										
											2018-09-17 23:13:06 +08:00
										 |  |  | 		alignmentPeriod = "+3600s" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-17 20:34:18 +08:00
										 |  |  | 	params.Add("aggregation.crossSeriesReducer", primaryAggregation) | 
					
						
							|  |  |  | 	params.Add("aggregation.perSeriesAligner", perSeriesAligner) | 
					
						
							| 
									
										
										
										
											2018-09-17 22:30:06 +08:00
										 |  |  | 	params.Add("aggregation.alignmentPeriod", alignmentPeriod) | 
					
						
							| 
									
										
										
										
											2018-09-12 06:24:59 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 17:02:31 +08:00
										 |  |  | 	groupBys := query.Model.Get("groupBys").MustArray() | 
					
						
							|  |  |  | 	if len(groupBys) > 0 { | 
					
						
							|  |  |  | 		for i := 0; i < len(groupBys); i++ { | 
					
						
							|  |  |  | 			params.Add("aggregation.groupByFields", groupBys[i].(string)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-09-12 06:24:59 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 21:17:35 +08:00
										 |  |  | func (e *StackdriverExecutor) executeQuery(ctx context.Context, query *StackdriverQuery, tsdbQuery *tsdb.TsdbQuery) (*tsdb.QueryResult, StackdriverResponse, error) { | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 	queryResult := &tsdb.QueryResult{Meta: simplejson.New(), RefId: query.RefID} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 15:56:49 +08:00
										 |  |  | 	req, err := e.createRequest(ctx, e.dsInfo) | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		queryResult.Error = err | 
					
						
							| 
									
										
										
										
											2018-09-27 21:17:35 +08:00
										 |  |  | 		return queryResult, StackdriverResponse{}, nil | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	req.URL.RawQuery = query.Params.Encode() | 
					
						
							|  |  |  | 	queryResult.Meta.Set("rawQuery", req.URL.RawQuery) | 
					
						
							| 
									
										
										
										
											2018-09-26 23:50:08 +08:00
										 |  |  | 	alignmentPeriod, ok := req.URL.Query()["aggregation.alignmentPeriod"] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ok { | 
					
						
							|  |  |  | 		re := regexp.MustCompile("[0-9]+") | 
					
						
							|  |  |  | 		seconds, err := strconv.ParseInt(re.FindString(alignmentPeriod[0]), 10, 64) | 
					
						
							|  |  |  | 		if err == nil { | 
					
						
							|  |  |  | 			queryResult.Meta.Set("alignmentPeriod", seconds) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	span, ctx := opentracing.StartSpanFromContext(ctx, "stackdriver query") | 
					
						
							|  |  |  | 	span.SetTag("target", query.Target) | 
					
						
							|  |  |  | 	span.SetTag("from", tsdbQuery.TimeRange.From) | 
					
						
							|  |  |  | 	span.SetTag("until", tsdbQuery.TimeRange.To) | 
					
						
							|  |  |  | 	span.SetTag("datasource_id", e.dsInfo.Id) | 
					
						
							|  |  |  | 	span.SetTag("org_id", e.dsInfo.OrgId) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	defer span.Finish() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	opentracing.GlobalTracer().Inject( | 
					
						
							|  |  |  | 		span.Context(), | 
					
						
							|  |  |  | 		opentracing.HTTPHeaders, | 
					
						
							|  |  |  | 		opentracing.HTTPHeadersCarrier(req.Header)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	res, err := ctxhttp.Do(ctx, e.httpClient, req) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		queryResult.Error = err | 
					
						
							| 
									
										
										
										
											2018-09-27 21:17:35 +08:00
										 |  |  | 		return queryResult, StackdriverResponse{}, nil | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	data, err := e.unmarshalResponse(res) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		queryResult.Error = err | 
					
						
							| 
									
										
										
										
											2018-09-27 21:17:35 +08:00
										 |  |  | 		return queryResult, StackdriverResponse{}, nil | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 21:17:35 +08:00
										 |  |  | 	return queryResult, data, nil | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-14 23:46:42 +08:00
										 |  |  | func (e *StackdriverExecutor) unmarshalResponse(res *http.Response) (StackdriverResponse, error) { | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	body, err := ioutil.ReadAll(res.Body) | 
					
						
							|  |  |  | 	defer res.Body.Close() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2018-09-14 23:46:42 +08:00
										 |  |  | 		return StackdriverResponse{}, err | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if res.StatusCode/100 != 2 { | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 		slog.Error("Request failed", "status", res.Status, "body", string(body)) | 
					
						
							| 
									
										
										
										
											2018-09-14 23:46:42 +08:00
										 |  |  | 		return StackdriverResponse{}, fmt.Errorf(string(body)) | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-14 23:46:42 +08:00
										 |  |  | 	var data StackdriverResponse | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	err = json.Unmarshal(body, &data) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 		slog.Error("Failed to unmarshal Stackdriver response", "error", err, "status", res.Status, "body", string(body)) | 
					
						
							| 
									
										
										
										
											2018-09-14 23:46:42 +08:00
										 |  |  | 		return StackdriverResponse{}, err | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return data, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-20 06:52:14 +08:00
										 |  |  | func (e *StackdriverExecutor) parseResponse(queryRes *tsdb.QueryResult, data StackdriverResponse, query *StackdriverQuery) error { | 
					
						
							| 
									
										
										
										
											2018-09-13 17:02:31 +08:00
										 |  |  | 	metricLabels := make(map[string][]string) | 
					
						
							| 
									
										
										
										
											2018-09-13 22:01:03 +08:00
										 |  |  | 	resourceLabels := make(map[string][]string) | 
					
						
							| 
									
										
										
										
											2018-10-24 17:18:49 +08:00
										 |  |  | 	var resourceTypes []string | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, series := range data.TimeSeries { | 
					
						
							|  |  |  | 		if !containsLabel(resourceTypes, series.Resource.Type) { | 
					
						
							|  |  |  | 			resourceTypes = append(resourceTypes, series.Resource.Type) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-09-13 17:02:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 01:07:44 +08:00
										 |  |  | 	for _, series := range data.TimeSeries { | 
					
						
							|  |  |  | 		points := make([]tsdb.TimePoint, 0) | 
					
						
							| 
									
										
										
										
											2018-09-12 07:03:16 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 		defaultMetricName := series.Metric.Type | 
					
						
							| 
									
										
										
										
											2018-10-24 17:18:49 +08:00
										 |  |  | 		if len(resourceTypes) > 1 { | 
					
						
							|  |  |  | 			defaultMetricName += " " + series.Resource.Type | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-09-11 21:52:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 17:02:31 +08:00
										 |  |  | 		for key, value := range series.Metric.Labels { | 
					
						
							| 
									
										
										
										
											2018-09-13 22:01:03 +08:00
										 |  |  | 			if !containsLabel(metricLabels[key], value) { | 
					
						
							|  |  |  | 				metricLabels[key] = append(metricLabels[key], value) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-09-20 06:52:14 +08:00
										 |  |  | 			if len(query.GroupBys) == 0 || containsLabel(query.GroupBys, "metric.label."+key) { | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 				defaultMetricName += " " + value | 
					
						
							| 
									
										
										
										
											2018-09-20 06:52:14 +08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-09-11 21:52:37 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-09-13 17:02:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 22:01:03 +08:00
										 |  |  | 		for key, value := range series.Resource.Labels { | 
					
						
							|  |  |  | 			if !containsLabel(resourceLabels[key], value) { | 
					
						
							|  |  |  | 				resourceLabels[key] = append(resourceLabels[key], value) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-09-20 06:52:14 +08:00
										 |  |  | 			if containsLabel(query.GroupBys, "resource.label."+key) { | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 				defaultMetricName += " " + value | 
					
						
							| 
									
										
										
										
											2018-09-20 06:52:14 +08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-09-13 22:01:03 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-09-13 17:02:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 18:42:30 +08:00
										 |  |  | 		// reverse the order to be ascending
 | 
					
						
							|  |  |  | 		if series.ValueType != "DISTRIBUTION" { | 
					
						
							|  |  |  | 			for i := len(series.Points) - 1; i >= 0; i-- { | 
					
						
							|  |  |  | 				point := series.Points[i] | 
					
						
							|  |  |  | 				value := point.Value.DoubleValue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if series.ValueType == "INT64" { | 
					
						
							|  |  |  | 					parsedValue, err := strconv.ParseFloat(point.Value.IntValue, 64) | 
					
						
							|  |  |  | 					if err == nil { | 
					
						
							|  |  |  | 						value = parsedValue | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 18:42:30 +08:00
										 |  |  | 				if series.ValueType == "BOOL" { | 
					
						
							|  |  |  | 					if point.Value.BoolValue { | 
					
						
							|  |  |  | 						value = 1 | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						value = 0 | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				points = append(points, tsdb.NewTimePoint(null.FloatFrom(value), float64((point.Interval.EndTime).Unix())*1000)) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-24 17:18:49 +08:00
										 |  |  | 			metricName := formatLegendKeys(series.Metric.Type, defaultMetricName, series.Resource.Type, series.Metric.Labels, series.Resource.Labels, make(map[string]string), query) | 
					
						
							| 
									
										
										
										
											2018-10-02 18:42:30 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			queryRes.Series = append(queryRes.Series, &tsdb.TimeSeries{ | 
					
						
							|  |  |  | 				Name:   metricName, | 
					
						
							|  |  |  | 				Points: points, | 
					
						
							|  |  |  | 			}) | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			buckets := make(map[int]*tsdb.TimeSeries) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			for i := len(series.Points) - 1; i >= 0; i-- { | 
					
						
							|  |  |  | 				point := series.Points[i] | 
					
						
							|  |  |  | 				if len(point.Value.DistributionValue.BucketCounts) == 0 { | 
					
						
							|  |  |  | 					continue | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				maxKey := 0 | 
					
						
							|  |  |  | 				for i := 0; i < len(point.Value.DistributionValue.BucketCounts); i++ { | 
					
						
							|  |  |  | 					value, err := strconv.ParseFloat(point.Value.DistributionValue.BucketCounts[i], 64) | 
					
						
							|  |  |  | 					if err != nil { | 
					
						
							|  |  |  | 						continue | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					if _, ok := buckets[i]; !ok { | 
					
						
							|  |  |  | 						// set lower bounds
 | 
					
						
							|  |  |  | 						// https://cloud.google.com/monitoring/api/ref_v3/rest/v3/TimeSeries#Distribution
 | 
					
						
							|  |  |  | 						bucketBound := calcBucketBound(point.Value.DistributionValue.BucketOptions, i) | 
					
						
							|  |  |  | 						additionalLabels := map[string]string{"bucket": bucketBound} | 
					
						
							|  |  |  | 						buckets[i] = &tsdb.TimeSeries{ | 
					
						
							| 
									
										
										
										
											2018-10-24 17:18:49 +08:00
										 |  |  | 							Name:   formatLegendKeys(series.Metric.Type, defaultMetricName, series.Resource.Type, series.Metric.Labels, series.Resource.Labels, additionalLabels, query), | 
					
						
							| 
									
										
										
										
											2018-10-02 18:42:30 +08:00
										 |  |  | 							Points: make([]tsdb.TimePoint, 0), | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						if maxKey < i { | 
					
						
							|  |  |  | 							maxKey = i | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					buckets[i].Points = append(buckets[i].Points, tsdb.NewTimePoint(null.FloatFrom(value), float64((point.Interval.EndTime).Unix())*1000)) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// fill empty bucket
 | 
					
						
							|  |  |  | 				for i := 0; i < maxKey; i++ { | 
					
						
							|  |  |  | 					if _, ok := buckets[i]; !ok { | 
					
						
							|  |  |  | 						bucketBound := calcBucketBound(point.Value.DistributionValue.BucketOptions, i) | 
					
						
							|  |  |  | 						additionalLabels := map[string]string{"bucket": bucketBound} | 
					
						
							|  |  |  | 						buckets[i] = &tsdb.TimeSeries{ | 
					
						
							| 
									
										
										
										
											2018-10-24 17:18:49 +08:00
										 |  |  | 							Name:   formatLegendKeys(series.Metric.Type, defaultMetricName, series.Resource.Type, series.Metric.Labels, series.Resource.Labels, additionalLabels, query), | 
					
						
							| 
									
										
										
										
											2018-10-02 18:42:30 +08:00
										 |  |  | 							Points: make([]tsdb.TimePoint, 0), | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			for i := 0; i < len(buckets); i++ { | 
					
						
							|  |  |  | 				queryRes.Series = append(queryRes.Series, buckets[i]) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-09-11 01:07:44 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 22:01:03 +08:00
										 |  |  | 	queryRes.Meta.Set("resourceLabels", resourceLabels) | 
					
						
							| 
									
										
										
										
											2018-09-13 17:02:31 +08:00
										 |  |  | 	queryRes.Meta.Set("metricLabels", metricLabels) | 
					
						
							| 
									
										
										
										
											2018-09-20 06:52:14 +08:00
										 |  |  | 	queryRes.Meta.Set("groupBys", query.GroupBys) | 
					
						
							| 
									
										
										
										
											2018-10-24 17:18:49 +08:00
										 |  |  | 	queryRes.Meta.Set("resourceTypes", resourceTypes) | 
					
						
							| 
									
										
										
										
											2018-09-13 17:02:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2018-09-11 01:07:44 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 22:01:03 +08:00
										 |  |  | func containsLabel(labels []string, newLabel string) bool { | 
					
						
							|  |  |  | 	for _, val := range labels { | 
					
						
							|  |  |  | 		if val == newLabel { | 
					
						
							|  |  |  | 			return true | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return false | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-24 17:18:49 +08:00
										 |  |  | func formatLegendKeys(metricType string, defaultMetricName string, resourceType string, metricLabels map[string]string, resourceLabels map[string]string, additionalLabels map[string]string, query *StackdriverQuery) string { | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 	if query.AliasBy == "" { | 
					
						
							|  |  |  | 		return defaultMetricName | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	result := legendKeyFormat.ReplaceAllFunc([]byte(query.AliasBy), func(in []byte) []byte { | 
					
						
							|  |  |  | 		metaPartName := strings.Replace(string(in), "{{", "", 1) | 
					
						
							|  |  |  | 		metaPartName = strings.Replace(metaPartName, "}}", "", 1) | 
					
						
							|  |  |  | 		metaPartName = strings.TrimSpace(metaPartName) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if metaPartName == "metric.type" { | 
					
						
							|  |  |  | 			return []byte(metricType) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-24 17:18:49 +08:00
										 |  |  | 		if metaPartName == "resource.type" && resourceType != "" { | 
					
						
							|  |  |  | 			return []byte(resourceType) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 		metricPart := replaceWithMetricPart(metaPartName, metricType) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if metricPart != nil { | 
					
						
							|  |  |  | 			return metricPart | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		metaPartName = strings.Replace(metaPartName, "metric.label.", "", 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if val, exists := metricLabels[metaPartName]; exists { | 
					
						
							|  |  |  | 			return []byte(val) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		metaPartName = strings.Replace(metaPartName, "resource.label.", "", 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if val, exists := resourceLabels[metaPartName]; exists { | 
					
						
							|  |  |  | 			return []byte(val) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 18:42:30 +08:00
										 |  |  | 		if val, exists := additionalLabels[metaPartName]; exists { | 
					
						
							|  |  |  | 			return []byte(val) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 		return in | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return string(result) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func replaceWithMetricPart(metaPartName string, metricType string) []byte { | 
					
						
							|  |  |  | 	// https://cloud.google.com/monitoring/api/v3/metrics-details#label_names
 | 
					
						
							| 
									
										
										
										
											2018-09-28 23:15:55 +08:00
										 |  |  | 	shortMatches := metricNameFormat.FindStringSubmatch(metricType) | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if metaPartName == "metric.name" { | 
					
						
							| 
									
										
										
										
											2018-09-28 23:15:55 +08:00
										 |  |  | 		if len(shortMatches) > 0 { | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 			return []byte(shortMatches[2]) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if metaPartName == "metric.service" { | 
					
						
							| 
									
										
										
										
											2018-09-28 23:15:55 +08:00
										 |  |  | 		if len(shortMatches) > 0 { | 
					
						
							| 
									
										
										
										
											2018-09-24 04:04:24 +08:00
										 |  |  | 			return []byte(shortMatches[1]) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 18:42:30 +08:00
										 |  |  | func calcBucketBound(bucketOptions StackdriverBucketOptions, n int) string { | 
					
						
							|  |  |  | 	bucketBound := "0" | 
					
						
							|  |  |  | 	if n == 0 { | 
					
						
							|  |  |  | 		return bucketBound | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if bucketOptions.LinearBuckets != nil { | 
					
						
							|  |  |  | 		bucketBound = strconv.FormatInt(bucketOptions.LinearBuckets.Offset+(bucketOptions.LinearBuckets.Width*int64(n-1)), 10) | 
					
						
							|  |  |  | 	} else if bucketOptions.ExponentialBuckets != nil { | 
					
						
							|  |  |  | 		bucketBound = strconv.FormatInt(int64(bucketOptions.ExponentialBuckets.Scale*math.Pow(bucketOptions.ExponentialBuckets.GrowthFactor, float64(n-1))), 10) | 
					
						
							|  |  |  | 	} else if bucketOptions.ExplicitBuckets != nil { | 
					
						
							|  |  |  | 		bucketBound = strconv.FormatInt(bucketOptions.ExplicitBuckets.Bounds[(n-1)], 10) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return bucketBound | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 15:56:49 +08:00
										 |  |  | func (e *StackdriverExecutor) createRequest(ctx context.Context, dsInfo *models.DataSource) (*http.Request, error) { | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	u, _ := url.Parse(dsInfo.Url) | 
					
						
							|  |  |  | 	u.Path = path.Join(u.Path, "render") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-10 23:49:13 +08:00
										 |  |  | 	req, err := http.NewRequest(http.MethodGet, "https://monitoring.googleapis.com/", nil) | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2018-09-27 21:17:35 +08:00
										 |  |  | 		slog.Error("Failed to create request", "error", err) | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 		return nil, fmt.Errorf("Failed to create request. error: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-10 04:52:12 +08:00
										 |  |  | 	req.Header.Set("Content-Type", "application/json") | 
					
						
							| 
									
										
										
										
											2018-09-21 17:35:55 +08:00
										 |  |  | 	req.Header.Set("User-Agent", fmt.Sprintf("Grafana/%s", setting.BuildVersion)) | 
					
						
							| 
									
										
										
										
											2018-09-10 04:52:12 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-10 23:49:13 +08:00
										 |  |  | 	// find plugin
 | 
					
						
							|  |  |  | 	plugin, ok := plugins.DataSources[dsInfo.Type] | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							|  |  |  | 		return nil, errors.New("Unable to find datasource plugin Stackdriver") | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-09-10 23:49:13 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var stackdriverRoute *plugins.AppPluginRoute | 
					
						
							|  |  |  | 	for _, route := range plugin.Routes { | 
					
						
							|  |  |  | 		if route.Path == "stackdriver" { | 
					
						
							|  |  |  | 			stackdriverRoute = route | 
					
						
							|  |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-09 21:20:05 +08:00
										 |  |  | 	projectName := dsInfo.JsonData.Get("defaultProject").MustString() | 
					
						
							| 
									
										
										
										
											2018-10-16 15:56:49 +08:00
										 |  |  | 	proxyPass := fmt.Sprintf("stackdriver%s", "v3/projects/"+projectName+"/timeSeries") | 
					
						
							| 
									
										
										
										
											2018-10-08 19:23:46 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-10 23:49:13 +08:00
										 |  |  | 	pluginproxy.ApplyRoute(ctx, req, proxyPass, stackdriverRoute, dsInfo) | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-12 04:41:24 +08:00
										 |  |  | 	return req, nil | 
					
						
							| 
									
										
										
										
											2018-09-04 19:21:58 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-10-10 17:55:45 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func (e *StackdriverExecutor) getDefaultProject(ctx context.Context) (string, error) { | 
					
						
							| 
									
										
										
										
											2018-10-10 20:17:03 +08:00
										 |  |  | 	authenticationType := e.dsInfo.JsonData.Get("authenticationType").MustString(jwtAuthentication) | 
					
						
							| 
									
										
										
										
											2018-10-10 19:38:21 +08:00
										 |  |  | 	if authenticationType == gceAuthentication { | 
					
						
							| 
									
										
										
										
											2018-10-10 17:55:45 +08:00
										 |  |  | 		defaultCredentials, err := google.FindDefaultCredentials(ctx, "https://www.googleapis.com/auth/monitoring.read") | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2018-10-10 19:38:21 +08:00
										 |  |  | 			return "", fmt.Errorf("Failed to retrieve default project from GCE metadata server. error: %v", err) | 
					
						
							| 
									
										
										
										
											2018-10-10 17:55:45 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-10-16 15:56:49 +08:00
										 |  |  | 		return defaultCredentials.ProjectID, nil | 
					
						
							| 
									
										
										
										
											2018-10-10 17:55:45 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-10-16 15:56:49 +08:00
										 |  |  | 	return e.dsInfo.JsonData.Get("defaultProject").MustString(), nil | 
					
						
							| 
									
										
										
										
											2018-10-10 17:55:45 +08:00
										 |  |  | } |