mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
DataSourceProxy: Handle URL parsing error (#23731)
* pluginproxy: Handle URL parsing error * pkg/api: Validate data source URLs * pkg/api: Return 400 for URL validation error
This commit is contained in:
parent
c3e3067cf4
commit
7d88018531
@ -1,6 +1,9 @@
|
|||||||
package api
|
package api
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
"github.com/grafana/grafana/pkg/api/pluginproxy"
|
"github.com/grafana/grafana/pkg/api/pluginproxy"
|
||||||
"github.com/grafana/grafana/pkg/infra/metrics"
|
"github.com/grafana/grafana/pkg/infra/metrics"
|
||||||
"github.com/grafana/grafana/pkg/models"
|
"github.com/grafana/grafana/pkg/models"
|
||||||
@ -32,7 +35,15 @@ func (hs *HTTPServer) ProxyDataSourceRequest(c *models.ReqContext) {
|
|||||||
// macaron does not include trailing slashes when resolving a wildcard path
|
// macaron does not include trailing slashes when resolving a wildcard path
|
||||||
proxyPath := ensureProxyPathTrailingSlash(c.Req.URL.Path, c.Params("*"))
|
proxyPath := ensureProxyPathTrailingSlash(c.Req.URL.Path, c.Params("*"))
|
||||||
|
|
||||||
proxy := pluginproxy.NewDataSourceProxy(ds, plugin, c, proxyPath, hs.Cfg)
|
proxy, err := pluginproxy.NewDataSourceProxy(ds, plugin, c, proxyPath, hs.Cfg)
|
||||||
|
if err != nil {
|
||||||
|
if errors.Is(err, pluginproxy.URLValidationError{}) {
|
||||||
|
c.JsonApiErr(400, fmt.Sprintf("Invalid data source URL: %q", ds.Url), err)
|
||||||
|
} else {
|
||||||
|
c.JsonApiErr(500, "Failed creating data source proxy", err)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
proxy.HandleRequest()
|
proxy.HandleRequest()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2,6 +2,8 @@ package api
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"net/url"
|
||||||
"sort"
|
"sort"
|
||||||
|
|
||||||
"github.com/grafana/grafana/pkg/api/dtos"
|
"github.com/grafana/grafana/pkg/api/dtos"
|
||||||
@ -10,6 +12,7 @@ import (
|
|||||||
"github.com/grafana/grafana/pkg/plugins"
|
"github.com/grafana/grafana/pkg/plugins"
|
||||||
"github.com/grafana/grafana/pkg/plugins/backendplugin"
|
"github.com/grafana/grafana/pkg/plugins/backendplugin"
|
||||||
"github.com/grafana/grafana/pkg/util"
|
"github.com/grafana/grafana/pkg/util"
|
||||||
|
"github.com/grafana/grafana/pkg/util/errutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
func GetDataSources(c *models.ReqContext) Response {
|
func GetDataSources(c *models.ReqContext) Response {
|
||||||
@ -124,8 +127,23 @@ func DeleteDataSourceByName(c *models.ReqContext) Response {
|
|||||||
return Success("Data source deleted")
|
return Success("Data source deleted")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func validateURL(u string) Response {
|
||||||
|
if u != "" {
|
||||||
|
_, err := url.Parse(u)
|
||||||
|
if err != nil {
|
||||||
|
return Error(400, fmt.Sprintf("Validation error, invalid URL: %q", u), errutil.Wrapf(err,
|
||||||
|
"invalid data source URL %q", u))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func AddDataSource(c *models.ReqContext, cmd models.AddDataSourceCommand) Response {
|
func AddDataSource(c *models.ReqContext, cmd models.AddDataSourceCommand) Response {
|
||||||
cmd.OrgId = c.OrgId
|
cmd.OrgId = c.OrgId
|
||||||
|
if resp := validateURL(cmd.Url); resp != nil {
|
||||||
|
return resp
|
||||||
|
}
|
||||||
|
|
||||||
if err := bus.Dispatch(&cmd); err != nil {
|
if err := bus.Dispatch(&cmd); err != nil {
|
||||||
if err == models.ErrDataSourceNameExists || err == models.ErrDataSourceUidExists {
|
if err == models.ErrDataSourceNameExists || err == models.ErrDataSourceUidExists {
|
||||||
@ -147,6 +165,9 @@ func AddDataSource(c *models.ReqContext, cmd models.AddDataSourceCommand) Respon
|
|||||||
func UpdateDataSource(c *models.ReqContext, cmd models.UpdateDataSourceCommand) Response {
|
func UpdateDataSource(c *models.ReqContext, cmd models.UpdateDataSourceCommand) Response {
|
||||||
cmd.OrgId = c.OrgId
|
cmd.OrgId = c.OrgId
|
||||||
cmd.Id = c.ParamsInt64(":id")
|
cmd.Id = c.ParamsInt64(":id")
|
||||||
|
if resp := validateURL(cmd.Url); resp != nil {
|
||||||
|
return resp
|
||||||
|
}
|
||||||
|
|
||||||
err := fillWithSecureJSONData(&cmd)
|
err := fillWithSecureJSONData(&cmd)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -31,6 +31,20 @@ var (
|
|||||||
client = newHTTPClient()
|
client = newHTTPClient()
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type URLValidationError struct {
|
||||||
|
error
|
||||||
|
|
||||||
|
url string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e URLValidationError) Error() string {
|
||||||
|
return fmt.Sprintf("Validation of URL %q failed: %s", e.url, e.error.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e URLValidationError) Unwrap() error {
|
||||||
|
return e.error
|
||||||
|
}
|
||||||
|
|
||||||
type DataSourceProxy struct {
|
type DataSourceProxy struct {
|
||||||
ds *models.DataSource
|
ds *models.DataSource
|
||||||
ctx *models.ReqContext
|
ctx *models.ReqContext
|
||||||
@ -70,8 +84,12 @@ func (lw *logWrapper) Write(p []byte) (n int, err error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewDataSourceProxy creates a new Datasource proxy
|
// NewDataSourceProxy creates a new Datasource proxy
|
||||||
func NewDataSourceProxy(ds *models.DataSource, plugin *plugins.DataSourcePlugin, ctx *models.ReqContext, proxyPath string, cfg *setting.Cfg) *DataSourceProxy {
|
func NewDataSourceProxy(ds *models.DataSource, plugin *plugins.DataSourcePlugin, ctx *models.ReqContext,
|
||||||
targetURL, _ := url.Parse(ds.Url)
|
proxyPath string, cfg *setting.Cfg) (*DataSourceProxy, error) {
|
||||||
|
targetURL, err := url.Parse(ds.Url)
|
||||||
|
if err != nil {
|
||||||
|
return nil, URLValidationError{error: err, url: ds.Url}
|
||||||
|
}
|
||||||
|
|
||||||
return &DataSourceProxy{
|
return &DataSourceProxy{
|
||||||
ds: ds,
|
ds: ds,
|
||||||
@ -80,7 +98,7 @@ func NewDataSourceProxy(ds *models.DataSource, plugin *plugins.DataSourcePlugin,
|
|||||||
proxyPath: proxyPath,
|
proxyPath: proxyPath,
|
||||||
targetUrl: targetURL,
|
targetUrl: targetURL,
|
||||||
cfg: cfg,
|
cfg: cfg,
|
||||||
}
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func newHTTPClient() httpClient {
|
func newHTTPClient() httpClient {
|
||||||
|
@ -7,11 +7,14 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
"net/http/httptest"
|
"net/http/httptest"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/grafana/grafana/pkg/components/securejsondata"
|
"github.com/grafana/grafana/pkg/components/securejsondata"
|
||||||
"github.com/grafana/grafana/pkg/models"
|
"github.com/grafana/grafana/pkg/models"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
|
||||||
"golang.org/x/oauth2"
|
"golang.org/x/oauth2"
|
||||||
macaron "gopkg.in/macaron.v1"
|
macaron "gopkg.in/macaron.v1"
|
||||||
@ -86,7 +89,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Convey("When matching route path", func() {
|
Convey("When matching route path", func() {
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "api/v4/some/method", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "api/v4/some/method", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
proxy.route = plugin.Routes[0]
|
proxy.route = plugin.Routes[0]
|
||||||
ApplyRoute(proxy.ctx.Req.Context(), req, proxy.proxyPath, proxy.route, proxy.ds)
|
ApplyRoute(proxy.ctx.Req.Context(), req, proxy.proxyPath, proxy.route, proxy.ds)
|
||||||
|
|
||||||
@ -97,7 +101,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
})
|
})
|
||||||
|
|
||||||
Convey("When matching route path and has dynamic url", func() {
|
Convey("When matching route path and has dynamic url", func() {
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "api/common/some/method", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "api/common/some/method", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
proxy.route = plugin.Routes[3]
|
proxy.route = plugin.Routes[3]
|
||||||
ApplyRoute(proxy.ctx.Req.Context(), req, proxy.proxyPath, proxy.route, proxy.ds)
|
ApplyRoute(proxy.ctx.Req.Context(), req, proxy.proxyPath, proxy.route, proxy.ds)
|
||||||
|
|
||||||
@ -109,21 +114,24 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
|
|
||||||
Convey("Validating request", func() {
|
Convey("Validating request", func() {
|
||||||
Convey("plugin route with valid role", func() {
|
Convey("plugin route with valid role", func() {
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "api/v4/some/method", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "api/v4/some/method", &setting.Cfg{})
|
||||||
err := proxy.validateRequest()
|
So(err, ShouldBeNil)
|
||||||
|
err = proxy.validateRequest()
|
||||||
So(err, ShouldBeNil)
|
So(err, ShouldBeNil)
|
||||||
})
|
})
|
||||||
|
|
||||||
Convey("plugin route with admin role and user is editor", func() {
|
Convey("plugin route with admin role and user is editor", func() {
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "api/admin", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "api/admin", &setting.Cfg{})
|
||||||
err := proxy.validateRequest()
|
So(err, ShouldBeNil)
|
||||||
|
err = proxy.validateRequest()
|
||||||
So(err, ShouldNotBeNil)
|
So(err, ShouldNotBeNil)
|
||||||
})
|
})
|
||||||
|
|
||||||
Convey("plugin route with admin role and user is admin", func() {
|
Convey("plugin route with admin role and user is admin", func() {
|
||||||
ctx.SignedInUser.OrgRole = models.ROLE_ADMIN
|
ctx.SignedInUser.OrgRole = models.ROLE_ADMIN
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "api/admin", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "api/admin", &setting.Cfg{})
|
||||||
err := proxy.validateRequest()
|
So(err, ShouldBeNil)
|
||||||
|
err = proxy.validateRequest()
|
||||||
So(err, ShouldBeNil)
|
So(err, ShouldBeNil)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
@ -191,7 +199,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
So(err, ShouldBeNil)
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
client = newFakeHTTPClient(json)
|
client = newFakeHTTPClient(json)
|
||||||
proxy1 := NewDataSourceProxy(ds, plugin, ctx, "pathwithtoken1", &setting.Cfg{})
|
proxy1, err := NewDataSourceProxy(ds, plugin, ctx, "pathwithtoken1", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
proxy1.route = plugin.Routes[0]
|
proxy1.route = plugin.Routes[0]
|
||||||
ApplyRoute(proxy1.ctx.Req.Context(), req, proxy1.proxyPath, proxy1.route, proxy1.ds)
|
ApplyRoute(proxy1.ctx.Req.Context(), req, proxy1.proxyPath, proxy1.route, proxy1.ds)
|
||||||
|
|
||||||
@ -205,7 +214,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
|
|
||||||
req, _ := http.NewRequest("GET", "http://localhost/asd", nil)
|
req, _ := http.NewRequest("GET", "http://localhost/asd", nil)
|
||||||
client = newFakeHTTPClient(json2)
|
client = newFakeHTTPClient(json2)
|
||||||
proxy2 := NewDataSourceProxy(ds, plugin, ctx, "pathwithtoken2", &setting.Cfg{})
|
proxy2, err := NewDataSourceProxy(ds, plugin, ctx, "pathwithtoken2", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
proxy2.route = plugin.Routes[1]
|
proxy2.route = plugin.Routes[1]
|
||||||
ApplyRoute(proxy2.ctx.Req.Context(), req, proxy2.proxyPath, proxy2.route, proxy2.ds)
|
ApplyRoute(proxy2.ctx.Req.Context(), req, proxy2.proxyPath, proxy2.route, proxy2.ds)
|
||||||
|
|
||||||
@ -220,7 +230,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
req, _ := http.NewRequest("GET", "http://localhost/asd", nil)
|
req, _ := http.NewRequest("GET", "http://localhost/asd", nil)
|
||||||
|
|
||||||
client = newFakeHTTPClient([]byte{})
|
client = newFakeHTTPClient([]byte{})
|
||||||
proxy3 := NewDataSourceProxy(ds, plugin, ctx, "pathwithtoken1", &setting.Cfg{})
|
proxy3, err := NewDataSourceProxy(ds, plugin, ctx, "pathwithtoken1", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
proxy3.route = plugin.Routes[0]
|
proxy3.route = plugin.Routes[0]
|
||||||
ApplyRoute(proxy3.ctx.Req.Context(), req, proxy3.proxyPath, proxy3.route, proxy3.ds)
|
ApplyRoute(proxy3.ctx.Req.Context(), req, proxy3.proxyPath, proxy3.route, proxy3.ds)
|
||||||
|
|
||||||
@ -241,7 +252,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
ds := &models.DataSource{Url: "htttp://graphite:8080", Type: models.DS_GRAPHITE}
|
ds := &models.DataSource{Url: "htttp://graphite:8080", Type: models.DS_GRAPHITE}
|
||||||
ctx := &models.ReqContext{}
|
ctx := &models.ReqContext{}
|
||||||
|
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "/render", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "/render", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
req, err := http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
req, err := http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
||||||
So(err, ShouldBeNil)
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
@ -266,7 +278,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ctx := &models.ReqContext{}
|
ctx := &models.ReqContext{}
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
req, err := http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
req, err := http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
||||||
So(err, ShouldBeNil)
|
So(err, ShouldBeNil)
|
||||||
@ -290,7 +303,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ctx := &models.ReqContext{}
|
ctx := &models.ReqContext{}
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
requestURL, _ := url.Parse("http://grafana.com/sub")
|
requestURL, _ := url.Parse("http://grafana.com/sub")
|
||||||
req := http.Request{URL: requestURL, Header: make(http.Header)}
|
req := http.Request{URL: requestURL, Header: make(http.Header)}
|
||||||
@ -316,7 +330,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ctx := &models.ReqContext{}
|
ctx := &models.ReqContext{}
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
requestURL, _ := url.Parse("http://grafana.com/sub")
|
requestURL, _ := url.Parse("http://grafana.com/sub")
|
||||||
req := http.Request{URL: requestURL, Header: make(http.Header)}
|
req := http.Request{URL: requestURL, Header: make(http.Header)}
|
||||||
@ -337,7 +352,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
Url: "http://host/root/",
|
Url: "http://host/root/",
|
||||||
}
|
}
|
||||||
ctx := &models.ReqContext{}
|
ctx := &models.ReqContext{}
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "/path/to/folder/", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "/path/to/folder/", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
req, err := http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
req, err := http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
||||||
req.Header.Add("Origin", "grafana.com")
|
req.Header.Add("Origin", "grafana.com")
|
||||||
req.Header.Add("Referer", "grafana.com")
|
req.Header.Add("Referer", "grafana.com")
|
||||||
@ -393,9 +409,9 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
Req: macaron.Request{Request: req},
|
Req: macaron.Request{Request: req},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "/path/to/folder/", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "/path/to/folder/", &setting.Cfg{})
|
||||||
req, err := http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
So(err, ShouldBeNil)
|
||||||
|
req, err = http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
||||||
So(err, ShouldBeNil)
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
proxy.getDirector()(req)
|
proxy.getDirector()(req)
|
||||||
@ -505,7 +521,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
Convey("When response header Set-Cookie is not set should remove proxied Set-Cookie header", func() {
|
Convey("When response header Set-Cookie is not set should remove proxied Set-Cookie header", func() {
|
||||||
writeErr = nil
|
writeErr = nil
|
||||||
ctx := setupCtx(nil)
|
ctx := setupCtx(nil)
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "/render", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "/render", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
proxy.HandleRequest()
|
proxy.HandleRequest()
|
||||||
|
|
||||||
@ -518,7 +535,8 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
ctx := setupCtx(func(w http.ResponseWriter) {
|
ctx := setupCtx(func(w http.ResponseWriter) {
|
||||||
w.Header().Set("Set-Cookie", "important_cookie=important_value")
|
w.Header().Set("Set-Cookie", "important_cookie=important_value")
|
||||||
})
|
})
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "/render", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "/render", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
proxy.HandleRequest()
|
proxy.HandleRequest()
|
||||||
|
|
||||||
@ -530,6 +548,24 @@ func TestDSRouteRule(t *testing.T) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestNewDataSourceProxy_InvalidURL(t *testing.T) {
|
||||||
|
ctx := models.ReqContext{
|
||||||
|
Context: &macaron.Context{
|
||||||
|
Req: macaron.Request{},
|
||||||
|
},
|
||||||
|
SignedInUser: &models.SignedInUser{OrgRole: models.ROLE_EDITOR},
|
||||||
|
}
|
||||||
|
ds := models.DataSource{
|
||||||
|
Type: "test",
|
||||||
|
Url: "://host/root",
|
||||||
|
}
|
||||||
|
cfg := setting.Cfg{}
|
||||||
|
plugin := plugins.DataSourcePlugin{}
|
||||||
|
_, err := NewDataSourceProxy(&ds, &plugin, &ctx, "api/method", &cfg)
|
||||||
|
require.Error(t, err)
|
||||||
|
assert.True(t, strings.HasPrefix(err.Error(), `Validation of URL "://host/root" failed`))
|
||||||
|
}
|
||||||
|
|
||||||
type CloseNotifierResponseRecorder struct {
|
type CloseNotifierResponseRecorder struct {
|
||||||
*httptest.ResponseRecorder
|
*httptest.ResponseRecorder
|
||||||
closeChan chan bool
|
closeChan chan bool
|
||||||
@ -553,7 +589,8 @@ func getDatasourceProxiedRequest(ctx *models.ReqContext, cfg *setting.Cfg) *http
|
|||||||
Url: "http://host/root/",
|
Url: "http://host/root/",
|
||||||
}
|
}
|
||||||
|
|
||||||
proxy := NewDataSourceProxy(ds, plugin, ctx, "", cfg)
|
proxy, err := NewDataSourceProxy(ds, plugin, ctx, "", cfg)
|
||||||
|
So(err, ShouldBeNil)
|
||||||
req, err := http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
req, err := http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
||||||
So(err, ShouldBeNil)
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
@ -662,7 +699,8 @@ func createAuthTest(dsType string, authType string, authCheck string, useSecureJ
|
|||||||
func runDatasourceAuthTest(test *Test) {
|
func runDatasourceAuthTest(test *Test) {
|
||||||
plugin := &plugins.DataSourcePlugin{}
|
plugin := &plugins.DataSourcePlugin{}
|
||||||
ctx := &models.ReqContext{}
|
ctx := &models.ReqContext{}
|
||||||
proxy := NewDataSourceProxy(test.datasource, plugin, ctx, "", &setting.Cfg{})
|
proxy, err := NewDataSourceProxy(test.datasource, plugin, ctx, "", &setting.Cfg{})
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
req, err := http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
req, err := http.NewRequest(http.MethodGet, "http://grafana.com/sub", nil)
|
||||||
So(err, ShouldBeNil)
|
So(err, ShouldBeNil)
|
||||||
|
Loading…
Reference in New Issue
Block a user