All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.netty.handler.codec.http.CombinedHttpHeaders Maven / Gradle / Ivy

Go to download

This artifact provides a single jar that contains all classes required to use remote Jakarta Enterprise Beans and Jakarta Messaging, including all dependencies. It is intended for use by those not using maven, maven users should just import the Jakarta Enterprise Beans and Jakarta Messaging BOM's instead (shaded JAR's cause lots of problems with maven, as it is very easy to inadvertently end up with different versions on classes on the class path).

There is a newer version: 35.0.0.Beta1
Show newest version
/*
 * Copyright 2015 The Netty Project
 *
 * The Netty Project licenses this file to you 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:
 *
 *   https://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.netty.handler.codec.http;

import io.netty.handler.codec.DefaultHeaders;
import io.netty.handler.codec.DefaultHeaders.NameValidator;
import io.netty.handler.codec.DefaultHeaders.ValueValidator;
import io.netty.handler.codec.Headers;
import io.netty.handler.codec.ValueConverter;
import io.netty.util.HashingStrategy;
import io.netty.util.internal.StringUtil;

import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import static io.netty.handler.codec.http.HttpHeaderNames.SET_COOKIE;
import static io.netty.util.AsciiString.CASE_INSENSITIVE_HASHER;
import static io.netty.util.internal.ObjectUtil.checkNotNull;
import static io.netty.util.internal.StringUtil.COMMA;
import static io.netty.util.internal.StringUtil.unescapeCsvFields;

/**
 * Will add multiple values for the same header as single header with a comma separated list of values.
 * 

* Please refer to section RFC 7230, 3.2.2. */ public class CombinedHttpHeaders extends DefaultHttpHeaders { /** * Create a combined HTTP header object, with optional validation. * * @param validate Should Netty validate header values to ensure they aren't malicious. * @deprecated Prefer instead to configuring a {@link HttpHeadersFactory} * by calling {@link DefaultHttpHeadersFactory#withCombiningHeaders(boolean) withCombiningHeaders(true)} * on {@link DefaultHttpHeadersFactory#headersFactory()}. */ @Deprecated public CombinedHttpHeaders(boolean validate) { super(new CombinedHttpHeadersImpl(CASE_INSENSITIVE_HASHER, valueConverter(), nameValidator(validate), valueValidator(validate))); } CombinedHttpHeaders(NameValidator nameValidator, ValueValidator valueValidator) { super(new CombinedHttpHeadersImpl( CASE_INSENSITIVE_HASHER, valueConverter(), checkNotNull(nameValidator, "nameValidator"), checkNotNull(valueValidator, "valueValidator"))); } CombinedHttpHeaders( NameValidator nameValidator, ValueValidator valueValidator, int sizeHint) { super(new CombinedHttpHeadersImpl( CASE_INSENSITIVE_HASHER, valueConverter(), checkNotNull(nameValidator, "nameValidator"), checkNotNull(valueValidator, "valueValidator"), sizeHint)); } @Override public boolean containsValue(CharSequence name, CharSequence value, boolean ignoreCase) { return super.containsValue(name, StringUtil.trimOws(value), ignoreCase); } private static final class CombinedHttpHeadersImpl extends DefaultHeaders { /** * An estimate of the size of a header value. */ private static final int VALUE_LENGTH_ESTIMATE = 10; private CsvValueEscaper objectEscaper; private CsvValueEscaper charSequenceEscaper; private CsvValueEscaper objectEscaper() { if (objectEscaper == null) { objectEscaper = new CsvValueEscaper() { @Override public CharSequence escape(CharSequence name, Object value) { CharSequence converted; try { converted = valueConverter().convertObject(value); } catch (IllegalArgumentException e) { throw new IllegalArgumentException( "Failed to convert object value for header '" + name + '\'', e); } return StringUtil.escapeCsv(converted, true); } }; } return objectEscaper; } private CsvValueEscaper charSequenceEscaper() { if (charSequenceEscaper == null) { charSequenceEscaper = new CsvValueEscaper() { @Override public CharSequence escape(CharSequence name, CharSequence value) { return StringUtil.escapeCsv(value, true); } }; } return charSequenceEscaper; } CombinedHttpHeadersImpl(HashingStrategy nameHashingStrategy, ValueConverter valueConverter, NameValidator nameValidator, ValueValidator valueValidator) { this(nameHashingStrategy, valueConverter, nameValidator, valueValidator, 16); } CombinedHttpHeadersImpl(HashingStrategy nameHashingStrategy, ValueConverter valueConverter, NameValidator nameValidator, ValueValidator valueValidator, int sizeHint) { super(nameHashingStrategy, valueConverter, nameValidator, sizeHint, valueValidator); } @Override public Iterator valueIterator(CharSequence name) { Iterator itr = super.valueIterator(name); if (!itr.hasNext() || cannotBeCombined(name)) { return itr; } Iterator unescapedItr = unescapeCsvFields(itr.next()).iterator(); if (itr.hasNext()) { throw new IllegalStateException("CombinedHttpHeaders should only have one value"); } return unescapedItr; } @Override public List getAll(CharSequence name) { List values = super.getAll(name); if (values.isEmpty() || cannotBeCombined(name)) { return values; } if (values.size() != 1) { throw new IllegalStateException("CombinedHttpHeaders should only have one value"); } return unescapeCsvFields(values.get(0)); } @Override public CombinedHttpHeadersImpl add(Headers headers) { // Override the fast-copy mechanism used by DefaultHeaders if (headers == this) { throw new IllegalArgumentException("can't add to itself."); } if (headers instanceof CombinedHttpHeadersImpl) { if (isEmpty()) { // Can use the fast underlying copy addImpl(headers); } else { // Values are already escaped so don't escape again for (Map.Entry header : headers) { addEscapedValue(header.getKey(), header.getValue()); } } } else { for (Map.Entry header : headers) { add(header.getKey(), header.getValue()); } } return this; } @Override public CombinedHttpHeadersImpl set(Headers headers) { if (headers == this) { return this; } clear(); return add(headers); } @Override public CombinedHttpHeadersImpl setAll(Headers headers) { if (headers == this) { return this; } for (CharSequence key : headers.names()) { remove(key); } return add(headers); } @Override public CombinedHttpHeadersImpl add(CharSequence name, CharSequence value) { return addEscapedValue(name, charSequenceEscaper().escape(name, value)); } @Override public CombinedHttpHeadersImpl add(CharSequence name, CharSequence... values) { return addEscapedValue(name, commaSeparate(name, charSequenceEscaper(), values)); } @Override public CombinedHttpHeadersImpl add(CharSequence name, Iterable values) { return addEscapedValue(name, commaSeparate(name, charSequenceEscaper(), values)); } @Override public CombinedHttpHeadersImpl addObject(CharSequence name, Object value) { return addEscapedValue(name, commaSeparate(name, objectEscaper(), value)); } @Override public CombinedHttpHeadersImpl addObject(CharSequence name, Iterable values) { return addEscapedValue(name, commaSeparate(name, objectEscaper(), values)); } @Override public CombinedHttpHeadersImpl addObject(CharSequence name, Object... values) { return addEscapedValue(name, commaSeparate(name, objectEscaper(), values)); } @Override public CombinedHttpHeadersImpl set(CharSequence name, CharSequence... values) { set(name, commaSeparate(name, charSequenceEscaper(), values)); return this; } @Override public CombinedHttpHeadersImpl set(CharSequence name, Iterable values) { set(name, commaSeparate(name, charSequenceEscaper(), values)); return this; } @Override public CombinedHttpHeadersImpl setObject(CharSequence name, Object value) { set(name, commaSeparate(name, objectEscaper(), value)); return this; } @Override public CombinedHttpHeadersImpl setObject(CharSequence name, Object... values) { set(name, commaSeparate(name, objectEscaper(), values)); return this; } @Override public CombinedHttpHeadersImpl setObject(CharSequence name, Iterable values) { set(name, commaSeparate(name, objectEscaper(), values)); return this; } private static boolean cannotBeCombined(CharSequence name) { return SET_COOKIE.contentEqualsIgnoreCase(name); } private CombinedHttpHeadersImpl addEscapedValue(CharSequence name, CharSequence escapedValue) { CharSequence currentValue = get(name); if (currentValue == null || cannotBeCombined(name)) { super.add(name, escapedValue); } else { set(name, commaSeparateEscapedValues(currentValue, escapedValue)); } return this; } private static CharSequence commaSeparate(CharSequence name, CsvValueEscaper escaper, T... values) { StringBuilder sb = new StringBuilder(values.length * VALUE_LENGTH_ESTIMATE); if (values.length > 0) { int end = values.length - 1; for (int i = 0; i < end; i++) { sb.append(escaper.escape(name, values[i])).append(COMMA); } sb.append(escaper.escape(name, values[end])); } return sb; } private static CharSequence commaSeparate(CharSequence name, CsvValueEscaper escaper, Iterable values) { @SuppressWarnings("rawtypes") final StringBuilder sb = values instanceof Collection ? new StringBuilder(((Collection) values).size() * VALUE_LENGTH_ESTIMATE) : new StringBuilder(); Iterator iterator = values.iterator(); if (iterator.hasNext()) { T next = iterator.next(); while (iterator.hasNext()) { sb.append(escaper.escape(name, next)).append(COMMA); next = iterator.next(); } sb.append(escaper.escape(name, next)); } return sb; } private static CharSequence commaSeparateEscapedValues(CharSequence currentValue, CharSequence value) { return new StringBuilder(currentValue.length() + 1 + value.length()) .append(currentValue) .append(COMMA) .append(value); } /** * Escapes comma separated values (CSV). * * @param The type that a concrete implementation handles */ private interface CsvValueEscaper { /** * Appends the value to the specified {@link StringBuilder}, escaping if necessary. * * @param name the name of the header for the value being escaped * @param value the value to be appended, escaped if necessary */ CharSequence escape(CharSequence name, T value); } } }