flamenco/pkg/shaman/fileserver/receivefile.go
Sybren A. Stüvel 4e8e71e4e2 Initial checkin of Shaman of Flamenco 2
This is not yet working, it's just a direct copy of the Manager of Flamenco
2, with Logrus replaced by Zerolog. The API has been documented in
flamenco-manager.yaml as a starting point for the integration.
2022-03-25 14:10:26 +01:00

177 lines
5.7 KiB
Go

/* (c) 2019, Blender Foundation - Sybren A. Stüvel
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, sublicense, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to
* the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
package fileserver
import (
"context"
"fmt"
"io"
"net/http"
"git.blender.org/flamenco/pkg/shaman/filestore"
"git.blender.org/flamenco/pkg/shaman/hasher"
"git.blender.org/flamenco/pkg/shaman/httpserver"
"git.blender.org/flamenco/pkg/shaman/jwtauth"
"github.com/sirupsen/logrus"
)
// receiveFile streams a file from a HTTP request to disk.
func (fs *FileServer) receiveFile(ctx context.Context, w http.ResponseWriter, r *http.Request, checksum string, filesize int64) {
logger := packageLogger.WithFields(jwtauth.RequestLogFields(r))
bodyReader, err := httpserver.DecompressedReader(r)
if err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
defer bodyReader.Close()
originalFilename := r.Header.Get("X-Shaman-Original-Filename")
if originalFilename == "" {
originalFilename = "-not specified-"
}
logger = logger.WithField("originalFilename", originalFilename)
localPath, status := fs.fileStore.ResolveFile(checksum, filesize, filestore.ResolveEverything)
logger = logger.WithField("path", localPath)
if status == filestore.StatusStored {
logger.Info("uploaded file already exists")
w.Header().Set("Location", r.RequestURI)
http.Error(w, "File already stored", http.StatusAlreadyReported)
return
}
if status == filestore.StatusUploading && r.Header.Get("X-Shaman-Can-Defer-Upload") == "true" {
logger.Info("someone is uploading this file and client can defer")
http.Error(w, "File being uploaded, please defer", http.StatusAlreadyReported)
return
}
logger.Info("receiving file")
streamTo, err := fs.fileStore.OpenForUpload(checksum, filesize)
if err != nil {
logger.WithError(err).Error("unable to open file for writing uploaded data")
http.Error(w, "Unable to open file", http.StatusInternalServerError)
return
}
// clean up temporary file if it still exists at function exit.
defer func() {
streamTo.Close()
fs.fileStore.RemoveUploadedFile(streamTo.Name())
}()
// Abort this uploadwhen the file has been finished by someone else.
uploadDone := make(chan struct{})
uploadAlreadyCompleted := false
defer close(uploadDone)
receiverChannel := fs.receiveListenerFor(checksum, filesize)
go func() {
select {
case <-receiverChannel:
case <-uploadDone:
close(receiverChannel)
return
}
logger := logger.WithField("path", localPath)
logger.Info("file was completed during someone else's upload")
uploadAlreadyCompleted = true
err := r.Body.Close()
if err != nil {
logger.WithError(err).Warning("error closing connection")
}
}()
written, actualChecksum, err := hasher.Copy(streamTo, bodyReader)
if err != nil {
if closeErr := streamTo.Close(); closeErr != nil {
logger.WithFields(logrus.Fields{
logrus.ErrorKey: err,
"closeError": closeErr,
}).Error("error closing local file after other I/O error occured")
}
logger = logger.WithError(err)
if uploadAlreadyCompleted {
logger.Debug("aborted upload")
w.Header().Set("Location", r.RequestURI)
http.Error(w, "File already stored", http.StatusAlreadyReported)
} else if err == io.ErrUnexpectedEOF {
logger.Info("unexpected EOF, client probably just disconnected")
} else {
logger.Warning("unable to copy request body to file")
http.Error(w, "I/O error", http.StatusInternalServerError)
}
return
}
if err := streamTo.Close(); err != nil {
logger.WithError(err).Warning("error closing local file")
http.Error(w, "I/O error", http.StatusInternalServerError)
return
}
if written != filesize {
logger.WithFields(logrus.Fields{
"declaredSize": filesize,
"actualSize": written,
}).Warning("mismatch between expected and actual size")
http.Error(w,
fmt.Sprintf("Received %d bytes but you promised %d", written, filesize),
http.StatusExpectationFailed)
return
}
if actualChecksum != checksum {
logger.WithFields(logrus.Fields{
"declaredChecksum": checksum,
"actualChecksum": actualChecksum,
}).Warning("mismatch between expected and actual checksum")
http.Error(w,
"Declared and actual checksums differ",
http.StatusExpectationFailed)
return
}
logger.WithFields(logrus.Fields{
"receivedBytes": written,
"checksum": actualChecksum,
"tempFile": streamTo.Name(),
}).Debug("File received")
if err := fs.fileStore.MoveToStored(checksum, filesize, streamTo.Name()); err != nil {
logger.WithFields(logrus.Fields{
"tempFile": streamTo.Name(),
logrus.ErrorKey: err,
}).Error("unable to move file from 'upload' to 'stored' storage")
http.Error(w,
"unable to move file from 'upload' to 'stored' storage",
http.StatusInternalServerError)
return
}
http.Error(w, "", http.StatusNoContent)
}