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

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

The 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); } } }