summaryrefslogtreecommitdiff
path: root/workhorse/internal/upload/uploads.go
diff options
context:
space:
mode:
Diffstat (limited to 'workhorse/internal/upload/uploads.go')
-rw-r--r--workhorse/internal/upload/uploads.go66
1 files changed, 66 insertions, 0 deletions
diff --git a/workhorse/internal/upload/uploads.go b/workhorse/internal/upload/uploads.go
new file mode 100644
index 00000000000..3be39f9518f
--- /dev/null
+++ b/workhorse/internal/upload/uploads.go
@@ -0,0 +1,66 @@
+package upload
+
+import (
+ "bytes"
+ "context"
+ "fmt"
+ "io/ioutil"
+ "mime/multipart"
+ "net/http"
+
+ "gitlab.com/gitlab-org/gitlab-workhorse/internal/api"
+ "gitlab.com/gitlab-org/gitlab-workhorse/internal/filestore"
+ "gitlab.com/gitlab-org/gitlab-workhorse/internal/helper"
+ "gitlab.com/gitlab-org/gitlab-workhorse/internal/upload/exif"
+ "gitlab.com/gitlab-org/gitlab-workhorse/internal/zipartifacts"
+)
+
+// These methods are allowed to have thread-unsafe implementations.
+type MultipartFormProcessor interface {
+ ProcessFile(ctx context.Context, formName string, file *filestore.FileHandler, writer *multipart.Writer) error
+ ProcessField(ctx context.Context, formName string, writer *multipart.Writer) error
+ Finalize(ctx context.Context) error
+ Name() string
+}
+
+func HandleFileUploads(w http.ResponseWriter, r *http.Request, h http.Handler, preauth *api.Response, filter MultipartFormProcessor, opts *filestore.SaveFileOpts) {
+ var body bytes.Buffer
+ writer := multipart.NewWriter(&body)
+ defer writer.Close()
+
+ // Rewrite multipart form data
+ err := rewriteFormFilesFromMultipart(r, writer, preauth, filter, opts)
+ if err != nil {
+ switch err {
+ case ErrInjectedClientParam:
+ helper.CaptureAndFail(w, r, err, "Bad Request", http.StatusBadRequest)
+ case http.ErrNotMultipart:
+ h.ServeHTTP(w, r)
+ case filestore.ErrEntityTooLarge:
+ helper.RequestEntityTooLarge(w, r, err)
+ case zipartifacts.ErrBadMetadata:
+ helper.RequestEntityTooLarge(w, r, err)
+ case exif.ErrRemovingExif:
+ helper.CaptureAndFail(w, r, err, "Failed to process image", http.StatusUnprocessableEntity)
+ default:
+ helper.Fail500(w, r, fmt.Errorf("handleFileUploads: extract files from multipart: %v", err))
+ }
+ return
+ }
+
+ // Close writer
+ writer.Close()
+
+ // Hijack the request
+ r.Body = ioutil.NopCloser(&body)
+ r.ContentLength = int64(body.Len())
+ r.Header.Set("Content-Type", writer.FormDataContentType())
+
+ if err := filter.Finalize(r.Context()); err != nil {
+ helper.Fail500(w, r, fmt.Errorf("handleFileUploads: Finalize: %v", err))
+ return
+ }
+
+ // Proxy the request
+ h.ServeHTTP(w, r)
+}