/*
* Copyright (c) 2012, 2020 Oracle and/or its affiliates. All rights reserved.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v. 2.0, which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the
* Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
* version 2 with the GNU Classpath Exception, which is available at
* https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
*/
package org.glassfish.jersey.media.multipart;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import jakarta.ws.rs.core.MediaType;
import org.glassfish.jersey.media.multipart.internal.LocalizationMessages;
import org.glassfish.jersey.message.internal.MediaTypes;
Subclass of MultiPart
with specialized support for media type multipart/form-data
. See RFC 2388 for the formal definition of this media type.
For a server side application wishing to process an incoming multipart/form-data
message, the following features are provided:
- Incoming entities will be of type
FormDataMultiPart
, enabling access to the specialized methods.
- Incoming body parts will be of type
FormDataBodyPart
, enabling access to its specialized methods.
- Convenience method to return the
FormDataBodyPart
for a specified control name.
- Convenience method to return a
Map
of FormDataBodyPart
s for all fields, keyed by field name.
For a client side application wishing to construct an outgoing multipart/form-data
message, the following features are provided:
- Media type of the
FormDataMultiPart
instance will automatically set to multipart/form-data
.
- Builder pattern method to add simple field values as body parts of type
text/plain
.
- Builder pattern method to add named "file" field values with arbitrary media types.
TODO Consider supporting the use case of a nested multipart/mixed
body part to contain multiple uploaded files. Author: Craig McClanahan, Imran M Yousuf (imran at smartitengineering.com), Paul Sandoz, Michal Gajdos
/**
* Subclass of {@link MultiPart} with specialized support for media type {@code multipart/form-data}. See
* <a href="http://www.ietf.org/rfc/rfc2388.txt">RFC 2388</a> for the formal definition of this media type.
* <p/>
* For a server side application wishing to process an incoming {@code multipart/form-data} message, the following features
* are provided:
* <ul>
* <li>Incoming entities will be of type {@link FormDataMultiPart}, enabling access to the specialized methods.</li>
* <li>Incoming body parts will be of type {@link FormDataBodyPart}, enabling access to its specialized methods.</li>
* <li>Convenience method to return the {@link FormDataBodyPart} for a specified control name.</li>
* <li>Convenience method to return a {@code Map} of {@link FormDataBodyPart}s for all fields, keyed by field name.</li>
* </ul>
* <p/>
* For a client side application wishing to construct an outgoing
* {@code multipart/form-data} message, the following features
* are provided:
* <ul>
* <li>Media type of the {@link FormDataMultiPart} instance will automatically set to {@code multipart/form-data}.</li>
* <li>Builder pattern method to add simple field values as body parts of type {@code text/plain}.</li>
* <li>Builder pattern method to add named "file" field values with arbitrary media types.</li>
* </ul>
* <p/>
* TODO Consider supporting the use case of a nested {@code multipart/mixed} body part to contain multiple uploaded files.
*
* @author Craig McClanahan
* @author Imran M Yousuf (imran at smartitengineering.com)
* @author Paul Sandoz
* @author Michal Gajdos
*/
public class FormDataMultiPart extends MultiPart {
Instantiates a new FormDataMultiPart
instance with default characteristics. /**
* Instantiates a new {@code FormDataMultiPart} instance with
* default characteristics.
*/
public FormDataMultiPart() {
super(MediaType.MULTIPART_FORM_DATA_TYPE);
}
Builder pattern method to add a named field with a text value,
and return this instance.
Params: - name – the control name.
- value – the text value.
Returns: this instance.
/**
* Builder pattern method to add a named field with a text value,
* and return this instance.
*
* @param name the control name.
* @param value the text value.
* @return this instance.
*/
public FormDataMultiPart field(String name, String value) {
getBodyParts().add(new FormDataBodyPart(name, value));
return this;
}
Builder pattern method to add a named field with an arbitrary
media type and entity, and return this instance.
Params: - name – the control name.
- entity – entity value for the new field.
- mediaType – media type for the new field.
Returns: this instance.
/**
* Builder pattern method to add a named field with an arbitrary
* media type and entity, and return this instance.
*
* @param name the control name.
* @param entity entity value for the new field.
* @param mediaType media type for the new field.
* @return this instance.
*/
public FormDataMultiPart field(String name, Object entity, MediaType mediaType) {
getBodyParts().add(new FormDataBodyPart(name, entity, mediaType));
return this;
}
Gets a form data body part given a control name.
Params: - name – the control name.
Returns: the form data body part, otherwise null if no part is present with the given control name. If more that one part
is present with the same control name, then the first part that occurs is returned.
/**
* Gets a form data body part given a control name.
*
* @param name the control name.
* @return the form data body part, otherwise null if no part is present with the given control name. If more that one part
* is present with the same control name, then the first part that occurs is returned.
*/
public FormDataBodyPart getField(String name) {
FormDataBodyPart result = null;
for (BodyPart bodyPart : getBodyParts()) {
if (!(bodyPart instanceof FormDataBodyPart)) {
continue;
}
if (name.equals(((FormDataBodyPart) bodyPart).getName())) {
result = (FormDataBodyPart) bodyPart;
break; // Break to return first result found.
}
}
return result;
}
Gets a list of one or more form data body parts given a control name.
Params: - name – the control name.
Returns: the list of form data body parts, otherwise null if no parts are present with the given control name.
/**
* Gets a list of one or more form data body parts given a control name.
*
* @param name the control name.
* @return the list of form data body parts, otherwise null if no parts are present with the given control name.
*/
public List<FormDataBodyPart> getFields(String name) {
List<FormDataBodyPart> result = null;
for (BodyPart bodyPart : getBodyParts()) {
if (!(bodyPart instanceof FormDataBodyPart)) {
continue;
}
if (name.equals(((FormDataBodyPart) bodyPart).getName())) {
if (result == null) {
result = new ArrayList<>(1);
}
result.add((FormDataBodyPart) bodyPart);
}
}
return result;
}
Gets a map of form data body parts where the key is the control name
and the value is a list of one or more form data body parts.
Returns: return the map of form data body parts.
/**
* Gets a map of form data body parts where the key is the control name
* and the value is a list of one or more form data body parts.
*
* @return return the map of form data body parts.
*/
public Map<String, List<FormDataBodyPart>> getFields() {
Map<String, List<FormDataBodyPart>> map = new HashMap<>();
for (BodyPart bodyPart : getBodyParts()) {
if (!(bodyPart instanceof FormDataBodyPart)) {
continue;
}
FormDataBodyPart p = (FormDataBodyPart) bodyPart;
List<FormDataBodyPart> l = map.get(p.getName());
if (l == null) {
l = new ArrayList<>(1);
map.put(p.getName(), l);
}
l.add(p);
}
return map;
}
Disables changing the media type to anything other than multipart/form-data
. Params: - mediaType – the proposed media type.
Throws: - IllegalArgumentException – if the proposed media type is not
multipart/form-data
.
/**
* Disables changing the media type to anything other than {@code multipart/form-data}.
*
* @param mediaType the proposed media type.
* @throws IllegalArgumentException if the proposed media type is not {@code multipart/form-data}.
*/
@Override
public void setMediaType(MediaType mediaType) {
if (!MediaTypes.typeEqual(mediaType, MediaType.MULTIPART_FORM_DATA_TYPE)) {
throw new IllegalArgumentException(LocalizationMessages.FORM_DATA_MULTIPART_CANNOT_CHANGE_MEDIATYPE());
}
super.setMediaType(mediaType);
}
}