* Unit tests for backend * minor change * Implemented querNumericItems in backend * Added query type * Set alerting to true * Return TimeSeries from History * Updated alerting feature * Fix params marshal error * Fix params more * Update zabbixAPIConnector.js * Numbers, I guess * Params Output Type * Output marshaling * Unmarshal and decoder error catch * HistoryPoint and Unmarshal fixes * Unmarshal to History * Revert "Update zabbixAPIConnector.js" This reverts commit e0ffdff859b6f920893a47a709493f8076e38ef4. * Fix unmarshaling for real * Time range integer * Use more zabbix.Items * Update response_models.go * Update zabbix_api.go * Update models.go * Update zabbix_api.go * Tests * Adding more unit tests and cleaning up additional logging * Make history request param a pointer * Actually store datasource in cache * Debug logs and timings * Handle panics gracefully * Updated Regex filter parsing * Removed must compile * Clean up regex filter
136 lines
3.7 KiB
Go
136 lines
3.7 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"runtime/debug"
|
|
|
|
simplejson "github.com/bitly/go-simplejson"
|
|
"github.com/grafana/grafana_plugin_model/go/datasource"
|
|
hclog "github.com/hashicorp/go-hclog"
|
|
plugin "github.com/hashicorp/go-plugin"
|
|
"golang.org/x/net/context"
|
|
)
|
|
|
|
// ZabbixBackend implements the Grafana backend interface and forwards queries to the ZabbixDatasource
|
|
type ZabbixBackend struct {
|
|
plugin.NetRPCUnsupportedPlugin
|
|
logger hclog.Logger
|
|
datasourceCache *Cache
|
|
}
|
|
|
|
func (b *ZabbixBackend) newZabbixDatasource() *ZabbixDatasource {
|
|
ds := NewZabbixDatasource()
|
|
ds.logger = b.logger
|
|
return ds
|
|
}
|
|
|
|
// Query receives requests from the Grafana backend. Requests are filtered by query type and sent to the
|
|
// applicable ZabbixDatasource.
|
|
func (b *ZabbixBackend) Query(ctx context.Context, tsdbReq *datasource.DatasourceRequest) (resp *datasource.DatasourceResponse, _e error) {
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
pErr, _ := r.(error)
|
|
b.logger.Error("Fatal error in Zabbix Plugin Backend", "Error", pErr)
|
|
b.logger.Error(string(debug.Stack()))
|
|
resp = BuildErrorResponse(fmt.Errorf("Unrecoverable error in grafana-zabbix plugin backend"))
|
|
}
|
|
}()
|
|
|
|
zabbixDs := b.getCachedDatasource(tsdbReq)
|
|
|
|
queryType, err := GetQueryType(tsdbReq)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
switch queryType {
|
|
case "zabbixAPI":
|
|
resp, err = zabbixDs.ZabbixAPIQuery(ctx, tsdbReq)
|
|
case "query":
|
|
resp, err = zabbixDs.queryNumericItems(ctx, tsdbReq)
|
|
case "connectionTest":
|
|
resp, err = zabbixDs.TestConnection(ctx, tsdbReq)
|
|
default:
|
|
err = errors.New("Query not implemented")
|
|
return BuildErrorResponse(err), nil
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func (b *ZabbixBackend) getCachedDatasource(tsdbReq *datasource.DatasourceRequest) *ZabbixDatasource {
|
|
dsInfoHash := HashDatasourceInfo(tsdbReq.GetDatasource())
|
|
|
|
if cachedData, ok := b.datasourceCache.Get(dsInfoHash); ok {
|
|
if cachedDS, ok := cachedData.(*ZabbixDatasource); ok {
|
|
return cachedDS
|
|
}
|
|
}
|
|
|
|
if b.logger.IsDebug() {
|
|
dsInfo := tsdbReq.GetDatasource()
|
|
b.logger.Debug(fmt.Sprintf("Datasource cache miss (Org %d Id %d '%s' %s)", dsInfo.GetOrgId(), dsInfo.GetId(), dsInfo.GetName(), dsInfoHash))
|
|
}
|
|
|
|
ds := b.newZabbixDatasource()
|
|
b.datasourceCache.Set(dsInfoHash, ds)
|
|
return ds
|
|
}
|
|
|
|
// GetQueryType determines the query type from a query or list of queries
|
|
func GetQueryType(tsdbReq *datasource.DatasourceRequest) (string, error) {
|
|
queryType := "query"
|
|
if len(tsdbReq.Queries) > 0 {
|
|
firstQuery := tsdbReq.Queries[0]
|
|
queryJSON, err := simplejson.NewJson([]byte(firstQuery.ModelJson))
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
queryType = queryJSON.Get("queryType").MustString("query")
|
|
}
|
|
return queryType, nil
|
|
}
|
|
|
|
// BuildResponse transforms a Zabbix API response to a DatasourceResponse
|
|
func BuildResponse(responseData interface{}) (*datasource.DatasourceResponse, error) {
|
|
jsonBytes, err := json.Marshal(responseData)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &datasource.DatasourceResponse{
|
|
Results: []*datasource.QueryResult{
|
|
&datasource.QueryResult{
|
|
RefId: "zabbixAPI",
|
|
MetaJson: string(jsonBytes),
|
|
},
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
// BuildErrorResponse creates a QueryResult that forwards an error to the front-end
|
|
func BuildErrorResponse(err error) *datasource.DatasourceResponse {
|
|
return &datasource.DatasourceResponse{
|
|
Results: []*datasource.QueryResult{
|
|
&datasource.QueryResult{
|
|
RefId: "zabbixAPI",
|
|
Error: err.Error(),
|
|
},
|
|
},
|
|
}
|
|
}
|
|
|
|
// BuildMetricsResponse builds a response object using a given TimeSeries array
|
|
func BuildMetricsResponse(metrics []*datasource.TimeSeries) (*datasource.DatasourceResponse, error) {
|
|
return &datasource.DatasourceResponse{
|
|
Results: []*datasource.QueryResult{
|
|
&datasource.QueryResult{
|
|
RefId: "zabbixMetrics",
|
|
Series: metrics,
|
|
},
|
|
},
|
|
}, nil
|
|
}
|