diff options
Diffstat (limited to 'src/bun.js/bindings/webcore/HTTPHeaderMap.h')
-rw-r--r-- | src/bun.js/bindings/webcore/HTTPHeaderMap.h | 282 |
1 files changed, 282 insertions, 0 deletions
diff --git a/src/bun.js/bindings/webcore/HTTPHeaderMap.h b/src/bun.js/bindings/webcore/HTTPHeaderMap.h new file mode 100644 index 000000000..1fe19d311 --- /dev/null +++ b/src/bun.js/bindings/webcore/HTTPHeaderMap.h @@ -0,0 +1,282 @@ +/* + * Copyright (C) 2006 Apple Inc. All rights reserved. + * Copyright (C) 2009 Google Inc. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY + * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR + * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR + * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, + * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, + * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR + * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY + * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +#pragma once + +#include "HTTPHeaderNames.h" +#include <utility> +#include <wtf/text/WTFString.h> + +namespace WebCore { + +// FIXME: Not every header fits into a map. Notably, multiple Set-Cookie header fields are needed to set multiple cookies. + +class HTTPHeaderMap { +public: + struct CommonHeader { + HTTPHeaderName key; + String value; + + CommonHeader isolatedCopy() const & { return { key , value.isolatedCopy() }; } + CommonHeader isolatedCopy() && { return { key , WTFMove(value).isolatedCopy() }; } + template <class Encoder> void encode(Encoder&) const; + template <class Decoder> static std::optional<CommonHeader> decode(Decoder&); + + bool operator==(const CommonHeader& other) const { return key == other.key && value == other.value; } + }; + + struct UncommonHeader { + String key; + String value; + + UncommonHeader isolatedCopy() const & { return { key.isolatedCopy() , value.isolatedCopy() }; } + UncommonHeader isolatedCopy() && { return { WTFMove(key).isolatedCopy() , WTFMove(value).isolatedCopy() }; } + template <class Encoder> void encode(Encoder&) const; + template <class Decoder> static std::optional<UncommonHeader> decode(Decoder&); + + bool operator==(const UncommonHeader& other) const { return key == other.key && value == other.value; } + }; + + typedef Vector<CommonHeader, 0, CrashOnOverflow, 6> CommonHeadersVector; + typedef Vector<UncommonHeader, 0, CrashOnOverflow, 0> UncommonHeadersVector; + + class HTTPHeaderMapConstIterator { + public: + HTTPHeaderMapConstIterator(const HTTPHeaderMap& table, CommonHeadersVector::const_iterator commonHeadersIt, UncommonHeadersVector::const_iterator uncommonHeadersIt) + : m_table(table) + , m_commonHeadersIt(commonHeadersIt) + , m_uncommonHeadersIt(uncommonHeadersIt) + { + if (!updateKeyValue(m_commonHeadersIt)) + updateKeyValue(m_uncommonHeadersIt); + } + + struct KeyValue { + String key; + std::optional<HTTPHeaderName> keyAsHTTPHeaderName; + String value; + }; + + const KeyValue* get() const + { + ASSERT(*this != m_table.end()); + return &m_keyValue; + } + const KeyValue& operator*() const { return *get(); } + const KeyValue* operator->() const { return get(); } + + HTTPHeaderMapConstIterator& operator++() + { + if (m_commonHeadersIt != m_table.m_commonHeaders.end()) { + if (updateKeyValue(++m_commonHeadersIt)) + return *this; + } else + ++m_uncommonHeadersIt; + + updateKeyValue(m_uncommonHeadersIt); + return *this; + } + + bool operator!=(const HTTPHeaderMapConstIterator& other) const { return !(*this == other); } + bool operator==(const HTTPHeaderMapConstIterator& other) const + { + return m_commonHeadersIt == other.m_commonHeadersIt && m_uncommonHeadersIt == other.m_uncommonHeadersIt; + } + + private: + bool updateKeyValue(CommonHeadersVector::const_iterator it) + { + if (it == m_table.commonHeaders().end()) + return false; + m_keyValue.key = httpHeaderNameString(it->key).toStringWithoutCopying(); + m_keyValue.keyAsHTTPHeaderName = it->key; + m_keyValue.value = it->value; + return true; + } + bool updateKeyValue(UncommonHeadersVector::const_iterator it) + { + if (it == m_table.uncommonHeaders().end()) + return false; + m_keyValue.key = it->key; + m_keyValue.keyAsHTTPHeaderName = std::nullopt; + m_keyValue.value = it->value; + return true; + } + + const HTTPHeaderMap& m_table; + CommonHeadersVector::const_iterator m_commonHeadersIt; + UncommonHeadersVector::const_iterator m_uncommonHeadersIt; + KeyValue m_keyValue; + }; + typedef HTTPHeaderMapConstIterator const_iterator; + + WEBCORE_EXPORT HTTPHeaderMap(); + + // Gets a copy of the data suitable for passing to another thread. + WEBCORE_EXPORT HTTPHeaderMap isolatedCopy() const &; + WEBCORE_EXPORT HTTPHeaderMap isolatedCopy() &&; + + bool isEmpty() const { return m_commonHeaders.isEmpty() && m_uncommonHeaders.isEmpty(); } + int size() const { return m_commonHeaders.size() + m_uncommonHeaders.size(); } + + void clear() + { + m_commonHeaders.clear(); + m_uncommonHeaders.clear(); + } + + void shrinkToFit() + { + m_commonHeaders.shrinkToFit(); + m_uncommonHeaders.shrinkToFit(); + } + + WEBCORE_EXPORT String get(const String& name) const; + WEBCORE_EXPORT void set(const String& name, const String& value); + WEBCORE_EXPORT void add(const String& name, const String& value); + WEBCORE_EXPORT void append(const String& name, const String& value); + WEBCORE_EXPORT bool contains(const String&) const; + WEBCORE_EXPORT bool remove(const String&); + +#if USE(CF) + void set(CFStringRef name, const String& value); +#ifdef __OBJC__ + void set(NSString *name, const String& value) { set((__bridge CFStringRef)name, value); } +#endif +#endif + + WEBCORE_EXPORT String get(HTTPHeaderName) const; + void set(HTTPHeaderName, const String& value); + void add(HTTPHeaderName, const String& value); + bool addIfNotPresent(HTTPHeaderName, const String&); + WEBCORE_EXPORT bool contains(HTTPHeaderName) const; + WEBCORE_EXPORT bool remove(HTTPHeaderName); + + // Instead of passing a string literal to any of these functions, just use a HTTPHeaderName instead. + template<size_t length> String get(const char (&)[length]) const = delete; + template<size_t length> void set(const char (&)[length], const String&) = delete; + template<size_t length> bool contains(const char (&)[length]) = delete; + template<size_t length> bool remove(const char (&)[length]) = delete; + + const CommonHeadersVector& commonHeaders() const { return m_commonHeaders; } + const UncommonHeadersVector& uncommonHeaders() const { return m_uncommonHeaders; } + CommonHeadersVector& commonHeaders() { return m_commonHeaders; } + UncommonHeadersVector& uncommonHeaders() { return m_uncommonHeaders; } + + const_iterator begin() const { return const_iterator(*this, m_commonHeaders.begin(), m_uncommonHeaders.begin()); } + const_iterator end() const { return const_iterator(*this, m_commonHeaders.end(), m_uncommonHeaders.end()); } + + friend bool operator==(const HTTPHeaderMap& a, const HTTPHeaderMap& b) + { + if (a.m_commonHeaders.size() != b.m_commonHeaders.size() || a.m_uncommonHeaders.size() != b.m_uncommonHeaders.size()) + return false; + for (auto& commonHeader : a.m_commonHeaders) { + if (b.get(commonHeader.key) != commonHeader.value) + return false; + } + for (auto& uncommonHeader : a.m_uncommonHeaders) { + if (b.getUncommonHeader(uncommonHeader.key) != uncommonHeader.value) + return false; + } + return true; + } + + friend bool operator!=(const HTTPHeaderMap& a, const HTTPHeaderMap& b) + { + return !(a == b); + } + + template <class Encoder> void encode(Encoder&) const; + template <class Decoder> static WARN_UNUSED_RETURN bool decode(Decoder&, HTTPHeaderMap&); + +private: + void setUncommonHeader(const String& name, const String& value); + WEBCORE_EXPORT String getUncommonHeader(const String& name) const; + + CommonHeadersVector m_commonHeaders; + UncommonHeadersVector m_uncommonHeaders; +}; + +template <class Encoder> +void HTTPHeaderMap::CommonHeader::encode(Encoder& encoder) const +{ + encoder << key; + encoder << value; +} + +template <class Decoder> +auto HTTPHeaderMap::CommonHeader::decode(Decoder& decoder) -> std::optional<CommonHeader> +{ + HTTPHeaderName name; + if (!decoder.decode(name)) + return std::nullopt; + String value; + if (!decoder.decode(value)) + return std::nullopt; + + return CommonHeader { name, WTFMove(value) }; +} + +template <class Encoder> +void HTTPHeaderMap::UncommonHeader::encode(Encoder& encoder) const +{ + encoder << key; + encoder << value; +} + +template <class Decoder> +auto HTTPHeaderMap::UncommonHeader::decode(Decoder& decoder) -> std::optional<UncommonHeader> +{ + String name; + if (!decoder.decode(name)) + return std::nullopt; + String value; + if (!decoder.decode(value)) + return std::nullopt; + + return UncommonHeader { WTFMove(name), WTFMove(value) }; +} + +template <class Encoder> +void HTTPHeaderMap::encode(Encoder& encoder) const +{ + encoder << m_commonHeaders; + encoder << m_uncommonHeaders; +} + +template <class Decoder> +bool HTTPHeaderMap::decode(Decoder& decoder, HTTPHeaderMap& headerMap) +{ + if (!decoder.decode(headerMap.m_commonHeaders)) + return false; + + if (!decoder.decode(headerMap.m_uncommonHeaders)) + return false; + + return true; +} + +} // namespace WebCore |