package api import ( "bytes" "encoding/json" "fmt" "net/http" "time" "github.com/grafana/grafana/pkg/api/dtos" "github.com/grafana/grafana/pkg/bus" "github.com/grafana/grafana/pkg/components/simplejson" "github.com/grafana/grafana/pkg/infra/metrics" "github.com/grafana/grafana/pkg/models" "github.com/grafana/grafana/pkg/services/guardian" "github.com/grafana/grafana/pkg/setting" "github.com/grafana/grafana/pkg/util" ) var client = &http.Client{ Timeout: time.Second * 5, Transport: &http.Transport{Proxy: http.ProxyFromEnvironment}, } func GetSharingOptions(c *models.ReqContext) { c.JSON(200, util.DynMap{ "externalSnapshotURL": setting.ExternalSnapshotUrl, "externalSnapshotName": setting.ExternalSnapshotName, "externalEnabled": setting.ExternalEnabled, }) } type CreateExternalSnapshotResponse struct { Key string `json:"key"` DeleteKey string `json:"deleteKey"` Url string `json:"url"` DeleteUrl string `json:"deleteUrl"` } func createExternalDashboardSnapshot(cmd models.CreateDashboardSnapshotCommand) (*CreateExternalSnapshotResponse, error) { var createSnapshotResponse CreateExternalSnapshotResponse message := map[string]interface{}{ "name": cmd.Name, "expires": cmd.Expires, "dashboard": cmd.Dashboard, } messageBytes, err := simplejson.NewFromAny(message).Encode() if err != nil { return nil, err } response, err := client.Post(setting.ExternalSnapshotUrl+"/api/snapshots", "application/json", bytes.NewBuffer(messageBytes)) if err != nil { return nil, err } defer response.Body.Close() if response.StatusCode != 200 { return nil, fmt.Errorf("create external snapshot response status code %d", response.StatusCode) } if err := json.NewDecoder(response.Body).Decode(&createSnapshotResponse); err != nil { return nil, err } return &createSnapshotResponse, nil } // POST /api/snapshots func CreateDashboardSnapshot(c *models.ReqContext, cmd models.CreateDashboardSnapshotCommand) { if cmd.Name == "" { cmd.Name = "Unnamed snapshot" } var url string cmd.ExternalUrl = "" cmd.OrgId = c.OrgId cmd.UserId = c.UserId if cmd.External { if !setting.ExternalEnabled { c.JsonApiErr(403, "External dashboard creation is disabled", nil) return } response, err := createExternalDashboardSnapshot(cmd) if err != nil { c.JsonApiErr(500, "Failed to create external snapshot", err) return } url = response.Url cmd.Key = response.Key cmd.DeleteKey = response.DeleteKey cmd.ExternalUrl = response.Url cmd.ExternalDeleteUrl = response.DeleteUrl cmd.Dashboard = simplejson.New() metrics.MApiDashboardSnapshotExternal.Inc() } else { if cmd.Key == "" { var err error cmd.Key, err = util.GetRandomString(32) if err != nil { c.JsonApiErr(500, "Could not generate random string", err) return } } if cmd.DeleteKey == "" { var err error cmd.DeleteKey, err = util.GetRandomString(32) if err != nil { c.JsonApiErr(500, "Could not generate random string", err) return } } url = setting.ToAbsUrl("dashboard/snapshot/" + cmd.Key) metrics.MApiDashboardSnapshotCreate.Inc() } if err := bus.Dispatch(&cmd); err != nil { c.JsonApiErr(500, "Failed to create snapshot", err) return } c.JSON(200, util.DynMap{ "key": cmd.Key, "deleteKey": cmd.DeleteKey, "url": url, "deleteUrl": setting.ToAbsUrl("api/snapshots-delete/" + cmd.DeleteKey), }) } // GET /api/snapshots/:key func GetDashboardSnapshot(c *models.ReqContext) Response { key := c.Params(":key") query := &models.GetDashboardSnapshotQuery{Key: key} err := bus.Dispatch(query) if err != nil { return Error(500, "Failed to get dashboard snapshot", err) } snapshot := query.Result // expired snapshots should also be removed from db if snapshot.Expires.Before(time.Now()) { return Error(404, "Dashboard snapshot not found", err) } dashboard, err := snapshot.DashboardJSON() if err != nil { return Error(500, "Failed to get dashboard data for dashboard snapshot", err) } dto := dtos.DashboardFullWithMeta{ Dashboard: dashboard, Meta: dtos.DashboardMeta{ Type: models.DashTypeSnapshot, IsSnapshot: true, Created: snapshot.Created, Expires: snapshot.Expires, }, } metrics.MApiDashboardSnapshotGet.Inc() return JSON(200, dto).Header("Cache-Control", "public, max-age=3600") } func deleteExternalDashboardSnapshot(externalUrl string) error { response, err := client.Get(externalUrl) if err != nil { return err } defer response.Body.Close() if response.StatusCode == 200 { return nil } // Gracefully ignore "snapshot not found" errors as they could have already // been removed either via the cleanup script or by request. if response.StatusCode == 500 { var respJson map[string]interface{} if err := json.NewDecoder(response.Body).Decode(&respJson); err != nil { return err } if respJson["message"] == "Failed to get dashboard snapshot" { return nil } } return fmt.Errorf("unexpected response when deleting external snapshot, status code: %d", response.StatusCode) } // GET /api/snapshots-delete/:deleteKey func DeleteDashboardSnapshotByDeleteKey(c *models.ReqContext) Response { key := c.Params(":deleteKey") query := &models.GetDashboardSnapshotQuery{DeleteKey: key} err := bus.Dispatch(query) if err != nil { return Error(500, "Failed to get dashboard snapshot", err) } if query.Result.External { err := deleteExternalDashboardSnapshot(query.Result.ExternalDeleteUrl) if err != nil { return Error(500, "Failed to delete external dashboard", err) } } cmd := &models.DeleteDashboardSnapshotCommand{DeleteKey: query.Result.DeleteKey} if err := bus.Dispatch(cmd); err != nil { return Error(500, "Failed to delete dashboard snapshot", err) } return JSON(200, util.DynMap{"message": "Snapshot deleted. It might take an hour before it's cleared from any CDN caches."}) } // DELETE /api/snapshots/:key func DeleteDashboardSnapshot(c *models.ReqContext) Response { key := c.Params(":key") query := &models.GetDashboardSnapshotQuery{Key: key} err := bus.Dispatch(query) if err != nil { return Error(500, "Failed to get dashboard snapshot", err) } if query.Result == nil { return Error(404, "Failed to get dashboard snapshot", nil) } dashboard, err := query.Result.DashboardJSON() if err != nil { return Error(500, "Failed to get dashboard data for dashboard snapshot", err) } dashboardID := dashboard.Get("id").MustInt64() guardian := guardian.New(dashboardID, c.OrgId, c.SignedInUser) canEdit, err := guardian.CanEdit() if err != nil { return Error(500, "Error while checking permissions for snapshot", err) } if !canEdit && query.Result.UserId != c.SignedInUser.UserId { return Error(403, "Access denied to this snapshot", nil) } if query.Result.External { err := deleteExternalDashboardSnapshot(query.Result.ExternalDeleteUrl) if err != nil { return Error(500, "Failed to delete external dashboard", err) } } cmd := &models.DeleteDashboardSnapshotCommand{DeleteKey: query.Result.DeleteKey} if err := bus.Dispatch(cmd); err != nil { return Error(500, "Failed to delete dashboard snapshot", err) } return JSON(200, util.DynMap{"message": "Snapshot deleted. It might take an hour before it's cleared from any CDN caches."}) } // GET /api/dashboard/snapshots func SearchDashboardSnapshots(c *models.ReqContext) Response { query := c.Query("query") limit := c.QueryInt("limit") if limit == 0 { limit = 1000 } searchQuery := models.GetDashboardSnapshotsQuery{ Name: query, Limit: limit, OrgId: c.OrgId, SignedInUser: c.SignedInUser, } err := bus.Dispatch(&searchQuery) if err != nil { return Error(500, "Search failed", err) } dtos := make([]*models.DashboardSnapshotDTO, len(searchQuery.Result)) for i, snapshot := range searchQuery.Result { dtos[i] = &models.DashboardSnapshotDTO{ Id: snapshot.Id, Name: snapshot.Name, Key: snapshot.Key, OrgId: snapshot.OrgId, UserId: snapshot.UserId, External: snapshot.External, ExternalUrl: snapshot.ExternalUrl, Expires: snapshot.Expires, Created: snapshot.Created, Updated: snapshot.Updated, } } return JSON(200, dtos) }