/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 2010-2011 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* http://glassfish.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
package com.sun.jersey.multipart.impl;
import com.sun.jersey.multipart.BodyPart;
import com.sun.jersey.multipart.BodyPartEntity;
import com.sun.jersey.multipart.MultiPart;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import javax.ws.rs.Consumes;
import javax.ws.rs.GET;
import javax.ws.rs.POST;
import javax.ws.rs.PUT;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
/**
* <p>Resource file for {@link MultiPartReaderWriterTest}.</p>
*/
@Path("/multipart")
public class MultiPartResource {
@Path("zero")
@GET
@Produces("text/plain")
public String zero() {
return "Hello, world\r\n";
}
@Path("one")
@GET
@Produces("multipart/mixed")
public Response one() {
MultiPart entity = new MultiPart();
// Exercise manually adding part(s) to the bodyParts property
BodyPart part = new BodyPart("This is the only segment", new MediaType("text", "plain"));
entity.getBodyParts().add(part);
return Response.ok(entity).type("multipart/mixed").build();
}
@Path("two")
@GET
@Produces("multipart/mixed")
public Response two() {
// Exercise builder pattern with default content type
return Response.ok(new MultiPart().
bodyPart("This is the first segment", new MediaType("text", "plain")).
bodyPart("<outer><inner>value</inner></outer>", new MediaType("text", "xml"))).build();
}
@Path("three")
@GET
@Produces("multipart/mixed")
public Response three() {
// Exercise builder pattern with explicit content type
MultiPartBean bean = new MultiPartBean("myname", "myvalue");
return Response.ok(new MultiPart().
type(new MediaType("multipart", "mixed")).
bodyPart("This is the first segment", new MediaType("text", "plain")).
bodyPart(bean, new MediaType("x-application", "x-format"))).build();
}
@Path("four")
@PUT
@Produces("text/plain")
public Response four(MultiPart multiPart) throws IOException {
if (!(multiPart.getBodyParts().size() == 2)) {
return Response.ok("FAILED: Number of body parts is " + multiPart.getBodyParts().size() + " instead of 2").build();
}
BodyPart part0 = multiPart.getBodyParts().get(0);
if (!(part0.getMediaType().equals(new MediaType("text", "plain")))) {
return Response.ok("FAILED: First media type is " + part0.getMediaType()).build();
}
BodyPart part1 = multiPart.getBodyParts().get(1);
if (!(part1.getMediaType().equals(new MediaType("x-application", "x-format")))) {
return Response.ok("FAILED: Second media type is " + part1.getMediaType()).build();
}
BodyPartEntity bpe = (BodyPartEntity) part0.getEntity();
StringBuilder sb = new StringBuilder();
InputStream stream = bpe.getInputStream();
InputStreamReader reader = new InputStreamReader(stream);
char[] buffer = new char[2048];
while (true) {
int n = reader.read(buffer);
if (n < 0) {
break;
}
sb.append(buffer, 0, n);
}
if (!sb.toString().equals("This is the first segment")) {
return Response.ok("FAILED: First part name = " + sb.toString()).build();
}
MultiPartBean bean = part1.getEntityAs(MultiPartBean.class);
if (!bean.getName().equals("myname")) {
return Response.ok("FAILED: Second part name = " + bean.getName()).build();
}
if (!bean.getValue().equals("myvalue")) {
return Response.ok("FAILED: Second part value = " + bean.getValue()).build();
}
return Response.ok("SUCCESS: All tests passed").build();
}
// Note - this should never actually get reached, because the client
// is trying to post a MultiPart with no body parts inside, and that
// should throw a client side exception
@Path("six")
@POST
@Consumes("multipart/mixed")
@Produces("text/plain")
public Response six(MultiPart multiPart) {
String response = "All OK";
if (!"multipart".equals(multiPart.getMediaType().getType()) ||
!"mixed".equals(multiPart.getMediaType().getSubtype())) {
response = "MultiPart media type is " + multiPart.getMediaType().toString();
} else if (multiPart.getBodyParts().size() != 0) {
response = "Got " + multiPart.getBodyParts().size() + " body parts instead of zero";
}
return Response.ok(response).build();
}
@Path("ten")
@PUT
@Consumes("multipart/mixed")
@Produces("text/plain")
public Response ten(MultiPart mp) {
if (!(mp.getBodyParts().size() == 2)) {
return Response.ok("FAILED: Body part count is " + mp.getBodyParts().size() + " instead of 2").build();
} else if (!(mp.getBodyParts().get(1).getEntity() instanceof BodyPartEntity)) {
return Response.ok("FAILED: Second body part is " + mp.getBodyParts().get(1).getClass().getName() + " instead of BodyPartEntity").build();
}
BodyPartEntity bpe = (BodyPartEntity) mp.getBodyParts().get(1).getEntity();
try {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
InputStream stream = bpe.getInputStream();
byte[] buffer = new byte[2048];
while (true) {
int n = stream.read(buffer);
if (n < 0) {
break;
}
baos.write(buffer, 0, n);
}
if (baos.toByteArray().length > 0) {
return Response.ok("FAILED: Second body part had " + baos.toByteArray().length + " bytes instead of 0").build();
}
return Response.ok("SUCCESS: All tests passed").build();
} catch (IOException e) {
return Response.ok("FAILED: Threw IOException").build();
}
}
// Echo back a body part whose content may or may not exceed the size
// of the buffer threshold
@Path("eleven")
@PUT
@Consumes("multipart/mixed")
@Produces("multipart/mixed")
public Response eleven(MultiPart multiPart) throws IOException {
BodyPartEntity bpe = (BodyPartEntity) multiPart.getBodyParts().get(0).getEntity();
StringBuilder sb = new StringBuilder();
InputStream stream = bpe.getInputStream();
InputStreamReader reader = new InputStreamReader(stream);
char[] buffer = new char[2048];
while (true) {
int n = reader.read(buffer);
if (n < 0) {
break;
}
sb.append(buffer, 0, n);
}
return Response.ok(new MultiPart().bodyPart(sb.toString(), MediaType.TEXT_PLAIN_TYPE)).
type(new MediaType("multipart", "mixed")).build();
}
// Echo back the multipart that was sent
@Path("twelve")
@PUT
@Consumes("multipart/mixed")
@Produces("multipart/mixed")
public MultiPart twelve(MultiPart multiPart) throws IOException {
return multiPart;
}
// Call clean up explicitly
@Path("thirteen")
@PUT
@Consumes("multipart/mixed")
@Produces("multipart/mixed")
public String thirteen(MultiPart multiPart) throws IOException {
multiPart.cleanup();
return "cleanup";
}
}