multifilereader.go 2.89 KB
Newer Older
1 2 3 4
package http

import (
	"bytes"
5
	"fmt"
6 7
	"io"
	"mime/multipart"
8
	"net/textproto"
9 10 11 12

	cmds "github.com/jbenet/go-ipfs/commands"
)

13 14 15
// MultiFileReader reads from a `commands.File` (which can be a directory of files
// or a regular file) as HTTP multipart encoded data.
// WARNING: Not thread-safe!
16 17 18 19
type MultiFileReader struct {
	io.Reader

	files       cmds.File
20
	currentFile io.Reader
21 22 23
	buf         bytes.Buffer
	mpWriter    *multipart.Writer
	closed      bool
24 25 26 27

	// if true, the data will be type 'multipart/form-data'
	// if false, the data will be type 'multipart/mixed'
	form bool
28 29
}

30 31 32
// NewMultiFileReader constructs a MultiFileReader. `file` can be any `commands.File`.
// If `form` is set to true, the multipart data will have a Content-Type of 'multipart/form-data',
// if `form` is false, the Content-Type will be 'multipart/mixed'.
33
func NewMultiFileReader(file cmds.File, form bool) *MultiFileReader {
34 35
	mfr := &MultiFileReader{
		files: file,
36
		form:  form,
37 38 39 40 41 42 43
	}
	mfr.mpWriter = multipart.NewWriter(&mfr.buf)

	return mfr
}

func (mfr *MultiFileReader) Read(buf []byte) (written int, err error) {
44
	// if we are closed and the buffer is flushed, end reading
45 46 47 48 49 50
	if mfr.closed && mfr.buf.Len() == 0 {
		return 0, io.EOF
	}

	// if the current file isn't set, advance to the next file
	if mfr.currentFile == nil {
51
		file, err := mfr.files.NextFile()
52
		if err == io.EOF {
53 54 55 56 57 58
			mfr.mpWriter.Close()
			mfr.closed = true
		} else if err != nil {
			return 0, err
		}

59
		// handle starting a new file part
60
		if !mfr.closed {
61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86
			if file.IsDirectory() {
				// if file is a directory, create a multifilereader from it
				// (using 'multipart/mixed')
				mfr.currentFile = NewMultiFileReader(file, false)
			} else {
				// otherwise, use the file as a reader to read its contents
				mfr.currentFile = file
			}

			// write the boundary and headers
			header := make(textproto.MIMEHeader)
			if mfr.form {
				contentDisposition := fmt.Sprintf("form-data; name=\"file\"; filename=\"%s\"", file.FileName())
				header.Set("Content-Disposition", contentDisposition)
			} else {
				header.Set("Content-Disposition", fmt.Sprintf("file; filename=\"%s\"", file.FileName()))
			}

			if file.IsDirectory() {
				boundary := mfr.currentFile.(*MultiFileReader).Boundary()
				header.Set("Content-Type", fmt.Sprintf("multipart/mixed; boundary=%s", boundary))
			} else {
				header.Set("Content-Type", "application/octet-stream")
			}

			_, err := mfr.mpWriter.CreatePart(header)
87 88 89 90 91 92
			if err != nil {
				return 0, err
			}
		}
	}

93
	// if the buffer has something in it, read from it
94
	if mfr.buf.Len() > 0 {
95
		return mfr.buf.Read(buf)
96 97
	}

98 99 100
	// otherwise, read from file data
	written, err = mfr.currentFile.Read(buf)
	if err == io.EOF {
101
		mfr.currentFile = nil
102
		return written, nil
103 104 105 106
	}
	return written, err
}

107
// Boundary returns the boundary string to be used to separate files in the multipart data
108 109 110
func (mfr *MultiFileReader) Boundary() string {
	return mfr.mpWriter.Boundary()
}