Commit 18339264 authored by Alessio Caiazza's avatar Alessio Caiazza

Sticky EOF

parent 900f2dc9
...@@ -6,7 +6,6 @@ import ( ...@@ -6,7 +6,6 @@ import (
"errors" "errors"
"fmt" "fmt"
"io" "io"
"os"
"os/exec" "os/exec"
"regexp" "regexp"
...@@ -16,19 +15,15 @@ import ( ...@@ -16,19 +15,15 @@ import (
var ErrRemovingExif = errors.New("error while removing EXIF") var ErrRemovingExif = errors.New("error while removing EXIF")
type cleaner struct { type cleaner struct {
ctx context.Context ctx context.Context
cmd *exec.Cmd cmd *exec.Cmd
stdout io.Reader stdout io.Reader
stderr bytes.Buffer stderr bytes.Buffer
waitDone chan struct{} eof bool
waitErr error
} }
func NewCleaner(ctx context.Context, stdin io.Reader) (io.Reader, error) { func NewCleaner(ctx context.Context, stdin io.Reader) (io.ReadCloser, error) {
c := &cleaner{ c := &cleaner{ctx: ctx}
ctx: ctx,
waitDone: make(chan struct{}),
}
if err := c.startProcessing(stdin); err != nil { if err := c.startProcessing(stdin); err != nil {
return nil, err return nil, err
...@@ -37,28 +32,32 @@ func NewCleaner(ctx context.Context, stdin io.Reader) (io.Reader, error) { ...@@ -37,28 +32,32 @@ func NewCleaner(ctx context.Context, stdin io.Reader) (io.Reader, error) {
return c, nil return c, nil
} }
func (c *cleaner) Close() error {
if c.cmd == nil {
return nil
}
return c.cmd.Wait()
}
func (c *cleaner) Read(p []byte) (int, error) { func (c *cleaner) Read(p []byte) (int, error) {
if c.eof {
return 0, io.EOF
}
n, err := c.stdout.Read(p) n, err := c.stdout.Read(p)
if err == io.EOF { if err == io.EOF {
if waitErr := c.wait(); waitErr != nil { if waitErr := c.cmd.Wait(); waitErr != nil {
log.WithContextFields(c.ctx, log.Fields{ log.WithContextFields(c.ctx, log.Fields{
"command": c.cmd.Args, "command": c.cmd.Args,
"stderr": c.stderr.String(), "stderr": c.stderr.String(),
"error": waitErr.Error(), "error": waitErr.Error(),
}).Print("exiftool command failed") }).Print("exiftool command failed")
return n, ErrRemovingExif return n, ErrRemovingExif
} }
}
// Calling c.cmd.Wait() will close the stdout pipe, any attempt to read from it will fail with an os.PathError. c.eof = true
// see: https://gitlab.com/gitlab-org/gitlab-workhorse/-/issues/233
if _, ok := err.(*os.PathError); ok {
select {
case <-c.waitDone:
return n, io.EOF
default:
return n, err
}
} }
return n, err return n, err
...@@ -97,19 +96,10 @@ func (c *cleaner) startProcessing(stdin io.Reader) error { ...@@ -97,19 +96,10 @@ func (c *cleaner) startProcessing(stdin io.Reader) error {
if err = c.cmd.Start(); err != nil { if err = c.cmd.Start(); err != nil {
return fmt.Errorf("start %v: %v", c.cmd.Args, err) return fmt.Errorf("start %v: %v", c.cmd.Args, err)
} }
go func() {
c.waitErr = c.cmd.Wait()
close(c.waitDone)
}()
return nil return nil
} }
func (c *cleaner) wait() error {
<-c.waitDone
return c.waitErr
}
func IsExifFile(filename string) bool { func IsExifFile(filename string) bool {
filenameMatch := regexp.MustCompile(`(?i)\.(jpg|jpeg|tiff)$`) filenameMatch := regexp.MustCompile(`(?i)\.(jpg|jpeg|tiff)$`)
......
...@@ -174,12 +174,12 @@ func handleExifUpload(ctx context.Context, r io.Reader, filename string) (io.Rea ...@@ -174,12 +174,12 @@ func handleExifUpload(ctx context.Context, r io.Reader, filename string) (io.Rea
"filename": filename, "filename": filename,
}).Print("running exiftool to remove any metadata") }).Print("running exiftool to remove any metadata")
r, err := exif.NewCleaner(ctx, r) cleaner, err := exif.NewCleaner(ctx, r)
if err != nil { if err != nil {
return nil, err return nil, err
} }
return ioutil.NopCloser(r), nil return cleaner, nil
} }
func handleLsifUpload(ctx context.Context, reader io.Reader, tempPath, filename string, preauth *api.Response) (io.ReadCloser, error) { func handleLsifUpload(ctx context.Context, reader io.Reader, tempPath, filename string, preauth *api.Response) (io.ReadCloser, error) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment