/*
 * Copyright (c) 2011-2017 Contributors to the Eclipse Foundation
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License 2.0 which is available at
 * http://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
 * which is available at https://www.apache.org/licenses/LICENSE-2.0.
 *
 * SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
 */
package io.vertx.core.http.impl;

import io.netty.handler.codec.http.HttpHeaderNames;
import io.netty.handler.codec.http2.Http2Headers;
import io.vertx.core.MultiMap;

import java.util.AbstractList;
import java.util.AbstractMap;
import java.util.AbstractSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;

Author:Norman Maurer
/** * @author <a href="mailto:nmaurer@redhat.com">Norman Maurer</a> */
public class Http2HeadersAdaptor implements MultiMap { static CharSequence toLowerCase(CharSequence s) { StringBuilder buffer = null; int len = s.length(); for (int index = 0; index < len; index++) { char c = s.charAt(index); if (c >= 'A' && c <= 'Z') { if (buffer == null) { buffer = new StringBuilder(s); } buffer.setCharAt(index, (char)(c + ('a' - 'A'))); } } if (buffer != null) { return buffer.toString(); } else { return s; } } private final Http2Headers headers; private Set<String> names; public Http2HeadersAdaptor(Http2Headers headers) { List<CharSequence> cookies = headers.getAll(HttpHeaderNames.COOKIE); if (cookies != null && cookies.size() > 1) { // combine the cookie values into 1 header entry. // https://tools.ietf.org/html/rfc7540#section-8.1.2.5 String value = cookies.stream().collect(Collectors.joining("; ")); headers.set(HttpHeaderNames.COOKIE, value); } this.headers = headers; } @Override public String get(String name) { CharSequence val = headers.get(toLowerCase(name)); return val != null ? val.toString() : null; } @Override public List<String> getAll(String name) { List<CharSequence> all = headers.getAll(toLowerCase(name)); if (all != null) { return new AbstractList<String>() { @Override public String get(int index) { return all.get(index).toString(); } @Override public int size() { return all.size(); } }; } return null; } @Override public List<Map.Entry<String, String>> entries() { return headers.names() .stream() .map(name -> new AbstractMap.SimpleEntry<>(name.toString(), headers.get(name).toString())) .collect(Collectors.toList()); } @Override public boolean contains(String name) { return headers.contains(toLowerCase(name)); } @Override public boolean contains(String name, String value, boolean caseInsensitive) { return headers.contains(toLowerCase(name), value, caseInsensitive); } @Override public boolean isEmpty() { return headers.isEmpty(); } @Override public Set<String> names() { if (names == null) { names = new AbstractSet<String>() { @Override public Iterator<String> iterator() { Iterator<CharSequence> it = headers.names().iterator(); return new Iterator<String>() { @Override public boolean hasNext() { return it.hasNext(); } @Override public String next() { return it.next().toString(); } }; } @Override public int size() { return headers.size(); } }; } return names; } @Override public MultiMap add(String name, String value) { if (!io.vertx.core.http.HttpHeaders.DISABLE_HTTP_HEADERS_VALIDATION) { HttpUtils.validateHeader(name, value); } headers.add(toLowerCase(name), value); return this; } @Override public MultiMap add(String name, Iterable<String> values) { if (!io.vertx.core.http.HttpHeaders.DISABLE_HTTP_HEADERS_VALIDATION) { HttpUtils.validateHeader(name, values); } headers.add(toLowerCase(name), values); return this; } @Override public MultiMap addAll(MultiMap headers) { for (Map.Entry<String, String> entry: headers.entries()) { add(entry.getKey(), entry.getValue()); } return this; } @Override public MultiMap addAll(Map<String, String> map) { for (Map.Entry<String, String> entry: map.entrySet()) { add(entry.getKey(), entry.getValue()); } return this; } @Override public MultiMap set(String name, String value) { if (!io.vertx.core.http.HttpHeaders.DISABLE_HTTP_HEADERS_VALIDATION) { HttpUtils.validateHeader(name, value); } headers.set(toLowerCase(name), value); return this; } @Override public MultiMap set(String name, Iterable<String> values) { if (!io.vertx.core.http.HttpHeaders.DISABLE_HTTP_HEADERS_VALIDATION) { HttpUtils.validateHeader(name, values); } headers.set(toLowerCase(name), values); return this; } @Override public MultiMap setAll(MultiMap httpHeaders) { clear(); for (Map.Entry<String, String> entry: httpHeaders) { add(entry.getKey(), entry.getValue()); } return this; } @Override public MultiMap remove(String name) { headers.remove(toLowerCase(name)); return this; } @Override public MultiMap clear() { headers.clear(); return this; } @Override public Iterator<Map.Entry<String, String>> iterator() { return entries().iterator(); } @Override public int size() { return names().size(); } @Override public MultiMap setAll(Map<String, String> headers) { for (Map.Entry<String, String> entry: headers.entrySet()) { add(entry.getKey(), entry.getValue()); } return this; } @Override public String get(CharSequence name) { CharSequence val = headers.get(toLowerCase(name)); return val != null ? val.toString() : null; } @Override public List<String> getAll(CharSequence name) { List<CharSequence> all = headers.getAll(toLowerCase(name)); return all != null ? all.stream().map(CharSequence::toString).collect(Collectors.toList()) : null; } @Override public boolean contains(CharSequence name) { return headers.contains(toLowerCase(name)); } @Override public boolean contains(CharSequence name, CharSequence value, boolean caseInsensitive) { return headers.contains(toLowerCase(name), value, caseInsensitive); } @Override public MultiMap add(CharSequence name, CharSequence value) { if (!io.vertx.core.http.HttpHeaders.DISABLE_HTTP_HEADERS_VALIDATION) { HttpUtils.validateHeader(name, value); } headers.add(toLowerCase(name), value); return this; } @Override public MultiMap add(CharSequence name, Iterable<CharSequence> values) { if (!io.vertx.core.http.HttpHeaders.DISABLE_HTTP_HEADERS_VALIDATION) { HttpUtils.validateHeader(name, values); } headers.add(toLowerCase(name), values); return this; } @Override public MultiMap set(CharSequence name, CharSequence value) { if (!io.vertx.core.http.HttpHeaders.DISABLE_HTTP_HEADERS_VALIDATION) { HttpUtils.validateHeader(name, value); } headers.set(toLowerCase(name), value); return this; } @Override public MultiMap set(CharSequence name, Iterable<CharSequence> values) { if (!io.vertx.core.http.HttpHeaders.DISABLE_HTTP_HEADERS_VALIDATION) { HttpUtils.validateHeader(name, values); } headers.set(toLowerCase(name), values); return this; } @Override public MultiMap remove(CharSequence name) { headers.remove(toLowerCase(name)); return this; } }