io.helidon.http.ContentDisposition Maven / Gradle / Ivy
/*
* Copyright (c) 2022, 2023 Oracle and/or its affiliates.
*
* 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 io.helidon.http;
import java.net.URLDecoder;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.time.ZonedDateTime;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.OptionalLong;
import java.util.function.Function;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import io.helidon.common.GenericType;
import io.helidon.common.mapper.MapperException;
import io.helidon.common.mapper.MapperManager;
import io.helidon.common.mapper.Value;
/**
* A generic representation of the {@code Content-Disposition} header.
*
* Parameter encoding is not supported, other than
* URI percent
* encoding in the filename parameter. See {@link java.net.URLDecoder}.
*
* See also:
*
* - Communicating Presentation
* Information in Internet Messages: The Content-Disposition Header Field
*
* - Content-Disposition
* Header Field for each part
*
*
*/
public class ContentDisposition implements Header {
private static final String NAME_PARAMETER = "name";
private static final String FILENAME_PARAMETER = "filename";
private static final String CREATION_DATE_PARAMETER = "creation-date";
private static final String MODIFICATION_DATE_PARAMETER = "modification-date";
private static final String READ_DATE_PARAMETER = "read-date";
private static final String SIZE_PARAMETER = "size";
private static final ContentDisposition EMPTY = ContentDisposition.builder()
.type("")
.build();
private static final Pattern DISPOSITION_PART_PATTERN = Pattern.compile("^(.+?)=\"?(.+?)\"?$");
private final String type;
private final Map parameters;
private String value;
private ContentDisposition(Builder builder) {
this.type = builder.type;
this.parameters = new LinkedHashMap<>(builder.parameters);
}
/**
* A new builder to set up content disposition.
*
* @return builder
*/
public static Builder builder() {
return new Builder();
}
/**
* Parse a received header value.
*
* @param headerValue content disposition header value
* @return a parsed content disposition
*/
public static ContentDisposition parse(String headerValue) {
Builder builder = ContentDisposition.builder();
// first split by semicolon
String[] parts = headerValue.split("(?<=[^\\\\]);");
if (parts.length > 0) {
String type = parts[0];
if (type.indexOf('=') > -1) {
throw new IllegalArgumentException("No type defined");
} else {
builder.type(type.trim());
}
for (int i = 1; i < parts.length; i++) {
String part = parts[i];
Matcher matcher = DISPOSITION_PART_PATTERN.matcher(part.trim());
if (matcher.matches()) {
String name = matcher.group(1);
String value = matcher.group(2);
value = value.replace("\\\\", "\\");
value = value.replace("\\\"", "\"");
value = value.replace("\\;", ";");
builder.parameter(name, value);
}
}
}
return builder.build();
}
/**
* An empty content disposition.
*
* @return empty disposition with empty type
*/
public static ContentDisposition empty() {
return EMPTY;
}
@Override
public String name() {
return HeaderNames.CONTENT_DISPOSITION.defaultCase();
}
@Override
public HeaderName headerName() {
return HeaderNames.CONTENT_DISPOSITION;
}
@Override
public String get() {
if (value == null) {
StringBuilder sb = new StringBuilder();
sb.append(type);
for (Map.Entry param : parameters.entrySet()) {
sb.append(";");
sb.append(param.getKey());
sb.append("=");
if (SIZE_PARAMETER.equals(param.getKey())) {
sb.append(param.getValue());
} else {
sb.append("\"");
sb.append(param.getValue());
sb.append("\"");
}
}
value = sb.toString();
}
return value;
}
@Override
public Value as(Class type) throws MapperException {
return asString().as(type);
}
@Override
public Value as(GenericType type) throws MapperException {
return asString().as(type);
}
@Override
public Value as(Function super String, ? extends N> mapper) {
return asString().as(mapper);
}
@Override
public Optional asOptional() throws MapperException {
return asString().asOptional();
}
@Override
public Value asBoolean() {
return asString().asBoolean();
}
@Override
public Value asString() {
return Value.create(MapperManager.global(), name(), get(), GenericType.STRING, "http", "header");
}
@Override
public Value asInt() {
return asString().asInt();
}
@Override
public Value asLong() {
return asString().asLong();
}
@Override
public Value asDouble() {
return asString().asDouble();
}
@Override
public List allValues() {
return List.of(get());
}
@Override
public int valueCount() {
return 1;
}
@Override
public boolean sensitive() {
return false;
}
@Override
public boolean changing() {
return true;
}
@Override
public String toString() {
return get();
}
/**
* Get the value of the {@code name} parameter. In the case of a
* {@code form-data} disposition type the value is the original field name
* from the form.
*
* @return {@code Optional}, never {@code null}
*/
public Optional contentName() {
return Optional.ofNullable(parameters.get(NAME_PARAMETER));
}
/**
* Get the value of the {@code filename} parameter that can be used to
* suggest a filename to be used if the entity is detached and stored in a
* separate file.
*
* @return {@code Optional}, never {@code null}
*/
public Optional filename() {
String filename = null;
String value = parameters.get(FILENAME_PARAMETER);
if (value != null) {
filename = URLDecoder.decode(value, StandardCharsets.UTF_8);
}
return Optional.ofNullable(filename);
}
/**
* Get the value of the {@code creation-date} parameter that can be used
* to indicate the date at which the file was created.
*
* @return {@code Optional}, never {@code null}
*/
public Optional creationDate() {
return Optional.ofNullable(parameters.get(CREATION_DATE_PARAMETER)).map(DateTime::parse);
}
/**
* Get the value of the {@code modification-date} parameter that can be
* used to indicate the date at which the file was last modified.
*
* @return {@code Optional}, never {@code null}
*/
public Optional modificationDate() {
return Optional.ofNullable(parameters.get(MODIFICATION_DATE_PARAMETER)).map(DateTime::parse);
}
/**
* Get the value of the {@code modification-date} parameter that can be
* used to indicate the date at which the file was last read.
*
* @return {@code Optional}, never {@code null}
*/
public Optional readDate() {
return Optional.ofNullable(parameters.get(READ_DATE_PARAMETER)).map(DateTime::parse);
}
/**
* Get the value of the {@code size} parameter that can be
* used to indicate an approximate size of the file in octets.
*
* @return {@code OptionalLong}, never {@code null}
*/
public OptionalLong size() {
String size = parameters.get(SIZE_PARAMETER);
if (size != null) {
return OptionalLong.of(Long.parseLong(size));
}
return OptionalLong.empty();
}
/**
* Get the parameters map.
*
* @return map, never {@code null}
*/
public Map parameters() {
return Map.copyOf(parameters);
}
/**
* Content disposition type.
*
* @return type of this content disposition
*/
public String type() {
return type;
}
/**
* Fluent API builder for {@link ContentDisposition}.
*/
public static final class Builder implements io.helidon.common.Builder {
/**
* The form-data content disposition used by {@link io.helidon.common.media.type.MediaTypes#MULTIPART_FORM_DATA}.
*/
public static final String TYPE_FORM_DATA = "form-data";
private final Map parameters = new LinkedHashMap<>();
private String type = TYPE_FORM_DATA;
private Builder() {
}
@Override
public ContentDisposition build() {
return new ContentDisposition(this);
}
/**
* Set the content disposition type.
* Defaults to {@value #TYPE_FORM_DATA}.
*
* @param type content disposition type
* @return updated builder
*/
public Builder type(String type) {
this.type = type.toLowerCase();
return this;
}
/**
* Set the content disposition {@code name} parameter.
*
* @param name control name
* @return this builder
*/
public Builder name(String name) {
parameters.put(NAME_PARAMETER, name);
return this;
}
/**
* Set the content disposition {@code filename} parameter.
*
* @param filename filename parameter
* @return this builder
*/
public Builder filename(String filename) {
parameters.put(FILENAME_PARAMETER, URLEncoder.encode(filename, StandardCharsets.UTF_8));
return this;
}
/**
* Set the content disposition {@code creation-date} parameter.
*
* @param date date value
* @return this builder
*/
public Builder creationDate(ZonedDateTime date) {
parameters.put(CREATION_DATE_PARAMETER, date.format(DateTime.RFC_1123_DATE_TIME));
return this;
}
/**
* Set the content disposition {@code modification-date} parameter.
*
* @param date date value
* @return this builder
*/
public Builder modificationDate(ZonedDateTime date) {
parameters.put(MODIFICATION_DATE_PARAMETER, date.format(DateTime.RFC_1123_DATE_TIME));
return this;
}
/**
* Set the content disposition {@code read-date} parameter.
*
* @param date date value
* @return this builder
*/
public Builder readDate(ZonedDateTime date) {
parameters.put(READ_DATE_PARAMETER, date.format(DateTime.RFC_1123_DATE_TIME));
return this;
}
/**
* Set the content disposition {@code size} parameter.
*
* @param size size value
* @return this builder
*/
public Builder size(long size) {
parameters.put(SIZE_PARAMETER, Long.toString(size));
return this;
}
/**
* Add a new content disposition header parameter.
*
* @param name parameter name
* @param value parameter value
* @return this builder
*/
public Builder parameter(String name, String value) {
parameters.put(name, value);
return this;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy