Browse Source
It turns out that the suckiness of Zuul with multipart requests comes almost entirely from the Multipart handling in Spring's DispatcherServlet. This change makes the proxy routes available on an alternative path /zuul/<normal_path> (where /zuul is the default value of zuul.servletPath). I have tested those with 800MB file uploads using the main method in the FormZuulServletProxyApplicationTests and the main observation is that there is no OutOfMemory error (no-one tries to download the complete request body). It works with Ribbon and with the simple (HttpClient) filter. With Ribbon you will need to set some timeouts if you want to upload files as large as that, e.g. see application.yml in the tests: hystrix.command.default.execution.isolation.thread.timeoutInMilliseconds: 60000 ribbon: ConnectTimeout: 3000 ReadTimeout: 60000 You need to set "Transfer-Encoding: chunked" in the incoming request. Chrome does not do this by default apparently, but I was able to test with curl, e.g. $ curl -v -H "Transfer-Encoding: chunked" \ -F "file=@mylarg.iso" \ localhost:9999/zuul/direct/file The old proxy paths through the DispatcherServlet are still available (for backwards compatibility and for convenience of having the paths available at the root of the context path). Fixes gh-254pull/6/head
Dave Syer
10 years ago
8 changed files with 322 additions and 6 deletions
@ -0,0 +1,237 @@
@@ -0,0 +1,237 @@
|
||||
/* |
||||
* Copyright 2013-2015 the original author or authors. |
||||
* |
||||
* Licensed under the Apache License, Version 2.0 (the "License"); |
||||
* you may not use this file except in compliance with the License. |
||||
* You may obtain a copy of the License at |
||||
* |
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
* |
||||
* Unless required by applicable law or agreed to in writing, software |
||||
* distributed under the License is distributed on an "AS IS" BASIS, |
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||
* See the License for the specific language governing permissions and |
||||
* limitations under the License. |
||||
*/ |
||||
|
||||
package org.springframework.cloud.netflix.zuul; |
||||
|
||||
import java.io.IOException; |
||||
import java.io.InputStream; |
||||
import java.util.Arrays; |
||||
import java.util.Map; |
||||
|
||||
import lombok.extern.slf4j.Slf4j; |
||||
|
||||
import org.junit.Test; |
||||
import org.junit.runner.RunWith; |
||||
import org.springframework.beans.factory.annotation.Value; |
||||
import org.springframework.boot.actuate.trace.InMemoryTraceRepository; |
||||
import org.springframework.boot.actuate.trace.TraceRepository; |
||||
import org.springframework.boot.autoconfigure.EnableAutoConfiguration; |
||||
import org.springframework.boot.builder.SpringApplicationBuilder; |
||||
import org.springframework.boot.test.IntegrationTest; |
||||
import org.springframework.boot.test.SpringApplicationConfiguration; |
||||
import org.springframework.boot.test.TestRestTemplate; |
||||
import org.springframework.cloud.netflix.ribbon.RibbonClient; |
||||
import org.springframework.cloud.netflix.ribbon.RibbonClients; |
||||
import org.springframework.context.annotation.Bean; |
||||
import org.springframework.context.annotation.Configuration; |
||||
import org.springframework.http.HttpEntity; |
||||
import org.springframework.http.HttpHeaders; |
||||
import org.springframework.http.HttpMethod; |
||||
import org.springframework.http.HttpStatus; |
||||
import org.springframework.http.MediaType; |
||||
import org.springframework.http.ResponseEntity; |
||||
import org.springframework.test.annotation.DirtiesContext; |
||||
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; |
||||
import org.springframework.test.context.web.WebAppConfiguration; |
||||
import org.springframework.util.LinkedMultiValueMap; |
||||
import org.springframework.util.MultiValueMap; |
||||
import org.springframework.web.bind.annotation.RequestMapping; |
||||
import org.springframework.web.bind.annotation.RequestMethod; |
||||
import org.springframework.web.bind.annotation.RequestParam; |
||||
import org.springframework.web.bind.annotation.RestController; |
||||
import org.springframework.web.multipart.MultipartFile; |
||||
|
||||
import com.netflix.appinfo.EurekaInstanceConfig; |
||||
import com.netflix.loadbalancer.BaseLoadBalancer; |
||||
import com.netflix.loadbalancer.ILoadBalancer; |
||||
import com.netflix.loadbalancer.Server; |
||||
import com.netflix.zuul.ZuulFilter; |
||||
|
||||
import static org.junit.Assert.assertEquals; |
||||
|
||||
@RunWith(SpringJUnit4ClassRunner.class) |
||||
@SpringApplicationConfiguration(classes = FormZuulServletProxyApplication.class) |
||||
@WebAppConfiguration |
||||
@IntegrationTest({ "server.port:0", "zuul.routes.simple:/zuul/simple/**" }) |
||||
@DirtiesContext |
||||
public class FormZuulServletProxyApplicationTests { |
||||
|
||||
@Value("${local.server.port}") |
||||
private int port; |
||||
|
||||
@Test |
||||
public void postWithForm() { |
||||
MultiValueMap<String, String> form = new LinkedMultiValueMap<String, String>(); |
||||
form.set("foo", "bar"); |
||||
HttpHeaders headers = new HttpHeaders(); |
||||
headers.setContentType(MediaType.APPLICATION_FORM_URLENCODED); |
||||
ResponseEntity<String> result = new TestRestTemplate().exchange( |
||||
"http://localhost:" + this.port + "/zuul/simple/form", HttpMethod.POST, |
||||
new HttpEntity<MultiValueMap<String, String>>(form, headers), |
||||
String.class); |
||||
assertEquals(HttpStatus.OK, result.getStatusCode()); |
||||
assertEquals("Posted! {foo=[bar]}", result.getBody()); |
||||
} |
||||
|
||||
@Test |
||||
public void postWithMultipartForm() { |
||||
MultiValueMap<String, String> form = new LinkedMultiValueMap<String, String>(); |
||||
form.set("foo", "bar"); |
||||
HttpHeaders headers = new HttpHeaders(); |
||||
headers.setContentType(MediaType.MULTIPART_FORM_DATA); |
||||
ResponseEntity<String> result = new TestRestTemplate().exchange( |
||||
"http://localhost:" + this.port + "/zuul/simple/form", HttpMethod.POST, |
||||
new HttpEntity<MultiValueMap<String, String>>(form, headers), |
||||
String.class); |
||||
assertEquals(HttpStatus.OK, result.getStatusCode()); |
||||
assertEquals("Posted! {foo=[bar]}", result.getBody()); |
||||
} |
||||
|
||||
@Test |
||||
public void postWithMultipartFile() { |
||||
MultiValueMap<String, Object> form = new LinkedMultiValueMap<String, Object>(); |
||||
HttpHeaders part = new HttpHeaders(); |
||||
part.setContentType(MediaType.TEXT_PLAIN); |
||||
part.setContentDispositionFormData("file", "foo.txt"); |
||||
form.set("foo", new HttpEntity<byte[]>("bar".getBytes(), part)); |
||||
HttpHeaders headers = new HttpHeaders(); |
||||
headers.setContentType(MediaType.MULTIPART_FORM_DATA); |
||||
ResponseEntity<String> result = new TestRestTemplate().exchange( |
||||
"http://localhost:" + this.port + "/zuul/simple/file", HttpMethod.POST, |
||||
new HttpEntity<MultiValueMap<String, Object>>(form, headers), |
||||
String.class); |
||||
assertEquals(HttpStatus.OK, result.getStatusCode()); |
||||
assertEquals("Posted! bar", result.getBody()); |
||||
} |
||||
|
||||
@Test |
||||
public void postWithUTF8Form() { |
||||
MultiValueMap<String, String> form = new LinkedMultiValueMap<String, String>(); |
||||
form.set("foo", "bar"); |
||||
HttpHeaders headers = new HttpHeaders(); |
||||
headers.setContentType(MediaType |
||||
.valueOf(MediaType.APPLICATION_FORM_URLENCODED_VALUE + "; charset=UTF-8")); |
||||
ResponseEntity<String> result = new TestRestTemplate().exchange( |
||||
"http://localhost:" + this.port + "/zuul/simple/form", HttpMethod.POST, |
||||
new HttpEntity<MultiValueMap<String, String>>(form, headers), |
||||
String.class); |
||||
assertEquals(HttpStatus.OK, result.getStatusCode()); |
||||
assertEquals("Posted! {foo=[bar]}", result.getBody()); |
||||
} |
||||
} |
||||
|
||||
// Don't use @SpringBootApplication because we don't want to component scan
|
||||
@Configuration |
||||
@EnableAutoConfiguration |
||||
@RestController |
||||
@EnableZuulProxy |
||||
@RibbonClients(@RibbonClient(name = "simple", configuration = ServletFormRibbonClientConfiguration.class)) |
||||
@Slf4j |
||||
class FormZuulServletProxyApplication { |
||||
|
||||
@RequestMapping(value = "/form", method = RequestMethod.POST) |
||||
public String accept(@RequestParam MultiValueMap<String, String> form) |
||||
throws IOException { |
||||
return "Posted! " + form; |
||||
} |
||||
|
||||
// TODO: Why does this not work if you add @RequestParam as above?
|
||||
@RequestMapping(value = "/file", method = RequestMethod.POST) |
||||
public String file(@RequestParam(required = false) MultipartFile file) |
||||
throws IOException { |
||||
byte[] bytes = new byte[0]; |
||||
if (file != null) { |
||||
if (file.getSize() > 1024) { |
||||
bytes = new byte[1024]; |
||||
InputStream inputStream = file.getInputStream(); |
||||
inputStream.read(bytes); |
||||
byte[] buffer = new byte[1024 * 1024 * 10]; |
||||
while (inputStream.read(buffer) >= 0) { |
||||
log.info("Read more bytes"); |
||||
} |
||||
} |
||||
else { |
||||
bytes = file.getBytes(); |
||||
} |
||||
} |
||||
return "Posted! " + new String(bytes); |
||||
} |
||||
|
||||
@Bean |
||||
public ZuulFilter sampleFilter() { |
||||
return new ZuulFilter() { |
||||
|
||||
@Override |
||||
public String filterType() { |
||||
return "pre"; |
||||
} |
||||
|
||||
@Override |
||||
public boolean shouldFilter() { |
||||
return true; |
||||
} |
||||
|
||||
@Override |
||||
public Object run() { |
||||
return null; |
||||
} |
||||
|
||||
@Override |
||||
public int filterOrder() { |
||||
return 0; |
||||
} |
||||
|
||||
}; |
||||
} |
||||
|
||||
@Bean |
||||
public TraceRepository traceRepository() { |
||||
return new InMemoryTraceRepository() { |
||||
@Override |
||||
public void add(Map<String, Object> map) { |
||||
if (map.containsKey("body")) { |
||||
map.get("body"); |
||||
} |
||||
super.add(map); |
||||
} |
||||
}; |
||||
} |
||||
|
||||
public static void main(String[] args) { |
||||
new SpringApplicationBuilder(FormZuulProxyApplication.class).properties( |
||||
"zuul.routes.simple:/zuul/simple/**", |
||||
"zuul.routes.direct.url:http://localhost:9999", |
||||
"zuul.routes.direct.path:/zuul/direct/**", |
||||
"multipart.maxFileSize:4096MB", "multipart.maxRequestSize:4096MB").run( |
||||
args); |
||||
} |
||||
|
||||
} |
||||
|
||||
// Load balancer with fixed server list for "simple" pointing to localhost
|
||||
@Configuration |
||||
class ServletFormRibbonClientConfiguration { |
||||
|
||||
@Bean |
||||
public ILoadBalancer ribbonLoadBalancer(EurekaInstanceConfig instance) { |
||||
BaseLoadBalancer balancer = new BaseLoadBalancer(); |
||||
balancer.setServersList(Arrays.asList(new Server("localhost", instance |
||||
.getNonSecurePort()))); |
||||
// balancer.setServersList(Arrays.asList(new Server("localhost", 8000)));
|
||||
return balancer; |
||||
} |
||||
|
||||
} |
Loading…
Reference in new issue