aboutsummaryrefslogtreecommitdiffstats
path: root/controller-server/src/main/java/com/yahoo/vespa/hosted/controller/restapi/application/MultipartParser.java
blob: a28f0e9733d80c4c69c987ab84b923d10a01d954 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
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
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
// Copyright Yahoo. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
package com.yahoo.vespa.hosted.controller.restapi.application;

import com.yahoo.container.jdisc.HttpRequest;
import com.yahoo.vespa.hosted.controller.api.application.v4.EnvironmentResource;
import org.apache.commons.fileupload.MultipartStream;
import org.apache.commons.fileupload.ParameterParser;
import org.apache.commons.fileupload.util.Streams;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.util.HashMap;
import java.util.Map;

/**
 * Provides reading a multipart/form-data request type into a map of bytes for each part,
 * indexed by the parts (form field) name.
 * 
 * @author bratseth
 */
public class MultipartParser {

    private final long maxDataLength;

    public MultipartParser() {
        this(2 * (long) Math.pow(1024, 3)); // 2 GB
    }

    MultipartParser(long maxDataLength) {
        this.maxDataLength = maxDataLength;
    }

    /**
     * Parses the given multipart request and returns all the parts indexed by their name.
     * 
     * @throws IllegalArgumentException if this request is not a well-formed request with Content-Type multipart/form-data
     */
    public Map<String, byte[]> parse(HttpRequest request) {
        return parse(request.getHeader("Content-Type"), request.getData(), request.getUri());
    }

    /**
     * Parses the given data stream for the given uri using the provided content-type header to determine boundaries.
     *
     * @throws IllegalArgumentException if this is not a well-formed request with Content-Type multipart/form-data
     */
    public Map<String, byte[]> parse(String contentTypeHeader, InputStream data, URI uri) {
        try {
            LimitedOutputStream output = new LimitedOutputStream(maxDataLength);
            ParameterParser parameterParser = new ParameterParser();
            Map<String, String> contentType = parameterParser.parse(contentTypeHeader, ';');
            if (contentType.containsKey("application/zip")) {
                Streams.copy(data, output, false);
                return Map.of(EnvironmentResource.APPLICATION_ZIP, output.toByteArray());
            }
            if ( ! contentType.containsKey("multipart/form-data"))
                throw new IllegalArgumentException("Expected a multipart or application/zip message, but got Content-Type: " + contentTypeHeader);
            String boundary = contentType.get("boundary");
            if (boundary == null)
                throw new IllegalArgumentException("Missing boundary property in Content-Type header");
            MultipartStream multipartStream = new MultipartStream(data, boundary.getBytes(), 1 << 20, null);
            boolean nextPart = multipartStream.skipPreamble();
            Map<String, byte[]> parts = new HashMap<>();
            while (nextPart) {
                String[] headers = multipartStream.readHeaders().split("\r\n");
                String contentDispositionContent = findContentDispositionHeader(headers);
                if (contentDispositionContent == null)
                    throw new IllegalArgumentException("Missing Content-Disposition header in a multipart body part");
                Map<String, String> contentDisposition = parameterParser.parse(contentDispositionContent, ';');
                multipartStream.readBodyData(output);
                parts.put(contentDisposition.get("name"), output.toByteArray());
                output.reset();
                nextPart = multipartStream.readBoundary();
            }
            return parts;
        }
        catch (MultipartStream.MalformedStreamException e) {
            throw new IllegalArgumentException("Malformed multipart/form-data request", e);
        } 
        catch (IOException e) {
            throw new IllegalArgumentException("IO error reading multipart request " + uri, e);
        }
    }
    
    private String findContentDispositionHeader(String[] headers) {
        String contentDisposition = "Content-Disposition:";
        for (String header : headers) {
            if (header.length() < contentDisposition.length()) continue;
            if ( ! header.substring(0, contentDisposition.length()).equalsIgnoreCase(contentDisposition)) continue;
            return header.substring(contentDisposition.length() + 1);
        }
        return null;
    }

    /** A {@link java.io.ByteArrayOutputStream} that limits the number of bytes written to it */
    private static class LimitedOutputStream extends ByteArrayOutputStream {

        private long remaining;

        /** Create a new OutputStream that can fit up to len bytes */
        private LimitedOutputStream(long len) {
            this.remaining = len;
        }

        @Override
        public synchronized void write(int b) {
            requireCapacity(1);
            super.write(b);
            remaining--;
        }

        @Override
        public synchronized void write(byte[] b, int off, int len) {
            requireCapacity(len);
            super.write(b, off, len);
            remaining -= len;
        }

        private void requireCapacity(int len) {
            if (len > remaining) throw new IllegalArgumentException("Too many bytes to write");
        }

    }

}