blob: c0d2d8d998cbaa91fc73ad32d48b8cb0ff18b2c5 [file] [log] [blame]
// Licensed to Apache Software Foundation (ASF) under one or more contributor
// license agreements. See the NOTICE file distributed with
// this work for additional information regarding copyright
// ownership. Apache Software Foundation (ASF) licenses this file to you under
// the Apache License, Version 2.0 (the "License"); you may
// not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing,
// software distributed under the License is distributed on an
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied. See the License for the
// specific language governing permissions and limitations
// under the License.
package dashboard
import (
"encoding/json"
"io/ioutil"
"os"
"strings"
"github.com/machinebox/graphql"
"github.com/urfave/cli"
"github.com/apache/skywalking-cli/assets"
"github.com/apache/skywalking-cli/graphql/client"
"github.com/apache/skywalking-cli/graphql/schema"
)
type MetricTemplate struct {
Condition schema.TopNCondition `json:"condition"`
Title string `json:"title"`
Aggregation string `json:"aggregation"`
AggregationNum string `json:"aggregationNum"`
}
type ChartTemplate struct {
Condition schema.MetricsCondition `json:"condition"`
Title string `json:"title"`
Unit string `json:"unit"`
Labels string `json:"labels"`
}
type GlobalTemplate struct {
Metrics []MetricTemplate `json:"metrics"`
ResponseLatency ChartTemplate `json:"responseLatency"`
HeatMap ChartTemplate `json:"heatMap"`
}
type GlobalData struct {
Metrics [][]*schema.SelectedRecord `json:"metrics"`
ResponseLatency []*schema.MetricsValues `json:"responseLatency"`
HeatMap schema.HeatMap `json:"heatMap"`
}
// Use singleton pattern to make sure to load template only once.
var globalTemplate *GlobalTemplate
const DefaultTemplatePath = "templates/Dashboard.Global.json"
// LoadTemplate reads UI template from file.
func LoadTemplate(filename string) (*GlobalTemplate, error) {
if globalTemplate != nil {
return globalTemplate, nil
}
var t GlobalTemplate
var byteValue []byte
if filename == DefaultTemplatePath {
jsonFile := assets.Read(filename)
byteValue = []byte(jsonFile)
} else {
jsonFile, err := os.Open(filename)
if err != nil {
return nil, err
}
defer jsonFile.Close()
byteValue, err = ioutil.ReadAll(jsonFile)
if err != nil {
return nil, err
}
}
if err := json.Unmarshal(byteValue, &t); err != nil {
return nil, err
}
globalTemplate = &t
return globalTemplate, nil
}
func Metrics(ctx *cli.Context, duration schema.Duration) [][]*schema.SelectedRecord {
var ret [][]*schema.SelectedRecord
template, err := LoadTemplate(ctx.String("template"))
if err != nil {
return nil
}
for _, m := range template.Metrics {
var response map[string][]*schema.SelectedRecord
request := graphql.NewRequest(assets.Read("graphqls/dashboard/SortMetrics.graphql"))
request.Var("condition", m.Condition)
request.Var("duration", duration)
client.ExecuteQueryOrFail(ctx, request, &response)
ret = append(ret, response["result"])
}
return ret
}
func responseLatency(ctx *cli.Context, duration schema.Duration) []*schema.MetricsValues {
var response map[string][]*schema.MetricsValues
template, err := LoadTemplate(ctx.String("template"))
if err != nil {
return nil
}
// labels in the template file is string type,
// need to convert to string array for graphql query.
labels := strings.Split(template.ResponseLatency.Labels, ",")
request := graphql.NewRequest(assets.Read("graphqls/dashboard/LabeledMetricsValues.graphql"))
request.Var("duration", duration)
request.Var("condition", template.ResponseLatency.Condition)
request.Var("labels", labels)
client.ExecuteQueryOrFail(ctx, request, &response)
return response["result"]
}
func heatMap(ctx *cli.Context, duration schema.Duration) schema.HeatMap {
var response map[string]schema.HeatMap
template, err := LoadTemplate(ctx.String("template"))
if err != nil {
return schema.HeatMap{}
}
request := graphql.NewRequest(assets.Read("graphqls/dashboard/HeatMap.graphql"))
request.Var("duration", duration)
request.Var("condition", template.HeatMap.Condition)
client.ExecuteQueryOrFail(ctx, request, &response)
return response["result"]
}
func Global(ctx *cli.Context, duration schema.Duration) *GlobalData {
var globalData GlobalData
globalData.Metrics = Metrics(ctx, duration)
globalData.ResponseLatency = responseLatency(ctx, duration)
globalData.HeatMap = heatMap(ctx, duration)
return &globalData
}