mirror of
https://github.com/grafana/grafana.git
synced 2025-01-16 19:52:33 -06:00
2f942c57e8
Co-authored-by: Ryan McKinley <ryantxu@gmail.com>
245 lines
6.5 KiB
Go
245 lines
6.5 KiB
Go
package store
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"strings"
|
|
|
|
"github.com/grafana/grafana/pkg/api/response"
|
|
"github.com/grafana/grafana/pkg/models"
|
|
"github.com/grafana/grafana/pkg/util"
|
|
"github.com/grafana/grafana/pkg/web"
|
|
)
|
|
|
|
var errFileTooBig = response.Error(400, "Please limit file uploaded under 1MB", errors.New("file is too big"))
|
|
|
|
// HTTPStorageService passes raw HTTP requests to a well typed storage service
|
|
type HTTPStorageService interface {
|
|
List(c *models.ReqContext) response.Response
|
|
Read(c *models.ReqContext) response.Response
|
|
Delete(c *models.ReqContext) response.Response
|
|
DeleteFolder(c *models.ReqContext) response.Response
|
|
CreateFolder(c *models.ReqContext) response.Response
|
|
Upload(c *models.ReqContext) response.Response
|
|
}
|
|
|
|
type httpStorage struct {
|
|
store StorageService
|
|
}
|
|
|
|
func ProvideHTTPService(store StorageService) HTTPStorageService {
|
|
return &httpStorage{
|
|
store: store,
|
|
}
|
|
}
|
|
|
|
func UploadErrorToStatusCode(err error) int {
|
|
switch {
|
|
case errors.Is(err, ErrUploadFeatureDisabled):
|
|
return 404
|
|
|
|
case errors.Is(err, ErrUnsupportedStorage):
|
|
return 400
|
|
|
|
case errors.Is(err, ErrValidationFailed):
|
|
return 400
|
|
|
|
case errors.Is(err, ErrFileAlreadyExists):
|
|
return 400
|
|
|
|
default:
|
|
return 500
|
|
}
|
|
}
|
|
|
|
func (s *httpStorage) Upload(c *models.ReqContext) response.Response {
|
|
// 32 MB is the default used by FormFile()
|
|
if err := c.Req.ParseMultipartForm(32 << 20); err != nil {
|
|
return response.Error(400, "error in parsing form", err)
|
|
}
|
|
c.Req.Body = http.MaxBytesReader(c.Resp, c.Req.Body, MAX_UPLOAD_SIZE)
|
|
if err := c.Req.ParseMultipartForm(MAX_UPLOAD_SIZE); err != nil {
|
|
msg := fmt.Sprintf("Please limit file uploaded under %s", util.ByteCountSI(MAX_UPLOAD_SIZE))
|
|
return response.Error(400, msg, err)
|
|
}
|
|
|
|
files := c.Req.MultipartForm.File["file"]
|
|
if len(files) != 1 {
|
|
return response.JSON(400, map[string]interface{}{
|
|
"message": "please upload files one at a time",
|
|
"err": true,
|
|
})
|
|
}
|
|
|
|
folder, ok := getMultipartFormValue(c.Req, "folder")
|
|
if !ok || folder == "" {
|
|
return response.JSON(400, map[string]interface{}{
|
|
"message": "please specify the upload folder",
|
|
"err": true,
|
|
})
|
|
}
|
|
overwriteExistingFile, _ := getMultipartFormValue(c.Req, "overwriteExistingFile")
|
|
|
|
fileHeader := files[0]
|
|
if fileHeader.Size > MAX_UPLOAD_SIZE {
|
|
return errFileTooBig
|
|
}
|
|
|
|
// restrict file size based on file size
|
|
// open each file to copy contents
|
|
file, err := fileHeader.Open()
|
|
if err != nil {
|
|
return response.Error(500, "Internal Server Error", err)
|
|
}
|
|
err = file.Close()
|
|
if err != nil {
|
|
return response.Error(500, "Internal Server Error", err)
|
|
}
|
|
data, err := ioutil.ReadAll(file)
|
|
if err != nil {
|
|
return response.Error(500, "Internal Server Error", err)
|
|
}
|
|
|
|
if (len(data)) > MAX_UPLOAD_SIZE {
|
|
return errFileTooBig
|
|
}
|
|
|
|
path := folder + "/" + fileHeader.Filename
|
|
|
|
mimeType := http.DetectContentType(data)
|
|
|
|
err = s.store.Upload(c.Req.Context(), c.SignedInUser, &UploadRequest{
|
|
Contents: data,
|
|
MimeType: mimeType,
|
|
EntityType: EntityTypeImage,
|
|
Path: path,
|
|
OverwriteExistingFile: overwriteExistingFile == "true",
|
|
})
|
|
|
|
if err != nil {
|
|
return response.Error(UploadErrorToStatusCode(err), err.Error(), err)
|
|
}
|
|
|
|
return response.JSON(200, map[string]interface{}{
|
|
"message": "Uploaded successfully",
|
|
"path": path,
|
|
"file": fileHeader.Filename,
|
|
"err": true,
|
|
})
|
|
}
|
|
|
|
func getMultipartFormValue(req *http.Request, key string) (string, bool) {
|
|
v, ok := req.MultipartForm.Value[key]
|
|
if !ok || len(v) != 1 {
|
|
return "", false
|
|
}
|
|
return v[0], ok
|
|
}
|
|
|
|
func (s *httpStorage) Read(c *models.ReqContext) response.Response {
|
|
// full path is api/storage/read/upload/example.jpg, but we only want the part after read
|
|
scope, path := getPathAndScope(c)
|
|
file, err := s.store.Read(c.Req.Context(), c.SignedInUser, scope+"/"+path)
|
|
if err != nil {
|
|
return response.Error(400, "cannot call read", err)
|
|
}
|
|
|
|
if file == nil || file.Contents == nil {
|
|
return response.Error(404, "file does not exist", err)
|
|
}
|
|
|
|
// set the correct content type for svg
|
|
if strings.HasSuffix(path, ".svg") {
|
|
c.Resp.Header().Set("Content-Type", "image/svg+xml")
|
|
}
|
|
return response.Respond(200, file.Contents)
|
|
}
|
|
|
|
func (s *httpStorage) Delete(c *models.ReqContext) response.Response {
|
|
// full path is api/storage/delete/upload/example.jpg, but we only want the part after upload
|
|
scope, path := getPathAndScope(c)
|
|
|
|
err := s.store.Delete(c.Req.Context(), c.SignedInUser, scope+"/"+path)
|
|
if err != nil {
|
|
return response.Error(400, "failed to delete the file: "+err.Error(), err)
|
|
}
|
|
return response.JSON(200, map[string]interface{}{
|
|
"message": "Removed file from storage",
|
|
"success": true,
|
|
"path": path,
|
|
})
|
|
}
|
|
|
|
func (s *httpStorage) DeleteFolder(c *models.ReqContext) response.Response {
|
|
body, err := io.ReadAll(c.Req.Body)
|
|
if err != nil {
|
|
return response.Error(500, "error reading bytes", err)
|
|
}
|
|
|
|
cmd := &DeleteFolderCmd{}
|
|
err = json.Unmarshal(body, cmd)
|
|
if err != nil {
|
|
return response.Error(400, "error parsing body", err)
|
|
}
|
|
|
|
if cmd.Path == "" {
|
|
return response.Error(400, "empty path", err)
|
|
}
|
|
|
|
// full path is api/storage/delete/upload/example.jpg, but we only want the part after upload
|
|
_, path := getPathAndScope(c)
|
|
if err := s.store.DeleteFolder(c.Req.Context(), c.SignedInUser, cmd); err != nil {
|
|
return response.Error(400, "failed to delete the folder: "+err.Error(), err)
|
|
}
|
|
|
|
return response.JSON(200, map[string]interface{}{
|
|
"message": "Removed folder from storage",
|
|
"success": true,
|
|
"path": path,
|
|
})
|
|
}
|
|
|
|
func (s *httpStorage) CreateFolder(c *models.ReqContext) response.Response {
|
|
body, err := io.ReadAll(c.Req.Body)
|
|
if err != nil {
|
|
return response.Error(500, "error reading bytes", err)
|
|
}
|
|
|
|
cmd := &CreateFolderCmd{}
|
|
err = json.Unmarshal(body, cmd)
|
|
if err != nil {
|
|
return response.Error(400, "error parsing body", err)
|
|
}
|
|
|
|
if cmd.Path == "" {
|
|
return response.Error(400, "empty path", err)
|
|
}
|
|
|
|
if err := s.store.CreateFolder(c.Req.Context(), c.SignedInUser, cmd); err != nil {
|
|
return response.Error(400, "failed to create the folder: "+err.Error(), err)
|
|
}
|
|
|
|
return response.JSON(200, map[string]interface{}{
|
|
"message": "Folder created",
|
|
"success": true,
|
|
"path": cmd.Path,
|
|
})
|
|
}
|
|
|
|
func (s *httpStorage) List(c *models.ReqContext) response.Response {
|
|
params := web.Params(c.Req)
|
|
path := params["*"]
|
|
frame, err := s.store.List(c.Req.Context(), c.SignedInUser, path)
|
|
if err != nil {
|
|
return response.Error(400, "error reading path", err)
|
|
}
|
|
if frame == nil {
|
|
return response.Error(404, "not found", nil)
|
|
}
|
|
return response.JSONStreaming(http.StatusOK, frame)
|
|
}
|