package io.vertx.core.http;
import io.vertx.core.Handler;
import io.vertx.core.MultiMap;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.TreeSet;
import java.util.function.BiConsumer;
import java.util.function.Consumer;
public final class implements MultiMap {
private static final int = 17;
private static int (String name) {
int h = 0;
for (int i = name.length() - 1; i >= 0; i --) {
char c = name.charAt(i);
if (c >= 'A' && c <= 'Z') {
c += 32;
}
h = 31 * h + c;
}
if (h > 0) {
return h;
} else if (h == Integer.MIN_VALUE) {
return Integer.MAX_VALUE;
} else {
return -h;
}
}
private MultiMap (Iterable<Map.Entry<String, String>> map) {
clear();
for (Map.Entry<String, String> entry: map) {
add(entry.getKey(), entry.getValue());
}
return this;
}
@Override
public MultiMap (MultiMap headers) {
return set0(headers);
}
@Override
public MultiMap (Map<String, String> headers) {
return set0(headers.entrySet());
}
@Override
public int () {
return names().size();
}
private static boolean (String name1, String name2) {
int nameLen = name1.length();
if (nameLen != name2.length()) {
return false;
}
for (int i = nameLen - 1; i >= 0; i --) {
char c1 = name1.charAt(i);
char c2 = name2.charAt(i);
if (c1 != c2) {
if (c1 >= 'A' && c1 <= 'Z') {
c1 += 32;
}
if (c2 >= 'A' && c2 <= 'Z') {
c2 += 32;
}
if (c1 != c2) {
return false;
}
}
}
return true;
}
private static int (int hash) {
return hash % BUCKET_SIZE;
}
private final MapEntry[] = new MapEntry[BUCKET_SIZE];
private final MapEntry = new MapEntry(-1, null, null);
public () {
head.before = head.after = head;
}
@Override
public MultiMap (final String name, final String strVal) {
int h = hash(name);
int i = index(h);
add0(h, i, name, strVal);
return this;
}
@Override
public MultiMap (String name, Iterable<String> values) {
int h = hash(name);
int i = index(h);
for (String vstr: values) {
add0(h, i, name, vstr);
}
return this;
}
@Override
public MultiMap (MultiMap headers) {
for (Map.Entry<String, String> entry: headers.entries()) {
add(entry.getKey(), entry.getValue());
}
return this;
}
@Override
public MultiMap (Map<String, String> map) {
for (Map.Entry<String, String> entry: map.entrySet()) {
add(entry.getKey(), entry.getValue());
}
return this;
}
private void (int h, int i, final String name, final String value) {
MapEntry e = entries[i];
MapEntry newEntry;
entries[i] = newEntry = new MapEntry(h, name, value);
newEntry.next = e;
newEntry.addBefore(head);
}
@Override
public MultiMap (final String name) {
Objects.requireNonNull(name, "name");
int h = hash(name);
int i = index(h);
remove0(h, i, name);
return this;
}
private void (int h, int i, String name) {
MapEntry e = entries[i];
if (e == null) {
return;
}
for (;;) {
if (e.hash == h && eq(name, e.key)) {
e.remove();
MapEntry next = e.next;
if (next != null) {
entries[i] = next;
e = next;
} else {
entries[i] = null;
return;
}
} else {
break;
}
}
for (;;) {
MapEntry next = e.next;
if (next == null) {
break;
}
if (next.hash == h && eq(name, next.key)) {
e.next = next.next;
next.remove();
} else {
e = next;
}
}
}
@Override
public MultiMap (final String name, final String strVal) {
int h = hash(name);
int i = index(h);
remove0(h, i, name);
add0(h, i, name, strVal);
return this;
}
@Override
public MultiMap (final String name, final Iterable<String> values) {
Objects.requireNonNull(values, "values");
int h = hash(name);
int i = index(h);
remove0(h, i, name);
for (String v: values) {
if (v == null) {
break;
}
add0(h, i, name, v);
}
return this;
}
@Override
public MultiMap () {
for (int i = 0; i < entries.length; i ++) {
entries[i] = null;
}
head.before = head.after = head;
return this;
}
@Override
public String (final String name) {
Objects.requireNonNull(name, "name");
int h = hash(name);
int i = index(h);
MapEntry e = entries[i];
String value = null;
while (e != null) {
if (e.hash == h && eq(name, e.key)) {
value = e.getValue();
}
e = e.next;
}
return value;
}
@Override
public List<String> (final String name) {
Objects.requireNonNull(name, "name");
LinkedList<String> values = new LinkedList<>();
int h = hash(name);
int i = index(h);
MapEntry e = entries[i];
while (e != null) {
if (e.hash == h && eq(name, e.key)) {
values.addFirst(e.getValue());
}
e = e.next;
}
return values;
}
@Override
public void (Consumer<? super Map.Entry<String, String>> action) {
MapEntry e = head.after;
while (e != head) {
action.accept(e);
e = e.after;
}
}
@Override
public List<Map.Entry<String, String>> () {
List<Map.Entry<String, String>> all =
new LinkedList<>();
MapEntry e = head.after;
while (e != head) {
all.add(e);
e = e.after;
}
return all;
}
@Override
public Iterator<Map.Entry<String, String>> () {
return entries().iterator();
}
@Override
public boolean (String name) {
return get(name) != null;
}
@Override
public boolean () {
return head == head.after;
}
@Override
public Set<String> () {
Set<String> names = new TreeSet<>(String.CASE_INSENSITIVE_ORDER);
MapEntry e = head.after;
while (e != head) {
names.add(e.getKey());
e = e.after;
}
return names;
}
@Override
public String (CharSequence name) {
return get(name.toString());
}
@Override
public List<String> (CharSequence name) {
return getAll(name.toString());
}
@Override
public boolean (CharSequence name) {
return contains(name.toString());
}
@Override
public MultiMap (CharSequence name, CharSequence value) {
return add(name.toString(), value.toString());
}
@Override
public MultiMap (CharSequence name, Iterable<CharSequence> values) {
String n = name.toString();
for (CharSequence seq: values) {
add(n, seq.toString());
}
return this;
}
@Override
public MultiMap (CharSequence name, CharSequence value) {
return set(name.toString(), value.toString());
}
@Override
public MultiMap (CharSequence name, Iterable<CharSequence> values) {
remove(name);
String n = name.toString();
for (CharSequence seq: values) {
add(n, seq.toString());
}
return this;
}
@Override
public MultiMap (CharSequence name) {
return remove(name.toString());
}
public String () {
StringBuilder sb = new StringBuilder();
for (Map.Entry<String, String> entry: this) {
sb.append(entry).append('\n');
}
return sb.toString();
}
private static final class implements Map.Entry<String, String> {
final int ;
final String ;
String ;
MapEntry ;
MapEntry , ;
(int hash, String key, String value) {
this.hash = hash;
this.key = key;
this.value = value;
}
void () {
before.after = after;
after.before = before;
}
void (MapEntry e) {
after = e;
before = e.before;
before.after = this;
after.before = this;
}
@Override
public String () {
return key;
}
@Override
public String () {
return value;
}
@Override
public String (String value) {
Objects.requireNonNull(value, "value");
String oldValue = this.value;
this.value = value;
return oldValue;
}
@Override
public String () {
return getKey() + ": " + getValue();
}
}
}