2017-05-26 07:57:09 +00:00
|
|
|
// Copyright 2017 Vector Creations Ltd
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package writers
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
2017-05-26 12:42:51 +00:00
|
|
|
"fmt"
|
2017-05-31 11:46:21 +00:00
|
|
|
"io"
|
2017-06-01 14:04:41 +00:00
|
|
|
"mime"
|
2017-05-26 07:57:09 +00:00
|
|
|
"net/http"
|
2017-05-31 11:46:21 +00:00
|
|
|
"os"
|
2017-05-31 15:56:11 +00:00
|
|
|
"path/filepath"
|
2017-05-26 12:42:51 +00:00
|
|
|
"regexp"
|
2017-05-31 11:46:21 +00:00
|
|
|
"strconv"
|
2017-06-06 23:12:49 +00:00
|
|
|
"strings"
|
2017-05-31 15:56:11 +00:00
|
|
|
"sync"
|
2017-05-26 07:57:09 +00:00
|
|
|
|
|
|
|
log "github.com/Sirupsen/logrus"
|
|
|
|
"github.com/matrix-org/dendrite/clientapi/jsonerror"
|
2017-06-19 14:21:04 +00:00
|
|
|
"github.com/matrix-org/dendrite/common/config"
|
2017-05-31 11:46:21 +00:00
|
|
|
"github.com/matrix-org/dendrite/mediaapi/fileutils"
|
|
|
|
"github.com/matrix-org/dendrite/mediaapi/storage"
|
2017-06-06 23:12:49 +00:00
|
|
|
"github.com/matrix-org/dendrite/mediaapi/thumbnailer"
|
2017-05-26 07:57:09 +00:00
|
|
|
"github.com/matrix-org/dendrite/mediaapi/types"
|
|
|
|
"github.com/matrix-org/gomatrixserverlib"
|
|
|
|
"github.com/matrix-org/util"
|
|
|
|
)
|
|
|
|
|
2017-05-26 12:42:51 +00:00
|
|
|
const mediaIDCharacters = "A-Za-z0-9_=-"
|
|
|
|
|
|
|
|
// Note: unfortunately regex.MustCompile() cannot be assigned to a const
|
|
|
|
var mediaIDRegex = regexp.MustCompile("[" + mediaIDCharacters + "]+")
|
|
|
|
|
2017-06-06 23:12:49 +00:00
|
|
|
// downloadRequest metadata included in or derivable from a download or thumbnail request
|
2017-05-26 07:57:09 +00:00
|
|
|
// https://matrix.org/docs/spec/client_server/r0.2.0.html#get-matrix-media-r0-download-servername-mediaid
|
2017-06-06 23:12:49 +00:00
|
|
|
// http://matrix.org/docs/spec/client_server/r0.2.0.html#get-matrix-media-r0-thumbnail-servername-mediaid
|
2017-05-26 07:57:09 +00:00
|
|
|
type downloadRequest struct {
|
2017-06-06 23:12:49 +00:00
|
|
|
MediaMetadata *types.MediaMetadata
|
|
|
|
IsThumbnailRequest bool
|
|
|
|
ThumbnailSize types.ThumbnailSize
|
|
|
|
Logger *log.Entry
|
2017-05-26 07:57:09 +00:00
|
|
|
}
|
|
|
|
|
2017-06-06 23:12:49 +00:00
|
|
|
// Download implements /download amd /thumbnail
|
2017-05-31 11:46:21 +00:00
|
|
|
// Files from this server (i.e. origin == cfg.ServerName) are served directly
|
2017-05-31 15:56:11 +00:00
|
|
|
// Files from remote servers (i.e. origin != cfg.ServerName) are cached locally.
|
|
|
|
// If they are present in the cache, they are served directly.
|
|
|
|
// If they are not present in the cache, they are obtained from the remote server and
|
|
|
|
// simultaneously served back to the client and written into the cache.
|
2017-06-19 14:21:04 +00:00
|
|
|
func Download(w http.ResponseWriter, req *http.Request, origin gomatrixserverlib.ServerName, mediaID types.MediaID, cfg *config.Dendrite, db *storage.Database, activeRemoteRequests *types.ActiveRemoteRequests, activeThumbnailGeneration *types.ActiveThumbnailGeneration, isThumbnailRequest bool) {
|
2017-05-26 07:57:09 +00:00
|
|
|
r := &downloadRequest{
|
|
|
|
MediaMetadata: &types.MediaMetadata{
|
|
|
|
MediaID: mediaID,
|
|
|
|
Origin: origin,
|
|
|
|
},
|
2017-06-06 23:12:49 +00:00
|
|
|
IsThumbnailRequest: isThumbnailRequest,
|
2017-05-31 12:30:57 +00:00
|
|
|
Logger: util.GetLogger(req.Context()).WithFields(log.Fields{
|
|
|
|
"Origin": origin,
|
|
|
|
"MediaID": mediaID,
|
|
|
|
}),
|
2017-05-26 07:57:09 +00:00
|
|
|
}
|
|
|
|
|
2017-06-06 23:12:49 +00:00
|
|
|
if r.IsThumbnailRequest {
|
|
|
|
width, err := strconv.Atoi(req.FormValue("width"))
|
|
|
|
if err != nil {
|
|
|
|
width = -1
|
|
|
|
}
|
|
|
|
height, err := strconv.Atoi(req.FormValue("height"))
|
|
|
|
if err != nil {
|
|
|
|
height = -1
|
|
|
|
}
|
|
|
|
r.ThumbnailSize = types.ThumbnailSize{
|
|
|
|
Width: width,
|
|
|
|
Height: height,
|
|
|
|
ResizeMethod: strings.ToLower(req.FormValue("method")),
|
|
|
|
}
|
|
|
|
r.Logger.WithFields(log.Fields{
|
|
|
|
"RequestedWidth": r.ThumbnailSize.Width,
|
|
|
|
"RequestedHeight": r.ThumbnailSize.Height,
|
|
|
|
"RequestedResizeMethod": r.ThumbnailSize.ResizeMethod,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-05-26 07:57:09 +00:00
|
|
|
// request validation
|
|
|
|
if req.Method != "GET" {
|
|
|
|
r.jsonErrorResponse(w, util.JSONResponse{
|
|
|
|
Code: 405,
|
|
|
|
JSON: jsonerror.Unknown("request method must be GET"),
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if resErr := r.Validate(); resErr != nil {
|
|
|
|
r.jsonErrorResponse(w, *resErr)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-06-06 23:12:49 +00:00
|
|
|
if resErr := r.doDownload(w, cfg, db, activeRemoteRequests, activeThumbnailGeneration); resErr != nil {
|
2017-05-31 11:46:21 +00:00
|
|
|
r.jsonErrorResponse(w, *resErr)
|
|
|
|
return
|
|
|
|
}
|
2017-05-26 07:57:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *downloadRequest) jsonErrorResponse(w http.ResponseWriter, res util.JSONResponse) {
|
|
|
|
// Marshal JSON response into raw bytes to send as the HTTP body
|
|
|
|
resBytes, err := json.Marshal(res.JSON)
|
|
|
|
if err != nil {
|
|
|
|
r.Logger.WithError(err).Error("Failed to marshal JSONResponse")
|
|
|
|
// this should never fail to be marshalled so drop err to the floor
|
|
|
|
res = util.MessageResponse(500, "Internal Server Error")
|
|
|
|
resBytes, _ = json.Marshal(res.JSON)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set status code and write the body
|
|
|
|
w.WriteHeader(res.Code)
|
|
|
|
r.Logger.WithField("code", res.Code).Infof("Responding (%d bytes)", len(resBytes))
|
|
|
|
w.Write(resBytes)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Validate validates the downloadRequest fields
|
|
|
|
func (r *downloadRequest) Validate() *util.JSONResponse {
|
2017-05-26 12:42:51 +00:00
|
|
|
if mediaIDRegex.MatchString(string(r.MediaMetadata.MediaID)) == false {
|
2017-05-26 07:57:09 +00:00
|
|
|
return &util.JSONResponse{
|
|
|
|
Code: 404,
|
2017-05-26 12:42:51 +00:00
|
|
|
JSON: jsonerror.NotFound(fmt.Sprintf("mediaId must be a non-empty string using only characters in %v", mediaIDCharacters)),
|
2017-05-26 07:57:09 +00:00
|
|
|
}
|
|
|
|
}
|
2017-05-26 12:59:45 +00:00
|
|
|
// Note: the origin will be validated either by comparison to the configured server name of this homeserver
|
|
|
|
// or by a DNS SRV record lookup when creating a request for remote files
|
2017-05-26 07:57:09 +00:00
|
|
|
if r.MediaMetadata.Origin == "" {
|
|
|
|
return &util.JSONResponse{
|
|
|
|
Code: 404,
|
|
|
|
JSON: jsonerror.NotFound("serverName must be a non-empty string"),
|
|
|
|
}
|
|
|
|
}
|
2017-06-06 23:12:49 +00:00
|
|
|
|
|
|
|
if r.IsThumbnailRequest {
|
|
|
|
if r.ThumbnailSize.Width <= 0 || r.ThumbnailSize.Height <= 0 {
|
|
|
|
return &util.JSONResponse{
|
|
|
|
Code: 400,
|
|
|
|
JSON: jsonerror.Unknown("width and height must be greater than 0"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Default method to scale if not set
|
|
|
|
if r.ThumbnailSize.ResizeMethod == "" {
|
|
|
|
r.ThumbnailSize.ResizeMethod = "scale"
|
|
|
|
}
|
|
|
|
if r.ThumbnailSize.ResizeMethod != "crop" && r.ThumbnailSize.ResizeMethod != "scale" {
|
|
|
|
return &util.JSONResponse{
|
|
|
|
Code: 400,
|
|
|
|
JSON: jsonerror.Unknown("method must be one of crop or scale"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-05-26 07:57:09 +00:00
|
|
|
return nil
|
|
|
|
}
|
2017-05-31 11:46:21 +00:00
|
|
|
|
2017-06-19 14:21:04 +00:00
|
|
|
func (r *downloadRequest) doDownload(w http.ResponseWriter, cfg *config.Dendrite, db *storage.Database, activeRemoteRequests *types.ActiveRemoteRequests, activeThumbnailGeneration *types.ActiveThumbnailGeneration) *util.JSONResponse {
|
2017-05-31 11:46:21 +00:00
|
|
|
// check if we have a record of the media in our database
|
2017-05-31 12:29:28 +00:00
|
|
|
mediaMetadata, err := db.GetMediaMetadata(r.MediaMetadata.MediaID, r.MediaMetadata.Origin)
|
|
|
|
if err != nil {
|
|
|
|
r.Logger.WithError(err).Error("Error querying the database.")
|
2017-05-31 15:41:42 +00:00
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return &resErr
|
2017-05-31 12:29:28 +00:00
|
|
|
}
|
|
|
|
if mediaMetadata == nil {
|
2017-06-19 14:21:04 +00:00
|
|
|
if r.MediaMetadata.Origin == cfg.Matrix.ServerName {
|
2017-05-31 11:46:21 +00:00
|
|
|
// If we do not have a record and the origin is local, the file is not found
|
|
|
|
return &util.JSONResponse{
|
|
|
|
Code: 404,
|
|
|
|
JSON: jsonerror.NotFound(fmt.Sprintf("File with media ID %q does not exist", r.MediaMetadata.MediaID)),
|
|
|
|
}
|
|
|
|
}
|
2017-05-31 15:56:11 +00:00
|
|
|
// If we do not have a record and the origin is remote, we need to fetch it and respond with that file
|
2017-06-06 23:12:49 +00:00
|
|
|
resErr := r.getRemoteFile(cfg, db, activeRemoteRequests, activeThumbnailGeneration)
|
2017-06-01 10:32:15 +00:00
|
|
|
if resErr != nil {
|
|
|
|
return resErr
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If we have a record, we can respond from the local file
|
|
|
|
r.MediaMetadata = mediaMetadata
|
2017-05-31 11:46:21 +00:00
|
|
|
}
|
2017-06-19 14:21:04 +00:00
|
|
|
return r.respondFromLocalFile(w, cfg.Media.AbsBasePath, activeThumbnailGeneration, cfg.Media.MaxThumbnailGenerators, db, cfg.Media.DynamicThumbnails, cfg.Media.ThumbnailSizes)
|
2017-05-31 11:46:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// respondFromLocalFile reads a file from local storage and writes it to the http.ResponseWriter
|
|
|
|
// Returns a util.JSONResponse error in case of error
|
2017-06-19 14:21:04 +00:00
|
|
|
func (r *downloadRequest) respondFromLocalFile(w http.ResponseWriter, absBasePath config.Path, activeThumbnailGeneration *types.ActiveThumbnailGeneration, maxThumbnailGenerators int, db *storage.Database, dynamicThumbnails bool, thumbnailSizes []config.ThumbnailSize) *util.JSONResponse {
|
2017-05-31 11:46:21 +00:00
|
|
|
filePath, err := fileutils.GetPathFromBase64Hash(r.MediaMetadata.Base64Hash, absBasePath)
|
|
|
|
if err != nil {
|
2017-05-31 12:33:49 +00:00
|
|
|
r.Logger.WithError(err).Error("Failed to get file path from metadata")
|
2017-05-31 15:41:42 +00:00
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return &resErr
|
2017-05-31 11:46:21 +00:00
|
|
|
}
|
|
|
|
file, err := os.Open(filePath)
|
|
|
|
defer file.Close()
|
|
|
|
if err != nil {
|
2017-05-31 12:33:49 +00:00
|
|
|
r.Logger.WithError(err).Error("Failed to open file")
|
2017-05-31 15:41:42 +00:00
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return &resErr
|
2017-05-31 11:46:21 +00:00
|
|
|
}
|
|
|
|
stat, err := file.Stat()
|
|
|
|
if err != nil {
|
2017-05-31 12:33:49 +00:00
|
|
|
r.Logger.WithError(err).Error("Failed to stat file")
|
2017-05-31 15:41:42 +00:00
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return &resErr
|
2017-05-31 11:46:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if r.MediaMetadata.FileSizeBytes > 0 && int64(r.MediaMetadata.FileSizeBytes) != stat.Size() {
|
|
|
|
r.Logger.WithFields(log.Fields{
|
|
|
|
"fileSizeDatabase": r.MediaMetadata.FileSizeBytes,
|
|
|
|
"fileSizeDisk": stat.Size(),
|
|
|
|
}).Warn("File size in database and on-disk differ.")
|
2017-05-31 15:41:42 +00:00
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return &resErr
|
2017-05-31 11:46:21 +00:00
|
|
|
}
|
|
|
|
|
2017-06-06 23:12:49 +00:00
|
|
|
var responseFile *os.File
|
|
|
|
var responseMetadata *types.MediaMetadata
|
|
|
|
if r.IsThumbnailRequest {
|
|
|
|
thumbFile, thumbMetadata, resErr := r.getThumbnailFile(types.Path(filePath), activeThumbnailGeneration, maxThumbnailGenerators, db, dynamicThumbnails, thumbnailSizes)
|
|
|
|
if thumbFile != nil {
|
|
|
|
defer thumbFile.Close()
|
|
|
|
}
|
|
|
|
if resErr != nil {
|
|
|
|
return resErr
|
|
|
|
}
|
|
|
|
if thumbFile == nil {
|
|
|
|
r.Logger.WithFields(log.Fields{
|
|
|
|
"UploadName": r.MediaMetadata.UploadName,
|
|
|
|
"Base64Hash": r.MediaMetadata.Base64Hash,
|
|
|
|
"FileSizeBytes": r.MediaMetadata.FileSizeBytes,
|
|
|
|
"ContentType": r.MediaMetadata.ContentType,
|
|
|
|
}).Info("No good thumbnail found. Responding with original file.")
|
|
|
|
responseFile = file
|
|
|
|
responseMetadata = r.MediaMetadata
|
|
|
|
} else {
|
|
|
|
r.Logger.Info("Responding with thumbnail")
|
|
|
|
responseFile = thumbFile
|
|
|
|
responseMetadata = thumbMetadata.MediaMetadata
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
r.Logger.WithFields(log.Fields{
|
|
|
|
"UploadName": r.MediaMetadata.UploadName,
|
|
|
|
"Base64Hash": r.MediaMetadata.Base64Hash,
|
|
|
|
"FileSizeBytes": r.MediaMetadata.FileSizeBytes,
|
|
|
|
"ContentType": r.MediaMetadata.ContentType,
|
|
|
|
}).Info("Responding with file")
|
|
|
|
responseFile = file
|
|
|
|
responseMetadata = r.MediaMetadata
|
|
|
|
}
|
2017-05-31 11:46:21 +00:00
|
|
|
|
2017-06-06 23:12:49 +00:00
|
|
|
w.Header().Set("Content-Type", string(responseMetadata.ContentType))
|
|
|
|
w.Header().Set("Content-Length", strconv.FormatInt(int64(responseMetadata.FileSizeBytes), 10))
|
2017-05-31 11:46:21 +00:00
|
|
|
contentSecurityPolicy := "default-src 'none';" +
|
|
|
|
" script-src 'none';" +
|
|
|
|
" plugin-types application/pdf;" +
|
|
|
|
" style-src 'unsafe-inline';" +
|
|
|
|
" object-src 'self';"
|
|
|
|
w.Header().Set("Content-Security-Policy", contentSecurityPolicy)
|
|
|
|
|
2017-06-06 23:12:49 +00:00
|
|
|
if bytesResponded, err := io.Copy(w, responseFile); err != nil {
|
2017-05-31 11:46:21 +00:00
|
|
|
r.Logger.WithError(err).Warn("Failed to copy from cache")
|
|
|
|
if bytesResponded == 0 {
|
2017-05-31 15:41:42 +00:00
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return &resErr
|
2017-05-31 11:46:21 +00:00
|
|
|
}
|
|
|
|
// If we have written any data then we have already responded with 200 OK and all we can do is close the connection
|
2017-05-31 13:39:19 +00:00
|
|
|
return nil
|
2017-05-31 11:46:21 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2017-05-31 15:56:11 +00:00
|
|
|
|
2017-06-06 23:12:49 +00:00
|
|
|
// Note: Thumbnail generation may be ongoing asynchronously.
|
2017-06-19 14:21:04 +00:00
|
|
|
func (r *downloadRequest) getThumbnailFile(filePath types.Path, activeThumbnailGeneration *types.ActiveThumbnailGeneration, maxThumbnailGenerators int, db *storage.Database, dynamicThumbnails bool, thumbnailSizes []config.ThumbnailSize) (*os.File, *types.ThumbnailMetadata, *util.JSONResponse) {
|
2017-06-06 23:12:49 +00:00
|
|
|
var thumbnail *types.ThumbnailMetadata
|
|
|
|
var resErr *util.JSONResponse
|
|
|
|
|
|
|
|
if dynamicThumbnails {
|
|
|
|
thumbnail, resErr = r.generateThumbnail(filePath, r.ThumbnailSize, activeThumbnailGeneration, maxThumbnailGenerators, db)
|
|
|
|
if resErr != nil {
|
|
|
|
return nil, nil, resErr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// If dynamicThumbnails is true but there are too many thumbnails being actively generated, we can fall back
|
|
|
|
// to trying to use a pre-generated thumbnail
|
|
|
|
if thumbnail == nil {
|
|
|
|
thumbnails, err := db.GetThumbnails(r.MediaMetadata.MediaID, r.MediaMetadata.Origin)
|
|
|
|
if err != nil {
|
|
|
|
r.Logger.WithError(err).Error("Error looking up thumbnails")
|
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return nil, nil, &resErr
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we get a thumbnailSize, a pre-generated thumbnail would be best but it is not yet generated.
|
|
|
|
// If we get a thumbnail, we're done.
|
|
|
|
var thumbnailSize *types.ThumbnailSize
|
|
|
|
thumbnail, thumbnailSize = thumbnailer.SelectThumbnail(r.ThumbnailSize, thumbnails, thumbnailSizes)
|
|
|
|
// If dynamicThumbnails is true and we are not over-loaded then we would have generated what was requested above.
|
|
|
|
// So we don't try to generate a pre-generated thumbnail here.
|
|
|
|
if thumbnailSize != nil && dynamicThumbnails == false {
|
|
|
|
r.Logger.WithFields(log.Fields{
|
|
|
|
"Width": thumbnailSize.Width,
|
|
|
|
"Height": thumbnailSize.Height,
|
|
|
|
"ResizeMethod": thumbnailSize.ResizeMethod,
|
|
|
|
}).Info("Pre-generating thumbnail for immediate response.")
|
|
|
|
thumbnail, resErr = r.generateThumbnail(filePath, *thumbnailSize, activeThumbnailGeneration, maxThumbnailGenerators, db)
|
|
|
|
if resErr != nil {
|
|
|
|
return nil, nil, resErr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if thumbnail == nil {
|
|
|
|
return nil, nil, nil
|
|
|
|
}
|
|
|
|
r.Logger = r.Logger.WithFields(log.Fields{
|
|
|
|
"Width": thumbnail.ThumbnailSize.Width,
|
|
|
|
"Height": thumbnail.ThumbnailSize.Height,
|
|
|
|
"ResizeMethod": thumbnail.ThumbnailSize.ResizeMethod,
|
|
|
|
"FileSizeBytes": thumbnail.MediaMetadata.FileSizeBytes,
|
|
|
|
"ContentType": thumbnail.MediaMetadata.ContentType,
|
|
|
|
})
|
|
|
|
thumbPath := string(thumbnailer.GetThumbnailPath(types.Path(filePath), thumbnail.ThumbnailSize))
|
|
|
|
thumbFile, err := os.Open(string(thumbPath))
|
|
|
|
if err != nil {
|
|
|
|
r.Logger.WithError(err).Warn("Failed to open file")
|
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return thumbFile, nil, &resErr
|
|
|
|
}
|
|
|
|
thumbStat, err := thumbFile.Stat()
|
|
|
|
if err != nil {
|
|
|
|
r.Logger.WithError(err).Warn("Failed to stat file")
|
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return thumbFile, nil, &resErr
|
|
|
|
}
|
|
|
|
if types.FileSizeBytes(thumbStat.Size()) != thumbnail.MediaMetadata.FileSizeBytes {
|
|
|
|
r.Logger.WithError(err).Warn("Thumbnail file sizes on disk and in database differ")
|
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return thumbFile, nil, &resErr
|
|
|
|
}
|
|
|
|
return thumbFile, thumbnail, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *downloadRequest) generateThumbnail(filePath types.Path, thumbnailSize types.ThumbnailSize, activeThumbnailGeneration *types.ActiveThumbnailGeneration, maxThumbnailGenerators int, db *storage.Database) (*types.ThumbnailMetadata, *util.JSONResponse) {
|
|
|
|
logger := r.Logger.WithFields(log.Fields{
|
|
|
|
"Width": thumbnailSize.Width,
|
|
|
|
"Height": thumbnailSize.Height,
|
|
|
|
"ResizeMethod": thumbnailSize.ResizeMethod,
|
|
|
|
})
|
|
|
|
busy, err := thumbnailer.GenerateThumbnail(filePath, thumbnailSize, r.MediaMetadata, activeThumbnailGeneration, maxThumbnailGenerators, db, r.Logger)
|
|
|
|
if err != nil {
|
|
|
|
logger.WithError(err).Error("Error creating thumbnail")
|
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return nil, &resErr
|
|
|
|
}
|
|
|
|
if busy {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
var thumbnail *types.ThumbnailMetadata
|
|
|
|
thumbnail, err = db.GetThumbnail(r.MediaMetadata.MediaID, r.MediaMetadata.Origin, thumbnailSize.Width, thumbnailSize.Height, thumbnailSize.ResizeMethod)
|
|
|
|
if err != nil {
|
|
|
|
logger.WithError(err).Error("Error looking up thumbnails")
|
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return nil, &resErr
|
|
|
|
}
|
|
|
|
return thumbnail, nil
|
|
|
|
}
|
|
|
|
|
2017-06-01 10:32:15 +00:00
|
|
|
// getRemoteFile fetches the remote file and caches it locally
|
2017-05-31 15:56:11 +00:00
|
|
|
// A hash map of active remote requests to a struct containing a sync.Cond is used to only download remote files once,
|
|
|
|
// regardless of how many download requests are received.
|
2017-06-01 10:32:15 +00:00
|
|
|
// Note: The named errorResponse return variable is used in a deferred broadcast of the metadata and error response to waiting goroutines.
|
2017-05-31 15:56:11 +00:00
|
|
|
// Returns a util.JSONResponse error in case of error
|
2017-06-19 14:21:04 +00:00
|
|
|
func (r *downloadRequest) getRemoteFile(cfg *config.Dendrite, db *storage.Database, activeRemoteRequests *types.ActiveRemoteRequests, activeThumbnailGeneration *types.ActiveThumbnailGeneration) (errorResponse *util.JSONResponse) {
|
2017-06-01 10:32:15 +00:00
|
|
|
// Note: getMediaMetadataFromActiveRequest uses mutexes and conditions from activeRemoteRequests
|
|
|
|
mediaMetadata, resErr := r.getMediaMetadataFromActiveRequest(activeRemoteRequests)
|
2017-05-31 15:56:11 +00:00
|
|
|
if resErr != nil {
|
|
|
|
return resErr
|
|
|
|
} else if mediaMetadata != nil {
|
2017-06-01 10:32:15 +00:00
|
|
|
// If we got metadata from an active request, we can respond from the local file
|
2017-05-31 15:56:11 +00:00
|
|
|
r.MediaMetadata = mediaMetadata
|
|
|
|
} else {
|
2017-06-01 10:32:15 +00:00
|
|
|
// Note: This is an active request that MUST broadcastMediaMetadata to wake up waiting goroutines!
|
|
|
|
// Note: broadcastMediaMetadata uses mutexes and conditions from activeRemoteRequests
|
2017-06-01 12:44:00 +00:00
|
|
|
defer func() {
|
|
|
|
// Note: errorResponse is the named return variable so we wrap this in a closure to re-evaluate the arguments at defer-time
|
2017-06-01 12:54:59 +00:00
|
|
|
if err := recover(); err != nil {
|
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
r.broadcastMediaMetadata(activeRemoteRequests, &resErr)
|
|
|
|
panic(err)
|
|
|
|
}
|
2017-06-01 12:44:00 +00:00
|
|
|
r.broadcastMediaMetadata(activeRemoteRequests, errorResponse)
|
|
|
|
}()
|
2017-06-01 10:32:15 +00:00
|
|
|
|
|
|
|
// check if we have a record of the media in our database
|
|
|
|
mediaMetadata, err := db.GetMediaMetadata(r.MediaMetadata.MediaID, r.MediaMetadata.Origin)
|
|
|
|
if err != nil {
|
|
|
|
r.Logger.WithError(err).Error("Error querying the database.")
|
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return &resErr
|
|
|
|
}
|
|
|
|
|
|
|
|
if mediaMetadata == nil {
|
|
|
|
// If we do not have a record, we need to fetch the remote file first and then respond from the local file
|
2017-06-19 14:21:04 +00:00
|
|
|
resErr := r.fetchRemoteFileAndStoreMetadata(cfg.Media.AbsBasePath, *cfg.Media.MaxFileSizeBytes, db, cfg.Media.ThumbnailSizes, activeThumbnailGeneration, cfg.Media.MaxThumbnailGenerators)
|
2017-06-01 10:32:15 +00:00
|
|
|
if resErr != nil {
|
|
|
|
return resErr
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If we have a record, we can respond from the local file
|
|
|
|
r.MediaMetadata = mediaMetadata
|
2017-05-31 15:56:11 +00:00
|
|
|
}
|
|
|
|
}
|
2017-06-01 10:32:15 +00:00
|
|
|
return
|
2017-05-31 15:56:11 +00:00
|
|
|
}
|
|
|
|
|
2017-06-01 10:32:15 +00:00
|
|
|
func (r *downloadRequest) getMediaMetadataFromActiveRequest(activeRemoteRequests *types.ActiveRemoteRequests) (*types.MediaMetadata, *util.JSONResponse) {
|
|
|
|
// Check if there is an active remote request for the file
|
|
|
|
mxcURL := "mxc://" + string(r.MediaMetadata.Origin) + "/" + string(r.MediaMetadata.MediaID)
|
|
|
|
|
2017-05-31 15:56:11 +00:00
|
|
|
activeRemoteRequests.Lock()
|
|
|
|
defer activeRemoteRequests.Unlock()
|
|
|
|
|
|
|
|
if activeRemoteRequestResult, ok := activeRemoteRequests.MXCToResult[mxcURL]; ok {
|
|
|
|
r.Logger.Info("Waiting for another goroutine to fetch the remote file.")
|
|
|
|
|
2017-06-01 06:39:35 +00:00
|
|
|
// NOTE: Wait unlocks and locks again internally. There is still a deferred Unlock() that will unlock this.
|
2017-05-31 15:56:11 +00:00
|
|
|
activeRemoteRequestResult.Cond.Wait()
|
2017-06-01 10:32:15 +00:00
|
|
|
if activeRemoteRequestResult.ErrorResponse != nil {
|
|
|
|
return nil, activeRemoteRequestResult.ErrorResponse
|
2017-05-31 15:56:11 +00:00
|
|
|
}
|
|
|
|
|
2017-06-01 10:32:15 +00:00
|
|
|
if activeRemoteRequestResult.MediaMetadata == nil {
|
2017-05-31 15:56:11 +00:00
|
|
|
return nil, &util.JSONResponse{
|
|
|
|
Code: 404,
|
|
|
|
JSON: jsonerror.NotFound("File not found."),
|
|
|
|
}
|
|
|
|
}
|
2017-06-01 10:32:15 +00:00
|
|
|
|
|
|
|
return activeRemoteRequestResult.MediaMetadata, nil
|
2017-05-31 15:56:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// No active remote request so create one
|
|
|
|
activeRemoteRequests.MXCToResult[mxcURL] = &types.RemoteRequestResult{
|
|
|
|
Cond: &sync.Cond{L: activeRemoteRequests},
|
|
|
|
}
|
2017-06-01 10:32:15 +00:00
|
|
|
|
2017-05-31 15:56:11 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2017-06-01 10:32:15 +00:00
|
|
|
// broadcastMediaMetadata broadcasts the media metadata and error response to waiting goroutines
|
2017-05-31 15:56:11 +00:00
|
|
|
// Only the owner of the activeRemoteRequestResult for this origin and media ID should call this function.
|
2017-06-01 10:32:15 +00:00
|
|
|
func (r *downloadRequest) broadcastMediaMetadata(activeRemoteRequests *types.ActiveRemoteRequests, errorResponse *util.JSONResponse) {
|
|
|
|
activeRemoteRequests.Lock()
|
|
|
|
defer activeRemoteRequests.Unlock()
|
|
|
|
mxcURL := "mxc://" + string(r.MediaMetadata.Origin) + "/" + string(r.MediaMetadata.MediaID)
|
|
|
|
if activeRemoteRequestResult, ok := activeRemoteRequests.MXCToResult[mxcURL]; ok {
|
|
|
|
r.Logger.Info("Signalling other goroutines waiting for this goroutine to fetch the file.")
|
|
|
|
activeRemoteRequestResult.MediaMetadata = r.MediaMetadata
|
|
|
|
activeRemoteRequestResult.ErrorResponse = errorResponse
|
|
|
|
activeRemoteRequestResult.Cond.Broadcast()
|
|
|
|
}
|
|
|
|
delete(activeRemoteRequests.MXCToResult, mxcURL)
|
|
|
|
}
|
2017-05-31 15:56:11 +00:00
|
|
|
|
2017-06-01 10:32:15 +00:00
|
|
|
// fetchRemoteFileAndStoreMetadata fetches the file from the remote server and stores its metadata in the database
|
2017-06-19 14:21:04 +00:00
|
|
|
func (r *downloadRequest) fetchRemoteFileAndStoreMetadata(absBasePath config.Path, maxFileSizeBytes config.FileSizeBytes, db *storage.Database, thumbnailSizes []config.ThumbnailSize, activeThumbnailGeneration *types.ActiveThumbnailGeneration, maxThumbnailGenerators int) *util.JSONResponse {
|
2017-05-31 15:56:11 +00:00
|
|
|
finalPath, duplicate, resErr := r.fetchRemoteFile(absBasePath, maxFileSizeBytes)
|
|
|
|
if resErr != nil {
|
|
|
|
return resErr
|
|
|
|
}
|
|
|
|
|
|
|
|
r.Logger.WithFields(log.Fields{
|
|
|
|
"Base64Hash": r.MediaMetadata.Base64Hash,
|
|
|
|
"UploadName": r.MediaMetadata.UploadName,
|
|
|
|
"FileSizeBytes": r.MediaMetadata.FileSizeBytes,
|
2017-06-06 23:12:49 +00:00
|
|
|
"ContentType": r.MediaMetadata.ContentType,
|
2017-05-31 15:56:11 +00:00
|
|
|
}).Info("Storing file metadata to media repository database")
|
|
|
|
|
|
|
|
// FIXME: timeout db request
|
|
|
|
if err := db.StoreMediaMetadata(r.MediaMetadata); err != nil {
|
|
|
|
// If the file is a duplicate (has the same hash as an existing file) then
|
|
|
|
// there is valid metadata in the database for that file. As such we only
|
|
|
|
// remove the file if it is not a duplicate.
|
|
|
|
if duplicate == false {
|
|
|
|
finalDir := filepath.Dir(string(finalPath))
|
|
|
|
fileutils.RemoveDir(types.Path(finalDir), r.Logger)
|
|
|
|
}
|
|
|
|
// NOTE: It should really not be possible to fail the uniqueness test here so
|
|
|
|
// there is no need to handle that separately
|
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return &resErr
|
|
|
|
}
|
|
|
|
|
2017-06-06 23:12:49 +00:00
|
|
|
go func() {
|
|
|
|
busy, err := thumbnailer.GenerateThumbnails(finalPath, thumbnailSizes, r.MediaMetadata, activeThumbnailGeneration, maxThumbnailGenerators, db, r.Logger)
|
|
|
|
if err != nil {
|
|
|
|
r.Logger.WithError(err).Warn("Error generating thumbnails")
|
|
|
|
}
|
|
|
|
if busy {
|
|
|
|
r.Logger.Warn("Maximum number of active thumbnail generators reached. Skipping pre-generation.")
|
|
|
|
}
|
|
|
|
}()
|
2017-05-31 15:56:11 +00:00
|
|
|
|
|
|
|
r.Logger.WithFields(log.Fields{
|
|
|
|
"UploadName": r.MediaMetadata.UploadName,
|
|
|
|
"Base64Hash": r.MediaMetadata.Base64Hash,
|
|
|
|
"FileSizeBytes": r.MediaMetadata.FileSizeBytes,
|
2017-06-06 23:12:49 +00:00
|
|
|
"ContentType": r.MediaMetadata.ContentType,
|
2017-05-31 15:56:11 +00:00
|
|
|
}).Infof("Remote file cached")
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-06-19 14:21:04 +00:00
|
|
|
func (r *downloadRequest) fetchRemoteFile(absBasePath config.Path, maxFileSizeBytes config.FileSizeBytes) (types.Path, bool, *util.JSONResponse) {
|
2017-05-31 15:56:11 +00:00
|
|
|
r.Logger.Info("Fetching remote file")
|
|
|
|
|
|
|
|
// create request for remote file
|
|
|
|
resp, resErr := r.createRemoteRequest()
|
|
|
|
if resErr != nil {
|
|
|
|
return "", false, resErr
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
// get metadata from request and set metadata on response
|
|
|
|
contentLength, err := strconv.ParseInt(resp.Header.Get("Content-Length"), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
r.Logger.WithError(err).Warn("Failed to parse content length")
|
|
|
|
return "", false, &util.JSONResponse{
|
|
|
|
Code: 502,
|
|
|
|
JSON: jsonerror.Unknown("Invalid response from remote server"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if contentLength > int64(maxFileSizeBytes) {
|
|
|
|
return "", false, &util.JSONResponse{
|
|
|
|
Code: 413,
|
|
|
|
JSON: jsonerror.Unknown(fmt.Sprintf("Remote file is too large (%v > %v bytes)", contentLength, maxFileSizeBytes)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
r.MediaMetadata.FileSizeBytes = types.FileSizeBytes(contentLength)
|
|
|
|
r.MediaMetadata.ContentType = types.ContentType(resp.Header.Get("Content-Type"))
|
2017-06-01 14:04:41 +00:00
|
|
|
_, params, err := mime.ParseMediaType(resp.Header.Get("Content-Disposition"))
|
|
|
|
if err == nil && params["filename"] != "" {
|
|
|
|
r.MediaMetadata.UploadName = types.Filename(params["filename"])
|
|
|
|
}
|
2017-05-31 15:56:11 +00:00
|
|
|
|
|
|
|
r.Logger.Info("Transferring remote file")
|
|
|
|
|
|
|
|
// The file data is hashed but is NOT used as the MediaID, unlike in Upload. The hash is useful as a
|
|
|
|
// method of deduplicating files to save storage, as well as a way to conduct
|
|
|
|
// integrity checks on the file data in the repository.
|
|
|
|
// Data is truncated to maxFileSizeBytes. Content-Length was reported as 0 < Content-Length <= maxFileSizeBytes so this is OK.
|
|
|
|
hash, bytesWritten, tmpDir, err := fileutils.WriteTempFile(resp.Body, maxFileSizeBytes, absBasePath)
|
|
|
|
if err != nil {
|
|
|
|
r.Logger.WithError(err).WithFields(log.Fields{
|
|
|
|
"MaxFileSizeBytes": maxFileSizeBytes,
|
|
|
|
}).Warn("Error while downloading file from remote server")
|
|
|
|
fileutils.RemoveDir(tmpDir, r.Logger)
|
|
|
|
return "", false, &util.JSONResponse{
|
|
|
|
Code: 502,
|
|
|
|
JSON: jsonerror.Unknown("File could not be downloaded from remote server"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
r.Logger.Info("Remote file transferred")
|
|
|
|
|
|
|
|
// It's possible the bytesWritten to the temporary file is different to the reported Content-Length from the remote
|
|
|
|
// request's response. bytesWritten is therefore used as it is what would be sent to clients when reading from the local
|
|
|
|
// file.
|
|
|
|
r.MediaMetadata.FileSizeBytes = types.FileSizeBytes(bytesWritten)
|
|
|
|
r.MediaMetadata.Base64Hash = hash
|
|
|
|
|
|
|
|
// The database is the source of truth so we need to have moved the file first
|
|
|
|
finalPath, duplicate, err := fileutils.MoveFileWithHashCheck(tmpDir, r.MediaMetadata, absBasePath, r.Logger)
|
|
|
|
if err != nil {
|
|
|
|
r.Logger.WithError(err).Error("Failed to move file.")
|
|
|
|
resErr := jsonerror.InternalServerError()
|
|
|
|
return "", false, &resErr
|
|
|
|
}
|
|
|
|
if duplicate {
|
|
|
|
r.Logger.WithField("dst", finalPath).Info("File was stored previously - discarding duplicate")
|
|
|
|
// Continue on to store the metadata in the database
|
|
|
|
}
|
|
|
|
|
|
|
|
return types.Path(finalPath), duplicate, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *downloadRequest) createRemoteRequest() (*http.Response, *util.JSONResponse) {
|
2017-06-01 15:57:05 +00:00
|
|
|
matrixClient := gomatrixserverlib.NewClient()
|
2017-05-31 15:56:11 +00:00
|
|
|
|
2017-06-01 15:57:05 +00:00
|
|
|
resp, err := matrixClient.CreateMediaDownloadRequest(r.MediaMetadata.Origin, string(r.MediaMetadata.MediaID))
|
2017-05-31 15:56:11 +00:00
|
|
|
if err != nil {
|
2017-06-01 15:57:05 +00:00
|
|
|
r.Logger.WithError(err).Error("Failed to create download request")
|
2017-05-31 15:56:11 +00:00
|
|
|
return nil, &util.JSONResponse{
|
|
|
|
Code: 502,
|
|
|
|
JSON: jsonerror.Unknown(fmt.Sprintf("File with media ID %q could not be downloaded from %q", r.MediaMetadata.MediaID, r.MediaMetadata.Origin)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
if resp.StatusCode == 404 {
|
|
|
|
return nil, &util.JSONResponse{
|
|
|
|
Code: 404,
|
|
|
|
JSON: jsonerror.NotFound(fmt.Sprintf("File with media ID %q does not exist", r.MediaMetadata.MediaID)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
r.Logger.WithFields(log.Fields{
|
|
|
|
"StatusCode": resp.StatusCode,
|
|
|
|
}).Warn("Received error response")
|
|
|
|
return nil, &util.JSONResponse{
|
|
|
|
Code: 502,
|
|
|
|
JSON: jsonerror.Unknown(fmt.Sprintf("File with media ID %q could not be downloaded from %q", r.MediaMetadata.MediaID, r.MediaMetadata.Origin)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return resp, nil
|
|
|
|
}
|