| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126 |
- package elasticsearch
- import (
- "context"
- "encoding/json"
- "errors"
- "fmt"
- "github.com/grafana/grafana/pkg/log"
- "github.com/grafana/grafana/pkg/models"
- "github.com/grafana/grafana/pkg/setting"
- "github.com/grafana/grafana/pkg/tsdb"
- "golang.org/x/net/context/ctxhttp"
- "net/http"
- "net/url"
- "path"
- "strings"
- "time"
- )
- type ElasticsearchExecutor struct {
- Transport *http.Transport
- }
- var (
- glog log.Logger
- intervalCalculator tsdb.IntervalCalculator
- )
- func NewElasticsearchExecutor(dsInfo *models.DataSource) (tsdb.TsdbQueryEndpoint, error) {
- transport, err := dsInfo.GetHttpTransport()
- if err != nil {
- return nil, err
- }
- return &ElasticsearchExecutor{
- Transport: transport,
- }, nil
- }
- func init() {
- glog = log.New("tsdb.elasticsearch")
- tsdb.RegisterTsdbQueryEndpoint("elasticsearch", NewElasticsearchExecutor)
- intervalCalculator = tsdb.NewIntervalCalculator(&tsdb.IntervalOptions{MinInterval: time.Millisecond * 1})
- }
- func (e *ElasticsearchExecutor) Query(ctx context.Context, dsInfo *models.DataSource, tsdbQuery *tsdb.TsdbQuery) (*tsdb.Response, error) {
- result := &tsdb.Response{}
- result.Results = make(map[string]*tsdb.QueryResult)
- queryParser := ElasticSearchQueryParser{
- dsInfo,
- tsdbQuery.TimeRange,
- tsdbQuery.Queries,
- }
- payload, targets, err := queryParser.Parse()
- if err != nil {
- return nil, err
- }
- if setting.Env == setting.DEV {
- glog.Debug("Elasticsearch playload", "raw playload", payload)
- }
- glog.Info("Elasticsearch playload", "raw playload", payload)
- req, err := e.createRequest(dsInfo, payload)
- if err != nil {
- return nil, err
- }
- httpClient, err := dsInfo.GetHttpClient()
- if err != nil {
- return nil, err
- }
- resp, err := ctxhttp.Do(ctx, httpClient, req)
- if err != nil {
- return nil, err
- }
- if resp.StatusCode/100 != 2 {
- return nil, fmt.Errorf("elasticsearch returned statuscode invalid status code: %v", resp.Status)
- }
- var responses Responses
- dec := json.NewDecoder(resp.Body)
- defer resp.Body.Close()
- dec.UseNumber()
- err = dec.Decode(&responses)
- if err != nil {
- return nil, err
- }
- for _, res := range responses.Responses {
- if res.Err != nil {
- return nil, errors.New(res.getErrMsg())
- }
- }
- responseParser := ElasticsearchResponseParser{responses.Responses, targets}
- queryRes := responseParser.getTimeSeries()
- result.Results["A"] = queryRes
- return result, nil
- }
- func (e *ElasticsearchExecutor) createRequest(dsInfo *models.DataSource, query string) (*http.Request, error) {
- u, _ := url.Parse(dsInfo.Url)
- u.Path = path.Join(u.Path, "_msearch")
- req, err := http.NewRequest(http.MethodPost, u.String(), strings.NewReader(query))
- if err != nil {
- return nil, err
- }
- req.Header.Set("User-Agent", "Grafana")
- req.Header.Set("Content-Type", "application/json")
- if dsInfo.BasicAuth {
- req.SetBasicAuth(dsInfo.BasicAuthUser, dsInfo.BasicAuthPassword)
- }
- if !dsInfo.BasicAuth && dsInfo.User != "" {
- req.SetBasicAuth(dsInfo.User, dsInfo.Password)
- }
- glog.Debug("Elasticsearch request", "url", req.URL.String())
- glog.Debug("Elasticsearch request", "body", query)
- return req, nil
- }
|