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

io.helidon.http.HeadersImpl Maven / Gradle / Ivy

/*
 * Copyright (c) 2022, 2024 Oracle and/or its affiliates.
 *
 * Licensed 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
 *
 *     http://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.helidon.http;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.function.Consumer;
import java.util.function.Supplier;

@SuppressWarnings("unchecked")
class HeadersImpl> implements WritableHeaders {
    static final int KNOWN_HEADER_SIZE = HeaderNameEnum.values().length;
    /*
     Optimization for most commonly used header names
     */
    private final Header[] knownHeaders = new Header[KNOWN_HEADER_SIZE];
    private IntSet knownHeaderIndices = new IntSet(KNOWN_HEADER_SIZE);

    // custom (unknown) headers are slower
    private Map customHeaders = null;

    HeadersImpl() {
    }

    HeadersImpl(Headers headers) {
        for (Header header : headers) {
            set(header);
        }
    }

    @Override
    public List all(HeaderName name, Supplier> defaultSupplier) {
        Header headerValue = find(name);
        if (headerValue == null) {
            return defaultSupplier.get();
        }
        return headerValue.allValues();
    }

    @Override
    public boolean contains(HeaderName name) {
        return find(name) != null;
    }

    @Override
    public boolean contains(Header headerWithValue) {
        Header headerValue = find(headerWithValue.headerName());
        if (headerValue == null) {
            return false;
        }
        if (headerWithValue.valueCount() == 1 && headerValue.valueCount() == 1) {
            // just a string compare instead of list compare
            return headerWithValue.get().equals(headerValue.get());
        }
        return headerWithValue.allValues().equals(headerValue.allValues());
    }

    @Override
    public Header get(HeaderName name) {
        Header headerValue = find(name);
        if (headerValue == null) {
            throw new NoSuchElementException("Header " + name + " is not present in these headers");
        }
        return headerValue;
    }

    @Override
    public int size() {
        return (customHeaders == null ? 0 : customHeaders.size()) + knownHeaderIndices.size();
    }

    @Override
    public List acceptedTypes() {
        if (contains(HeaderNames.ACCEPT)) {
            List accepts = get(HeaderNames.ACCEPT).allValues(true);

            List mediaTypes = new ArrayList<>(accepts.size());
            for (String accept : accepts) {
                mediaTypes.add(HttpMediaType.create(accept));
            }
            Collections.sort(mediaTypes);
            return mediaTypes;
        } else {
            return List.of();
        }
    }

    @Override
    public Iterator
iterator() { return new HeaderIterator(); } @Override public T setIfAbsent(Header header) { Header found = find(header.headerName()); if (found == null) { set(header); } return (T) this; } @Override public T add(Header header) { HeaderName name = header.headerName(); Header headerValue = find(name); if (headerValue == null) { set(header); } else { HeaderWriteable writable; if (headerValue instanceof HeaderWriteable hvw) { writable = hvw; } else { writable = HeaderWriteable.create(header); } for (String value : header.allValues()) { writable.addValue(value); } set(writable); } return (T) this; } @Override public T remove(HeaderName name) { doRemove(name); return (T) this; } @Override public T remove(HeaderName name, Consumer
removedConsumer) { Header remove = doRemove(name); if (remove != null) { removedConsumer.accept(remove); } return (T) this; } @Override public T set(Header header) { HeaderName name = header.headerName(); Header usedHeader = header; if (header instanceof HeaderValueLazy) { // use it directly (lazy values are write once) } else if (header instanceof HeaderWriteable) { // we must create a new instance, as we risk modifying state of the provided header usedHeader = new HeaderValueCopy(header); } int index = name.index(); if (index == -1) { customHeaders().put(name, usedHeader); } else { knownHeaders[index] = usedHeader; knownHeaderIndices.add(index); } return (T) this; } @Override public T clear() { Arrays.fill(knownHeaders, null); knownHeaderIndices = new IntSet(KNOWN_HEADER_SIZE); customHeaders().clear(); return (T) this; } @Override public T from(Headers headers) { for (Header header : headers) { set(header); } return (T) this; } @Override public String toString() { StringBuilder builder = new StringBuilder(); for (Header headerValue : this) { for (String value : headerValue.allValues()) { builder.append(headerValue.name()) .append(": "); if (headerValue.sensitive()) { builder.append("*".repeat(value.length())); } else { builder.append(value); } List details = new ArrayList<>(2); if (headerValue.sensitive()) { details.add("sensitive"); } if (headerValue.changing()) { details.add("changing"); } if (!details.isEmpty()) { builder.append(" ("); builder.append(String.join(", ", details)); builder.append(")"); } builder.append("\n"); } } return builder.toString(); } public Header doRemove(HeaderName name) { if (name instanceof HeaderNameEnum) { int index = ((HeaderNameEnum) name).ordinal(); Header value = knownHeaders[index]; knownHeaders[index] = null; knownHeaderIndices.remove(index); return value; } return customHeaders().remove(name); } private Header find(HeaderName name) { int index = name.index(); if (index > -1) { return knownHeaders[index]; } return customHeaders().get(name); } private Map customHeaders() { if (customHeaders == null) { customHeaders = new HashMap<>(); } return customHeaders; } private class HeaderIterator implements Iterator
{ private final boolean noCustom = (customHeaders == null || customHeaders.isEmpty()); private boolean inKnown = true; private int last = -1; private Iterator
customHeadersIterator; @Override public boolean hasNext() { if (inKnown) { last = knownHeaderIndices.nextSetBit(last + 1); if (last >= 0) { return true; } inKnown = false; if (noCustom) { return false; } ensureCustom(); } return customHeadersIterator.hasNext(); } @Override public Header next() { if (last >= 0) { return knownHeaders[last]; } ensureCustom(); return customHeadersIterator.next(); } private void ensureCustom() { if (customHeadersIterator == null) { customHeadersIterator = customHeaders.values().iterator(); } } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy