src/java.net.http/share/classes/jdk/internal/net/http/hpack/SimpleHeaderTable.java
author chegar
Wed, 20 Jun 2018 18:23:56 +0100
branchhttp-client-branch
changeset 56795 03ece2518428
parent 56639 bd0dba2a0d51
parent 50681 4254bed3c09d
permissions -rw-r--r--
http-client-branch: merge with default
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
     1
/*
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
     2
 * Copyright (c) 2014, 2018, Oracle and/or its affiliates. All rights reserved.
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
     4
 *
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    10
 *
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    15
 * accompanied this code).
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    16
 *
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    20
 *
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    23
 * questions.
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    24
 */
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    25
package jdk.internal.net.http.hpack;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    26
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    27
import jdk.internal.net.http.hpack.HPACK.Logger;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    28
50681
4254bed3c09d 8204679: HTTP Client refresh
chegar
parents: 49944
diff changeset
    29
import java.util.List;
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    30
import java.util.NoSuchElementException;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    31
49944
4690a2871b44 8202423: Small HTTP Client refresh
chegar
parents: 49765
diff changeset
    32
import static jdk.internal.net.http.common.Utils.pow2Size;
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    33
import static jdk.internal.net.http.hpack.HPACK.Logger.Level.EXTRA;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    34
import static jdk.internal.net.http.hpack.HPACK.Logger.Level.NORMAL;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    35
import static java.lang.String.format;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    36
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    37
/*
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    38
 * A header table consists of two tables, the static table and the dynamic
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    39
 * table. Following the vocabulary of RFC 7541, the length of the header table
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    40
 * is the total number of entries in both the static and the dynamic tables.
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    41
 * The size of the table is the sum of the sizes of its dynamic table's entries.
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    42
 */
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    43
class SimpleHeaderTable {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    44
50681
4254bed3c09d 8204679: HTTP Client refresh
chegar
parents: 49944
diff changeset
    45
    /* An immutable list of static header fields */
4254bed3c09d 8204679: HTTP Client refresh
chegar
parents: 49944
diff changeset
    46
    protected static final List<HeaderField> staticTable = List.of(
4254bed3c09d 8204679: HTTP Client refresh
chegar
parents: 49944
diff changeset
    47
            new HeaderField(""), // A dummy to make the list index 1-based, instead of 0-based
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    48
            new HeaderField(":authority"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    49
            new HeaderField(":method", "GET"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    50
            new HeaderField(":method", "POST"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    51
            new HeaderField(":path", "/"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    52
            new HeaderField(":path", "/index.html"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    53
            new HeaderField(":scheme", "http"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    54
            new HeaderField(":scheme", "https"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    55
            new HeaderField(":status", "200"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    56
            new HeaderField(":status", "204"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    57
            new HeaderField(":status", "206"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    58
            new HeaderField(":status", "304"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    59
            new HeaderField(":status", "400"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    60
            new HeaderField(":status", "404"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    61
            new HeaderField(":status", "500"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    62
            new HeaderField("accept-charset"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    63
            new HeaderField("accept-encoding", "gzip, deflate"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    64
            new HeaderField("accept-language"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    65
            new HeaderField("accept-ranges"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    66
            new HeaderField("accept"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    67
            new HeaderField("access-control-allow-origin"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    68
            new HeaderField("age"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    69
            new HeaderField("allow"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    70
            new HeaderField("authorization"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    71
            new HeaderField("cache-control"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    72
            new HeaderField("content-disposition"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    73
            new HeaderField("content-encoding"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    74
            new HeaderField("content-language"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    75
            new HeaderField("content-length"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    76
            new HeaderField("content-location"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    77
            new HeaderField("content-range"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    78
            new HeaderField("content-type"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    79
            new HeaderField("cookie"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    80
            new HeaderField("date"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    81
            new HeaderField("etag"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    82
            new HeaderField("expect"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    83
            new HeaderField("expires"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    84
            new HeaderField("from"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    85
            new HeaderField("host"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    86
            new HeaderField("if-match"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    87
            new HeaderField("if-modified-since"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    88
            new HeaderField("if-none-match"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    89
            new HeaderField("if-range"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    90
            new HeaderField("if-unmodified-since"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    91
            new HeaderField("last-modified"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    92
            new HeaderField("link"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    93
            new HeaderField("location"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    94
            new HeaderField("max-forwards"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    95
            new HeaderField("proxy-authenticate"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    96
            new HeaderField("proxy-authorization"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    97
            new HeaderField("range"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    98
            new HeaderField("referer"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
    99
            new HeaderField("refresh"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   100
            new HeaderField("retry-after"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   101
            new HeaderField("server"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   102
            new HeaderField("set-cookie"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   103
            new HeaderField("strict-transport-security"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   104
            new HeaderField("transfer-encoding"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   105
            new HeaderField("user-agent"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   106
            new HeaderField("vary"),
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   107
            new HeaderField("via"),
50681
4254bed3c09d 8204679: HTTP Client refresh
chegar
parents: 49944
diff changeset
   108
            new HeaderField("www-authenticate"));
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   109
50681
4254bed3c09d 8204679: HTTP Client refresh
chegar
parents: 49944
diff changeset
   110
    protected static final int STATIC_TABLE_LENGTH = staticTable.size() - 1;
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   111
    protected static final int ENTRY_SIZE = 32;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   112
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   113
    private final Logger logger;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   114
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   115
    private int maxSize;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   116
    private int size;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   117
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   118
    public SimpleHeaderTable(int maxSize, Logger logger) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   119
        this.logger = logger;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   120
        setMaxSize(maxSize);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   121
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   122
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   123
    public int size() {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   124
        return size;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   125
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   126
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   127
    public int maxSize() {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   128
        return maxSize;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   129
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   130
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   131
    public int length() {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   132
        return STATIC_TABLE_LENGTH + buffer.size;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   133
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   134
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   135
    HeaderField get(int index) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   136
        checkIndex(index);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   137
        if (index <= STATIC_TABLE_LENGTH) {
50681
4254bed3c09d 8204679: HTTP Client refresh
chegar
parents: 49944
diff changeset
   138
            return staticTable.get(index);
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   139
        } else {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   140
            return buffer.get(index - STATIC_TABLE_LENGTH - 1);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   141
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   142
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   143
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   144
    void put(CharSequence name, CharSequence value) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   145
        // Invoking toString() will possibly allocate Strings. But that's
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   146
        // unavoidable at this stage. If a CharSequence is going to be stored in
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   147
        // the table, it must not be mutable (e.g. for the sake of hashing).
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   148
        put(new HeaderField(name.toString(), value.toString()));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   149
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   150
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   151
    private void put(HeaderField h) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   152
        if (logger.isLoggable(NORMAL)) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   153
            logger.log(NORMAL, () -> format("adding ('%s', '%s')",
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   154
                                            h.name, h.value));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   155
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   156
        int entrySize = sizeOf(h);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   157
        if (logger.isLoggable(EXTRA)) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   158
            logger.log(EXTRA, () -> format("size of ('%s', '%s') is %s",
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   159
                                           h.name, h.value, entrySize));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   160
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   161
        while (entrySize > maxSize - size && size != 0) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   162
            if (logger.isLoggable(EXTRA)) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   163
                logger.log(EXTRA, () -> format("insufficient space %s, must evict entry",
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   164
                                               (maxSize - size)));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   165
            }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   166
            evictEntry();
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   167
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   168
        if (entrySize > maxSize - size) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   169
            if (logger.isLoggable(EXTRA)) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   170
                logger.log(EXTRA, () -> format("not adding ('%s, '%s'), too big",
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   171
                                               h.name, h.value));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   172
            }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   173
            return;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   174
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   175
        size += entrySize;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   176
        add(h);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   177
        if (logger.isLoggable(EXTRA)) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   178
            logger.log(EXTRA, () -> format("('%s, '%s') added", h.name, h.value));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   179
            logger.log(EXTRA, this::toString);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   180
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   181
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   182
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   183
    void setMaxSize(int maxSize) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   184
        if (maxSize < 0) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   185
            throw new IllegalArgumentException(
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   186
                    "maxSize >= 0: maxSize=" + maxSize);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   187
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   188
        while (maxSize < size && size != 0) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   189
            evictEntry();
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   190
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   191
        this.maxSize = maxSize;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   192
        // A header table cannot accommodate more entries than this
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   193
        int upperBound = maxSize / ENTRY_SIZE;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   194
        buffer.resize(upperBound);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   195
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   196
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   197
    HeaderField evictEntry() {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   198
        HeaderField f = remove();
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   199
        int s = sizeOf(f);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   200
        this.size -= s;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   201
        if (logger.isLoggable(EXTRA)) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   202
            logger.log(EXTRA, () -> format("evicted entry ('%s', '%s') of size %s",
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   203
                                           f.name, f.value, s));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   204
            logger.log(EXTRA, this::toString);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   205
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   206
        return f;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   207
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   208
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   209
    @Override
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   210
    public String toString() {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   211
        double used = maxSize == 0 ? 0 : 100 * (((double) size) / maxSize);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   212
        return format("dynamic length: %d, full length: %s, used space: %s/%s (%.1f%%)",
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   213
                      buffer.size, length(), size, maxSize, used);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   214
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   215
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   216
    private int checkIndex(int index) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   217
        int len = length();
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   218
        if (index < 1 || index > len) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   219
            throw new IndexOutOfBoundsException(
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   220
                    format("1 <= index <= length(): index=%s, length()=%s",
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   221
                           index, len));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   222
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   223
        return index;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   224
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   225
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   226
    int sizeOf(HeaderField f) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   227
        return f.name.length() + f.value.length() + ENTRY_SIZE;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   228
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   229
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   230
    //
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   231
    // Diagnostic information in the form used in the RFC 7541
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   232
    //
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   233
    String getStateString() {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   234
        if (size == 0) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   235
            return "empty.";
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   236
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   237
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   238
        StringBuilder b = new StringBuilder();
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   239
        for (int i = 1, size = buffer.size; i <= size; i++) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   240
            HeaderField e = buffer.get(i - 1);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   241
            b.append(format("[%3d] (s = %3d) %s: %s\n", i,
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   242
                            sizeOf(e), e.name, e.value));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   243
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   244
        b.append(format("      Table size:%4s", this.size));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   245
        return b.toString();
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   246
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   247
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   248
    // Convert to a Value Object (JDK-8046159)?
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   249
    protected static final class HeaderField {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   250
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   251
        final String name;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   252
        final String value;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   253
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   254
        public HeaderField(String name) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   255
            this(name, "");
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   256
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   257
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   258
        public HeaderField(String name, String value) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   259
            this.name = name;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   260
            this.value = value;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   261
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   262
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   263
        @Override
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   264
        public String toString() {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   265
            return value.isEmpty() ? name : name + ": " + value;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   266
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   267
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   268
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   269
    private final CircularBuffer<HeaderField> buffer = new CircularBuffer<>(0);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   270
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   271
    protected void add(HeaderField f) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   272
        buffer.add(f);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   273
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   274
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   275
    protected HeaderField remove() {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   276
        return buffer.remove();
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   277
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   278
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   279
    //                    head
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   280
    //                    v
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   281
    // [ ][ ][A][B][C][D][ ][ ][ ]
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   282
    //        ^
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   283
    //        tail
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   284
    //
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   285
    //       |<- size ->| (4)
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   286
    // |<------ capacity ------->| (9)
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   287
    //
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   288
    static final class CircularBuffer<E> {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   289
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   290
        int tail, head, size, capacity;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   291
        Object[] elements;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   292
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   293
        CircularBuffer(int capacity) {
49944
4690a2871b44 8202423: Small HTTP Client refresh
chegar
parents: 49765
diff changeset
   294
            this.capacity = pow2Size(capacity);
4690a2871b44 8202423: Small HTTP Client refresh
chegar
parents: 49765
diff changeset
   295
            elements = new Object[this.capacity];
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   296
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   297
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   298
        void add(E elem) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   299
            if (size == capacity) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   300
                throw new IllegalStateException(
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   301
                        format("No room for '%s': capacity=%s", elem, capacity));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   302
            }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   303
            elements[head] = elem;
49944
4690a2871b44 8202423: Small HTTP Client refresh
chegar
parents: 49765
diff changeset
   304
            head = (head + 1) & (capacity - 1);
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   305
            size++;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   306
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   307
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   308
        @SuppressWarnings("unchecked")
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   309
        E remove() {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   310
            if (size == 0) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   311
                throw new NoSuchElementException("Empty");
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   312
            }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   313
            E elem = (E) elements[tail];
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   314
            elements[tail] = null;
49944
4690a2871b44 8202423: Small HTTP Client refresh
chegar
parents: 49765
diff changeset
   315
            tail = (tail + 1) & (capacity - 1);
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   316
            size--;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   317
            return elem;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   318
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   319
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   320
        @SuppressWarnings("unchecked")
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   321
        E get(int index) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   322
            if (index < 0 || index >= size) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   323
                throw new IndexOutOfBoundsException(
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   324
                        format("0 <= index <= capacity: index=%s, capacity=%s",
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   325
                               index, capacity));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   326
            }
49944
4690a2871b44 8202423: Small HTTP Client refresh
chegar
parents: 49765
diff changeset
   327
            int idx = (tail + (size - index - 1)) & (capacity - 1);
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   328
            return (E) elements[idx];
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   329
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   330
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   331
        public void resize(int newCapacity) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   332
            if (newCapacity < size) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   333
                throw new IllegalStateException(
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   334
                        format("newCapacity >= size: newCapacity=%s, size=%s",
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   335
                               newCapacity, size));
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   336
            }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   337
49944
4690a2871b44 8202423: Small HTTP Client refresh
chegar
parents: 49765
diff changeset
   338
            int capacity = pow2Size(newCapacity);
4690a2871b44 8202423: Small HTTP Client refresh
chegar
parents: 49765
diff changeset
   339
            Object[] newElements = new Object[capacity];
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   340
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   341
            if (tail < head || size == 0) {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   342
                System.arraycopy(elements, tail, newElements, 0, size);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   343
            } else {
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   344
                System.arraycopy(elements, tail, newElements, 0, elements.length - tail);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   345
                System.arraycopy(elements, 0, newElements, elements.length - tail, head);
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   346
            }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   347
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   348
            elements = newElements;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   349
            tail = 0;
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   350
            head = size;
49944
4690a2871b44 8202423: Small HTTP Client refresh
chegar
parents: 49765
diff changeset
   351
            this.capacity = capacity;
49765
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   352
        }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   353
    }
ee6f7a61f3a5 8197564: HTTP Client implementation
chegar
parents:
diff changeset
   354
}