Coverage Report

Created: 2026-05-30 06:25

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/http/src/header/name.rs
Line
Count
Source
1
use crate::byte_str::ByteStr;
2
use bytes::{Bytes, BytesMut};
3
4
use std::borrow::Borrow;
5
use std::convert::TryFrom;
6
use std::error::Error;
7
use std::fmt;
8
use std::hash::{Hash, Hasher};
9
use std::mem::MaybeUninit;
10
use std::str::FromStr;
11
12
/// Represents an HTTP header field name
13
///
14
/// Header field names identify the header. Header sets may include multiple
15
/// headers with the same name. The HTTP specification defines a number of
16
/// standard headers, but HTTP messages may include non-standard header names as
17
/// well as long as they adhere to the specification.
18
///
19
/// `HeaderName` is used as the [`HeaderMap`] key. Constants are available for
20
/// all standard header names in the [`header`] module.
21
///
22
/// # Representation
23
///
24
/// `HeaderName` represents standard header names using an `enum`, as such they
25
/// will not require an allocation for storage. All custom header names are
26
/// lower cased upon conversion to a `HeaderName` value. This avoids the
27
/// overhead of dynamically doing lower case conversion during the hash code
28
/// computation and the comparison operation.
29
///
30
/// [`HeaderMap`]: struct.HeaderMap.html
31
/// [`header`]: index.html
32
#[derive(Clone, Eq, PartialEq, Hash)]
33
pub struct HeaderName {
34
    inner: Repr<Custom>,
35
}
36
37
// Almost a full `HeaderName`
38
#[derive(Debug, Hash)]
39
pub struct HdrName<'a> {
40
    inner: Repr<MaybeLower<'a>>,
41
}
42
43
#[derive(Debug, Clone, Eq, PartialEq, Hash)]
44
enum Repr<T> {
45
    Standard(StandardHeader),
46
    Custom(T),
47
}
48
49
// Used to hijack the Hash impl
50
#[derive(Debug, Clone, Eq, PartialEq)]
51
struct Custom(ByteStr);
52
53
#[derive(Debug, Clone)]
54
// Invariant: If lower then buf is valid UTF-8.
55
struct MaybeLower<'a> {
56
    buf: &'a [u8],
57
    lower: bool,
58
}
59
60
/// A possible error when converting a `HeaderName` from another type.
61
pub struct InvalidHeaderName {
62
    _priv: (),
63
}
64
65
macro_rules! standard_headers {
66
    (
67
        $(
68
            $(#[$docs:meta])*
69
            ($konst:ident, $upcase:ident, $name_bytes:literal);
70
        )+
71
    ) => {
72
        #[derive(Debug, Clone, Copy, Eq, PartialEq, Hash)]
73
        enum StandardHeader {
74
            $(
75
                $konst,
76
            )+
77
        }
78
79
        $(
80
            $(#[$docs])*
81
            pub const $upcase: HeaderName = HeaderName {
82
                inner: Repr::Standard(StandardHeader::$konst),
83
            };
84
        )+
85
86
        impl StandardHeader {
87
            #[inline]
88
0
            fn as_str(&self) -> &'static str {
89
0
                match *self {
90
                    // Safety: test_parse_standard_headers ensures these &[u8]s are &str-safe.
91
                    $(
92
0
                    StandardHeader::$konst => unsafe { std::str::from_utf8_unchecked( $name_bytes ) },
93
                    )+
94
                }
95
0
            }
96
97
6.25k
            const fn from_bytes(name_bytes: &[u8]) -> Option<StandardHeader> {
98
6.25k
                match name_bytes {
99
                    $(
100
9
                        $name_bytes => Some(StandardHeader::$konst),
101
                    )+
102
6.00k
                    _ => None,
103
                }
104
6.25k
            }
105
        }
106
107
        #[cfg(test)]
108
        const TEST_HEADERS: &'static [(StandardHeader, &'static [u8])] = &[
109
            $(
110
            (StandardHeader::$konst, $name_bytes),
111
            )+
112
        ];
113
114
        #[test]
115
        fn test_parse_standard_headers() {
116
            for &(std, name_bytes) in TEST_HEADERS {
117
                // Test lower case
118
                assert_eq!(HeaderName::from_bytes(name_bytes).unwrap(), HeaderName::from(std));
119
120
                // Test upper case
121
                let upper = std::str::from_utf8(name_bytes).expect("byte string constants are all utf-8").to_uppercase();
122
                assert_eq!(HeaderName::from_bytes(upper.as_bytes()).unwrap(), HeaderName::from(std));
123
            }
124
        }
125
126
        #[test]
127
        fn test_standard_headers_into_bytes() {
128
            for &(std, name_bytes) in TEST_HEADERS {
129
                let name = std::str::from_utf8(name_bytes).unwrap();
130
                let std = HeaderName::from(std);
131
                // Test lower case
132
                let bytes: Bytes =
133
                    HeaderName::from_bytes(name_bytes).unwrap().inner.into();
134
                assert_eq!(bytes, name);
135
                assert_eq!(HeaderName::from_bytes(name_bytes).unwrap(), std);
136
137
                // Test upper case
138
                let upper = name.to_uppercase();
139
                let bytes: Bytes =
140
                    HeaderName::from_bytes(upper.as_bytes()).unwrap().inner.into();
141
                assert_eq!(bytes, name_bytes);
142
                assert_eq!(HeaderName::from_bytes(upper.as_bytes()).unwrap(),
143
                           std);
144
            }
145
146
        }
147
    }
148
}
149
150
// Generate constants for all standard HTTP headers. This includes a static hash
151
// code for the "fast hash" path. The hash code for static headers *do not* have
152
// to match the text representation of those headers. This is because header
153
// strings are always converted to the static values (when they match) before
154
// being hashed. This means that it is impossible to compare the static hash
155
// code of CONTENT_LENGTH with "content-length".
156
standard_headers! {
157
    /// Advertises which content types the client is able to understand.
158
    ///
159
    /// The Accept request HTTP header advertises which content types, expressed
160
    /// as MIME types, the client is able to understand. Using content
161
    /// negotiation, the server then selects one of the proposals, uses it and
162
    /// informs the client of its choice with the Content-Type response header.
163
    /// Browsers set adequate values for this header depending of the context
164
    /// where the request is done: when fetching a CSS stylesheet a different
165
    /// value is set for the request than when fetching an image, video or a
166
    /// script.
167
    (Accept, ACCEPT, b"accept");
168
169
    /// Advertises which character set the client is able to understand.
170
    ///
171
    /// The Accept-Charset request HTTP header advertises which character set
172
    /// the client is able to understand. Using content negotiation, the server
173
    /// then selects one of the proposals, uses it and informs the client of its
174
    /// choice within the Content-Type response header. Browsers usually don't
175
    /// set this header as the default value for each content type is usually
176
    /// correct and transmitting it would allow easier fingerprinting.
177
    ///
178
    /// If the server cannot serve any matching character set, it can
179
    /// theoretically send back a 406 (Not Acceptable) error code. But, for a
180
    /// better user experience, this is rarely done and the more common way is
181
    /// to ignore the Accept-Charset header in this case.
182
    (AcceptCharset, ACCEPT_CHARSET, b"accept-charset");
183
184
    /// Advertises which content encoding the client is able to understand.
185
    ///
186
    /// The Accept-Encoding request HTTP header advertises which content
187
    /// encoding, usually a compression algorithm, the client is able to
188
    /// understand. Using content negotiation, the server selects one of the
189
    /// proposals, uses it and informs the client of its choice with the
190
    /// Content-Encoding response header.
191
    ///
192
    /// Even if both the client and the server supports the same compression
193
    /// algorithms, the server may choose not to compress the body of a
194
    /// response, if the identity value is also acceptable. Two common cases
195
    /// lead to this:
196
    ///
197
    /// * The data to be sent is already compressed and a second compression
198
    /// won't lead to smaller data to be transmitted. This may the case with
199
    /// some image formats;
200
    ///
201
    /// * The server is overloaded and cannot afford the computational overhead
202
    /// induced by the compression requirement. Typically, Microsoft recommends
203
    /// not to compress if a server use more than 80 % of its computational
204
    /// power.
205
    ///
206
    /// As long as the identity value, meaning no compression, is not explicitly
207
    /// forbidden, by an identity;q=0 or a *;q=0 without another explicitly set
208
    /// value for identity, the server must never send back a 406 Not Acceptable
209
    /// error.
210
    (AcceptEncoding, ACCEPT_ENCODING, b"accept-encoding");
211
212
    /// Advertises which languages the client is able to understand.
213
    ///
214
    /// The Accept-Language request HTTP header advertises which languages the
215
    /// client is able to understand, and which locale variant is preferred.
216
    /// Using content negotiation, the server then selects one of the proposals,
217
    /// uses it and informs the client of its choice with the Content-Language
218
    /// response header. Browsers set adequate values for this header according
219
    /// their user interface language and even if a user can change it, this
220
    /// happens rarely (and is frown upon as it leads to fingerprinting).
221
    ///
222
    /// This header is a hint to be used when the server has no way of
223
    /// determining the language via another way, like a specific URL, that is
224
    /// controlled by an explicit user decision. It is recommended that the
225
    /// server never overrides an explicit decision. The content of the
226
    /// Accept-Language is often out of the control of the user (like when
227
    /// traveling and using an Internet Cafe in a different country); the user
228
    /// may also want to visit a page in another language than the locale of
229
    /// their user interface.
230
    ///
231
    /// If the server cannot serve any matching language, it can theoretically
232
    /// send back a 406 (Not Acceptable) error code. But, for a better user
233
    /// experience, this is rarely done and more common way is to ignore the
234
    /// Accept-Language header in this case.
235
    (AcceptLanguage, ACCEPT_LANGUAGE, b"accept-language");
236
237
    /// Marker used by the server to advertise partial request support.
238
    ///
239
    /// The Accept-Ranges response HTTP header is a marker used by the server to
240
    /// advertise its support of partial requests. The value of this field
241
    /// indicates the unit that can be used to define a range.
242
    ///
243
    /// In presence of an Accept-Ranges header, the browser may try to resume an
244
    /// interrupted download, rather than to start it from the start again.
245
    (AcceptRanges, ACCEPT_RANGES, b"accept-ranges");
246
247
    /// Preflight response indicating if the response to the request can be
248
    /// exposed to the page.
249
    ///
250
    /// The Access-Control-Allow-Credentials response header indicates whether
251
    /// or not the response to the request can be exposed to the page. It can be
252
    /// exposed when the true value is returned; it can't in other cases.
253
    ///
254
    /// Credentials are cookies, authorization headers or TLS client
255
    /// certificates.
256
    ///
257
    /// When used as part of a response to a preflight request, this indicates
258
    /// whether or not the actual request can be made using credentials. Note
259
    /// that simple GET requests are not preflighted, and so if a request is
260
    /// made for a resource with credentials, if this header is not returned
261
    /// with the resource, the response is ignored by the browser and not
262
    /// returned to web content.
263
    ///
264
    /// The Access-Control-Allow-Credentials header works in conjunction with
265
    /// the XMLHttpRequest.withCredentials property or with the credentials
266
    /// option in the Request() constructor of the Fetch API. Credentials must
267
    /// be set on both sides (the Access-Control-Allow-Credentials header and in
268
    /// the XHR or Fetch request) in order for the CORS request with credentials
269
    /// to succeed.
270
    (AccessControlAllowCredentials, ACCESS_CONTROL_ALLOW_CREDENTIALS, b"access-control-allow-credentials");
271
272
    /// Preflight response indicating permitted HTTP headers.
273
    ///
274
    /// The Access-Control-Allow-Headers response header is used in response to
275
    /// a preflight request to indicate which HTTP headers will be available via
276
    /// Access-Control-Expose-Headers when making the actual request.
277
    ///
278
    /// The simple headers, Accept, Accept-Language, Content-Language,
279
    /// Content-Type (but only with a MIME type of its parsed value (ignoring
280
    /// parameters) of either application/x-www-form-urlencoded,
281
    /// multipart/form-data, or text/plain), are always available and don't need
282
    /// to be listed by this header.
283
    ///
284
    /// This header is required if the request has an
285
    /// Access-Control-Request-Headers header.
286
    (AccessControlAllowHeaders, ACCESS_CONTROL_ALLOW_HEADERS, b"access-control-allow-headers");
287
288
    /// Preflight header response indicating permitted access methods.
289
    ///
290
    /// The Access-Control-Allow-Methods response header specifies the method or
291
    /// methods allowed when accessing the resource in response to a preflight
292
    /// request.
293
    (AccessControlAllowMethods, ACCESS_CONTROL_ALLOW_METHODS, b"access-control-allow-methods");
294
295
    /// Indicates whether the response can be shared with resources with the
296
    /// given origin.
297
    (AccessControlAllowOrigin, ACCESS_CONTROL_ALLOW_ORIGIN, b"access-control-allow-origin");
298
299
    /// Indicates which headers can be exposed as part of the response by
300
    /// listing their names.
301
    (AccessControlExposeHeaders, ACCESS_CONTROL_EXPOSE_HEADERS, b"access-control-expose-headers");
302
303
    /// Indicates how long the results of a preflight request can be cached.
304
    (AccessControlMaxAge, ACCESS_CONTROL_MAX_AGE, b"access-control-max-age");
305
306
    /// Informs the server which HTTP headers will be used when an actual
307
    /// request is made.
308
    (AccessControlRequestHeaders, ACCESS_CONTROL_REQUEST_HEADERS, b"access-control-request-headers");
309
310
    /// Informs the server know which HTTP method will be used when the actual
311
    /// request is made.
312
    (AccessControlRequestMethod, ACCESS_CONTROL_REQUEST_METHOD, b"access-control-request-method");
313
314
    /// Indicates the time in seconds the object has been in a proxy cache.
315
    ///
316
    /// The Age header is usually close to zero. If it is Age: 0, it was
317
    /// probably just fetched from the origin server; otherwise It is usually
318
    /// calculated as a difference between the proxy's current date and the Date
319
    /// general header included in the HTTP response.
320
    (Age, AGE, b"age");
321
322
    /// Lists the set of methods support by a resource.
323
    ///
324
    /// This header must be sent if the server responds with a 405 Method Not
325
    /// Allowed status code to indicate which request methods can be used. An
326
    /// empty Allow header indicates that the resource allows no request
327
    /// methods, which might occur temporarily for a given resource, for
328
    /// example.
329
    (Allow, ALLOW, b"allow");
330
331
    /// Advertises the availability of alternate services to clients.
332
    (AltSvc, ALT_SVC, b"alt-svc");
333
334
    /// Contains the credentials to authenticate a user agent with a server.
335
    ///
336
    /// Usually this header is included after the server has responded with a
337
    /// 401 Unauthorized status and the WWW-Authenticate header.
338
    (Authorization, AUTHORIZATION, b"authorization");
339
340
    /// Specifies directives for caching mechanisms in both requests and
341
    /// responses.
342
    ///
343
    /// Caching directives are unidirectional, meaning that a given directive in
344
    /// a request is not implying that the same directive is to be given in the
345
    /// response.
346
    (CacheControl, CACHE_CONTROL, b"cache-control");
347
348
    /// Indicates how caches have handled a response and its corresponding request.
349
    ///
350
    /// See [RFC 9211](https://www.rfc-editor.org/rfc/rfc9211.html).
351
    (CacheStatus, CACHE_STATUS, b"cache-status");
352
353
    /// Specifies directives that allow origin servers to control the behavior of CDN caches
354
    /// interposed between them and clients separately from other caches that might handle the
355
    /// response.
356
    ///
357
    /// See [RFC 9213](https://www.rfc-editor.org/rfc/rfc9213.html).
358
    (CdnCacheControl, CDN_CACHE_CONTROL, b"cdn-cache-control");
359
360
    /// Controls whether or not the network connection stays open after the
361
    /// current transaction finishes.
362
    ///
363
    /// If the value sent is keep-alive, the connection is persistent and not
364
    /// closed, allowing for subsequent requests to the same server to be done.
365
    ///
366
    /// Except for the standard hop-by-hop headers (Keep-Alive,
367
    /// Transfer-Encoding, TE, Connection, Trailer, Upgrade, Proxy-Authorization
368
    /// and Proxy-Authenticate), any hop-by-hop headers used by the message must
369
    /// be listed in the Connection header, so that the first proxy knows he has
370
    /// to consume them and not to forward them further. Standard hop-by-hop
371
    /// headers can be listed too (it is often the case of Keep-Alive, but this
372
    /// is not mandatory.
373
    (Connection, CONNECTION, b"connection");
374
375
    /// Indicates if the content is expected to be displayed inline.
376
    ///
377
    /// In a regular HTTP response, the Content-Disposition response header is a
378
    /// header indicating if the content is expected to be displayed inline in
379
    /// the browser, that is, as a Web page or as part of a Web page, or as an
380
    /// attachment, that is downloaded and saved locally.
381
    ///
382
    /// In a multipart/form-data body, the HTTP Content-Disposition general
383
    /// header is a header that can be used on the subpart of a multipart body
384
    /// to give information about the field it applies to. The subpart is
385
    /// delimited by the boundary defined in the Content-Type header. Used on
386
    /// the body itself, Content-Disposition has no effect.
387
    ///
388
    /// The Content-Disposition header is defined in the larger context of MIME
389
    /// messages for e-mail, but only a subset of the possible parameters apply
390
    /// to HTTP forms and POST requests. Only the value form-data, as well as
391
    /// the optional directive name and filename, can be used in the HTTP
392
    /// context.
393
    (ContentDisposition, CONTENT_DISPOSITION, b"content-disposition");
394
395
    /// Used to compress the media-type.
396
    ///
397
    /// When present, its value indicates what additional content encoding has
398
    /// been applied to the entity-body. It lets the client know, how to decode
399
    /// in order to obtain the media-type referenced by the Content-Type header.
400
    ///
401
    /// It is recommended to compress data as much as possible and therefore to
402
    /// use this field, but some types of resources, like jpeg images, are
403
    /// already compressed.  Sometimes using additional compression doesn't
404
    /// reduce payload size and can even make the payload longer.
405
    (ContentEncoding, CONTENT_ENCODING, b"content-encoding");
406
407
    /// Used to describe the languages intended for the audience.
408
    ///
409
    /// This header allows a user to differentiate according to the users' own
410
    /// preferred language. For example, if "Content-Language: de-DE" is set, it
411
    /// says that the document is intended for German language speakers
412
    /// (however, it doesn't indicate the document is written in German. For
413
    /// example, it might be written in English as part of a language course for
414
    /// German speakers).
415
    ///
416
    /// If no Content-Language is specified, the default is that the content is
417
    /// intended for all language audiences. Multiple language tags are also
418
    /// possible, as well as applying the Content-Language header to various
419
    /// media types and not only to textual documents.
420
    (ContentLanguage, CONTENT_LANGUAGE, b"content-language");
421
422
    /// Indicates the size of the entity-body.
423
    ///
424
    /// The header value must be a decimal indicating the number of octets sent
425
    /// to the recipient.
426
    (ContentLength, CONTENT_LENGTH, b"content-length");
427
428
    /// Indicates an alternate location for the returned data.
429
    ///
430
    /// The principal use case is to indicate the URL of the resource
431
    /// transmitted as the result of content negotiation.
432
    ///
433
    /// Location and Content-Location are different: Location indicates the
434
    /// target of a redirection (or the URL of a newly created document), while
435
    /// Content-Location indicates the direct URL to use to access the resource,
436
    /// without the need of further content negotiation. Location is a header
437
    /// associated with the response, while Content-Location is associated with
438
    /// the entity returned.
439
    (ContentLocation, CONTENT_LOCATION, b"content-location");
440
441
    /// Indicates where in a full body message a partial message belongs.
442
    (ContentRange, CONTENT_RANGE, b"content-range");
443
444
    /// Allows controlling resources the user agent is allowed to load for a
445
    /// given page.
446
    ///
447
    /// With a few exceptions, policies mostly involve specifying server origins
448
    /// and script endpoints. This helps guard against cross-site scripting
449
    /// attacks (XSS).
450
    (ContentSecurityPolicy, CONTENT_SECURITY_POLICY, b"content-security-policy");
451
452
    /// Allows experimenting with policies by monitoring their effects.
453
    ///
454
    /// The HTTP Content-Security-Policy-Report-Only response header allows web
455
    /// developers to experiment with policies by monitoring (but not enforcing)
456
    /// their effects. These violation reports consist of JSON documents sent
457
    /// via an HTTP POST request to the specified URI.
458
    (ContentSecurityPolicyReportOnly, CONTENT_SECURITY_POLICY_REPORT_ONLY, b"content-security-policy-report-only");
459
460
    /// Used to indicate the media type of the resource.
461
    ///
462
    /// In responses, a Content-Type header tells the client what the content
463
    /// type of the returned content actually is. Browsers will do MIME sniffing
464
    /// in some cases and will not necessarily follow the value of this header;
465
    /// to prevent this behavior, the header X-Content-Type-Options can be set
466
    /// to nosniff.
467
    ///
468
    /// In requests, (such as POST or PUT), the client tells the server what
469
    /// type of data is actually sent.
470
    (ContentType, CONTENT_TYPE, b"content-type");
471
472
    /// Contains stored HTTP cookies previously sent by the server with the
473
    /// Set-Cookie header.
474
    ///
475
    /// The Cookie header might be omitted entirely, if the privacy setting of
476
    /// the browser are set to block them, for example.
477
    (Cookie, COOKIE, b"cookie");
478
479
    /// Indicates the client's tracking preference.
480
    ///
481
    /// This header lets users indicate whether they would prefer privacy rather
482
    /// than personalized content.
483
    (Dnt, DNT, b"dnt");
484
485
    /// Contains the date and time at which the message was originated.
486
    (Date, DATE, b"date");
487
488
    /// Identifier for a specific version of a resource.
489
    ///
490
    /// This header allows caches to be more efficient, and saves bandwidth, as
491
    /// a web server does not need to send a full response if the content has
492
    /// not changed. On the other side, if the content has changed, etags are
493
    /// useful to help prevent simultaneous updates of a resource from
494
    /// overwriting each other ("mid-air collisions").
495
    ///
496
    /// If the resource at a given URL changes, a new Etag value must be
497
    /// generated. Etags are therefore similar to fingerprints and might also be
498
    /// used for tracking purposes by some servers. A comparison of them allows
499
    /// to quickly determine whether two representations of a resource are the
500
    /// same, but they might also be set to persist indefinitely by a tracking
501
    /// server.
502
    (Etag, ETAG, b"etag");
503
504
    /// Indicates expectations that need to be fulfilled by the server in order
505
    /// to properly handle the request.
506
    ///
507
    /// The only expectation defined in the specification is Expect:
508
    /// 100-continue, to which the server shall respond with:
509
    ///
510
    /// * 100 if the information contained in the header is sufficient to cause
511
    /// an immediate success,
512
    ///
513
    /// * 417 (Expectation Failed) if it cannot meet the expectation; or any
514
    /// other 4xx status otherwise.
515
    ///
516
    /// For example, the server may reject a request if its Content-Length is
517
    /// too large.
518
    ///
519
    /// No common browsers send the Expect header, but some other clients such
520
    /// as cURL do so by default.
521
    (Expect, EXPECT, b"expect");
522
523
    /// Contains the date/time after which the response is considered stale.
524
    ///
525
    /// Invalid dates, like the value 0, represent a date in the past and mean
526
    /// that the resource is already expired.
527
    ///
528
    /// If there is a Cache-Control header with the "max-age" or "s-max-age"
529
    /// directive in the response, the Expires header is ignored.
530
    (Expires, EXPIRES, b"expires");
531
532
    /// Contains information from the client-facing side of proxy servers that
533
    /// is altered or lost when a proxy is involved in the path of the request.
534
    ///
535
    /// The alternative and de-facto standard versions of this header are the
536
    /// X-Forwarded-For, X-Forwarded-Host and X-Forwarded-Proto headers.
537
    ///
538
    /// This header is used for debugging, statistics, and generating
539
    /// location-dependent content and by design it exposes privacy sensitive
540
    /// information, such as the IP address of the client. Therefore the user's
541
    /// privacy must be kept in mind when deploying this header.
542
    (Forwarded, FORWARDED, b"forwarded");
543
544
    /// Contains an Internet email address for a human user who controls the
545
    /// requesting user agent.
546
    ///
547
    /// If you are running a robotic user agent (e.g. a crawler), the From
548
    /// header should be sent, so you can be contacted if problems occur on
549
    /// servers, such as if the robot is sending excessive, unwanted, or invalid
550
    /// requests.
551
    (From, FROM, b"from");
552
553
    /// Specifies the domain name of the server and (optionally) the TCP port
554
    /// number on which the server is listening.
555
    ///
556
    /// If no port is given, the default port for the service requested (e.g.,
557
    /// "80" for an HTTP URL) is implied.
558
    ///
559
    /// A Host header field must be sent in all HTTP/1.1 request messages. A 400
560
    /// (Bad Request) status code will be sent to any HTTP/1.1 request message
561
    /// that lacks a Host header field or contains more than one.
562
    (Host, HOST, b"host");
563
564
    /// Makes a request conditional based on the E-Tag.
565
    ///
566
    /// For GET and HEAD methods, the server will send back the requested
567
    /// resource only if it matches one of the listed ETags. For PUT and other
568
    /// non-safe methods, it will only upload the resource in this case.
569
    ///
570
    /// The comparison with the stored ETag uses the strong comparison
571
    /// algorithm, meaning two files are considered identical byte to byte only.
572
    /// This is weakened when the  W/ prefix is used in front of the ETag.
573
    ///
574
    /// There are two common use cases:
575
    ///
576
    /// * For GET and HEAD methods, used in combination with an Range header, it
577
    /// can guarantee that the new ranges requested comes from the same resource
578
    /// than the previous one. If it doesn't match, then a 416 (Range Not
579
    /// Satisfiable) response is returned.
580
    ///
581
    /// * For other methods, and in particular for PUT, If-Match can be used to
582
    /// prevent the lost update problem. It can check if the modification of a
583
    /// resource that the user wants to upload will not override another change
584
    /// that has been done since the original resource was fetched. If the
585
    /// request cannot be fulfilled, the 412 (Precondition Failed) response is
586
    /// returned.
587
    (IfMatch, IF_MATCH, b"if-match");
588
589
    /// Makes a request conditional based on the modification date.
590
    ///
591
    /// The If-Modified-Since request HTTP header makes the request conditional:
592
    /// the server will send back the requested resource, with a 200 status,
593
    /// only if it has been last modified after the given date. If the request
594
    /// has not been modified since, the response will be a 304 without any
595
    /// body; the Last-Modified header will contain the date of last
596
    /// modification. Unlike If-Unmodified-Since, If-Modified-Since can only be
597
    /// used with a GET or HEAD.
598
    ///
599
    /// When used in combination with If-None-Match, it is ignored, unless the
600
    /// server doesn't support If-None-Match.
601
    ///
602
    /// The most common use case is to update a cached entity that has no
603
    /// associated ETag.
604
    (IfModifiedSince, IF_MODIFIED_SINCE, b"if-modified-since");
605
606
    /// Makes a request conditional based on the E-Tag.
607
    ///
608
    /// The If-None-Match HTTP request header makes the request conditional. For
609
    /// GET and HEAD methods, the server will send back the requested resource,
610
    /// with a 200 status, only if it doesn't have an ETag matching the given
611
    /// ones. For other methods, the request will be processed only if the
612
    /// eventually existing resource's ETag doesn't match any of the values
613
    /// listed.
614
    ///
615
    /// When the condition fails for GET and HEAD methods, then the server must
616
    /// return HTTP status code 304 (Not Modified). For methods that apply
617
    /// server-side changes, the status code 412 (Precondition Failed) is used.
618
    /// Note that the server generating a 304 response MUST generate any of the
619
    /// following header fields that would have been sent in a 200 (OK) response
620
    /// to the same request: Cache-Control, Content-Location, Date, ETag,
621
    /// Expires, and Vary.
622
    ///
623
    /// The comparison with the stored ETag uses the weak comparison algorithm,
624
    /// meaning two files are considered identical not only if they are
625
    /// identical byte to byte, but if the content is equivalent. For example,
626
    /// two pages that would differ only by the date of generation in the footer
627
    /// would be considered as identical.
628
    ///
629
    /// When used in combination with If-Modified-Since, it has precedence (if
630
    /// the server supports it).
631
    ///
632
    /// There are two common use cases:
633
    ///
634
    /// * For `GET` and `HEAD` methods, to update a cached entity that has an associated ETag.
635
    /// * For other methods, and in particular for `PUT`, `If-None-Match` used with
636
    /// the `*` value can be used to save a file not known to exist,
637
    /// guaranteeing that another upload didn't happen before, losing the data
638
    /// of the previous put; this problems is the variation of the lost update
639
    /// problem.
640
    (IfNoneMatch, IF_NONE_MATCH, b"if-none-match");
641
642
    /// Makes a request conditional based on range.
643
    ///
644
    /// The If-Range HTTP request header makes a range request conditional: if
645
    /// the condition is fulfilled, the range request will be issued and the
646
    /// server sends back a 206 Partial Content answer with the appropriate
647
    /// body. If the condition is not fulfilled, the full resource is sent back,
648
    /// with a 200 OK status.
649
    ///
650
    /// This header can be used either with a Last-Modified validator, or with
651
    /// an ETag, but not with both.
652
    ///
653
    /// The most common use case is to resume a download, to guarantee that the
654
    /// stored resource has not been modified since the last fragment has been
655
    /// received.
656
    (IfRange, IF_RANGE, b"if-range");
657
658
    /// Makes the request conditional based on the last modification date.
659
    ///
660
    /// The If-Unmodified-Since request HTTP header makes the request
661
    /// conditional: the server will send back the requested resource, or accept
662
    /// it in the case of a POST or another non-safe method, only if it has not
663
    /// been last modified after the given date. If the request has been
664
    /// modified after the given date, the response will be a 412 (Precondition
665
    /// Failed) error.
666
    ///
667
    /// There are two common use cases:
668
    ///
669
    /// * In conjunction non-safe methods, like POST, it can be used to
670
    /// implement an optimistic concurrency control, like done by some wikis:
671
    /// editions are rejected if the stored document has been modified since the
672
    /// original has been retrieved.
673
    ///
674
    /// * In conjunction with a range request with a If-Range header, it can be
675
    /// used to ensure that the new fragment requested comes from an unmodified
676
    /// document.
677
    (IfUnmodifiedSince, IF_UNMODIFIED_SINCE, b"if-unmodified-since");
678
679
    /// The Last-Modified header contains the date and time when the origin believes
680
    /// the resource was last modified.
681
    ///
682
    /// The value is a valid Date/Time string defined in [RFC9910](https://datatracker.ietf.org/doc/html/rfc9110#section-5.6.7)
683
    (LastModified, LAST_MODIFIED, b"last-modified");
684
685
    /// Allows the server to point an interested client to another resource
686
    /// containing metadata about the requested resource.
687
    (Link, LINK, b"link");
688
689
    /// Indicates the URL to redirect a page to.
690
    ///
691
    /// The Location response header indicates the URL to redirect a page to. It
692
    /// only provides a meaning when served with a 3xx status response.
693
    ///
694
    /// The HTTP method used to make the new request to fetch the page pointed
695
    /// to by Location depends of the original method and of the kind of
696
    /// redirection:
697
    ///
698
    /// * If 303 (See Also) responses always lead to the use of a GET method,
699
    /// 307 (Temporary Redirect) and 308 (Permanent Redirect) don't change the
700
    /// method used in the original request;
701
    ///
702
    /// * 301 (Permanent Redirect) and 302 (Found) doesn't change the method
703
    /// most of the time, though older user-agents may (so you basically don't
704
    /// know).
705
    ///
706
    /// All responses with one of these status codes send a Location header.
707
    ///
708
    /// Beside redirect response, messages with 201 (Created) status also
709
    /// include the Location header. It indicates the URL to the newly created
710
    /// resource.
711
    ///
712
    /// Location and Content-Location are different: Location indicates the
713
    /// target of a redirection (or the URL of a newly created resource), while
714
    /// Content-Location indicates the direct URL to use to access the resource
715
    /// when content negotiation happened, without the need of further content
716
    /// negotiation. Location is a header associated with the response, while
717
    /// Content-Location is associated with the entity returned.
718
    (Location, LOCATION, b"location");
719
720
    /// Indicates the max number of intermediaries the request should be sent
721
    /// through.
722
    (MaxForwards, MAX_FORWARDS, b"max-forwards");
723
724
    /// Indicates where a fetch originates from.
725
    ///
726
    /// It doesn't include any path information, but only the server name. It is
727
    /// sent with CORS requests, as well as with POST requests. It is similar to
728
    /// the Referer header, but, unlike this header, it doesn't disclose the
729
    /// whole path.
730
    (Origin, ORIGIN, b"origin");
731
732
    /// HTTP/1.0 header usually used for backwards compatibility.
733
    ///
734
    /// The Pragma HTTP/1.0 general header is an implementation-specific header
735
    /// that may have various effects along the request-response chain. It is
736
    /// used for backwards compatibility with HTTP/1.0 caches where the
737
    /// Cache-Control HTTP/1.1 header is not yet present.
738
    (Pragma, PRAGMA, b"pragma");
739
740
    /// Defines the authentication method that should be used to gain access to
741
    /// a proxy.
742
    ///
743
    /// Unlike `www-authenticate`, the `proxy-authenticate` header field applies
744
    /// only to the next outbound client on the response chain. This is because
745
    /// only the client that chose a given proxy is likely to have the
746
    /// credentials necessary for authentication. However, when multiple proxies
747
    /// are used within the same administrative domain, such as office and
748
    /// regional caching proxies within a large corporate network, it is common
749
    /// for credentials to be generated by the user agent and passed through the
750
    /// hierarchy until consumed. Hence, in such a configuration, it will appear
751
    /// as if Proxy-Authenticate is being forwarded because each proxy will send
752
    /// the same challenge set.
753
    ///
754
    /// The `proxy-authenticate` header is sent along with a `407 Proxy
755
    /// Authentication Required`.
756
    (ProxyAuthenticate, PROXY_AUTHENTICATE, b"proxy-authenticate");
757
758
    /// Contains the credentials to authenticate a user agent to a proxy server.
759
    ///
760
    /// This header is usually included after the server has responded with a
761
    /// 407 Proxy Authentication Required status and the Proxy-Authenticate
762
    /// header.
763
    (ProxyAuthorization, PROXY_AUTHORIZATION, b"proxy-authorization");
764
765
    /// Associates a specific cryptographic public key with a certain server.
766
    ///
767
    /// This decreases the risk of MITM attacks with forged certificates. If one
768
    /// or several keys are pinned and none of them are used by the server, the
769
    /// browser will not accept the response as legitimate, and will not display
770
    /// it.
771
    (PublicKeyPins, PUBLIC_KEY_PINS, b"public-key-pins");
772
773
    /// Sends reports of pinning violation to the report-uri specified in the
774
    /// header.
775
    ///
776
    /// Unlike `Public-Key-Pins`, this header still allows browsers to connect
777
    /// to the server if the pinning is violated.
778
    (PublicKeyPinsReportOnly, PUBLIC_KEY_PINS_REPORT_ONLY, b"public-key-pins-report-only");
779
780
    /// Indicates the part of a document that the server should return.
781
    ///
782
    /// Several parts can be requested with one Range header at once, and the
783
    /// server may send back these ranges in a multipart document. If the server
784
    /// sends back ranges, it uses the 206 Partial Content for the response. If
785
    /// the ranges are invalid, the server returns the 416 Range Not Satisfiable
786
    /// error. The server can also ignore the Range header and return the whole
787
    /// document with a 200 status code.
788
    (Range, RANGE, b"range");
789
790
    /// Contains the address of the previous web page from which a link to the
791
    /// currently requested page was followed.
792
    ///
793
    /// The Referer header allows servers to identify where people are visiting
794
    /// them from and may use that data for analytics, logging, or optimized
795
    /// caching, for example.
796
    (Referer, REFERER, b"referer");
797
798
    /// Governs which referrer information should be included with requests
799
    /// made.
800
    (ReferrerPolicy, REFERRER_POLICY, b"referrer-policy");
801
802
    /// Informs the web browser that the current page or frame should be
803
    /// refreshed.
804
    (Refresh, REFRESH, b"refresh");
805
806
    /// The Retry-After response HTTP header indicates how long the user agent
807
    /// should wait before making a follow-up request. There are two main cases
808
    /// this header is used:
809
    ///
810
    /// * When sent with a 503 (Service Unavailable) response, it indicates how
811
    /// long the service is expected to be unavailable.
812
    ///
813
    /// * When sent with a redirect response, such as 301 (Moved Permanently),
814
    /// it indicates the minimum time that the user agent is asked to wait
815
    /// before issuing the redirected request.
816
    (RetryAfter, RETRY_AFTER, b"retry-after");
817
818
    /// The |Sec-WebSocket-Accept| header field is used in the WebSocket
819
    /// opening handshake. It is sent from the server to the client to
820
    /// confirm that the server is willing to initiate the WebSocket
821
    /// connection.
822
    (SecWebSocketAccept, SEC_WEBSOCKET_ACCEPT, b"sec-websocket-accept");
823
824
    /// The |Sec-WebSocket-Extensions| header field is used in the WebSocket
825
    /// opening handshake. It is initially sent from the client to the
826
    /// server, and then subsequently sent from the server to the client, to
827
    /// agree on a set of protocol-level extensions to use for the duration
828
    /// of the connection.
829
    (SecWebSocketExtensions, SEC_WEBSOCKET_EXTENSIONS, b"sec-websocket-extensions");
830
831
    /// The |Sec-WebSocket-Key| header field is used in the WebSocket opening
832
    /// handshake. It is sent from the client to the server to provide part
833
    /// of the information used by the server to prove that it received a
834
    /// valid WebSocket opening handshake. This helps ensure that the server
835
    /// does not accept connections from non-WebSocket clients (e.g., HTTP
836
    /// clients) that are being abused to send data to unsuspecting WebSocket
837
    /// servers.
838
    (SecWebSocketKey, SEC_WEBSOCKET_KEY, b"sec-websocket-key");
839
840
    /// The |Sec-WebSocket-Protocol| header field is used in the WebSocket
841
    /// opening handshake. It is sent from the client to the server and back
842
    /// from the server to the client to confirm the subprotocol of the
843
    /// connection.  This enables scripts to both select a subprotocol and be
844
    /// sure that the server agreed to serve that subprotocol.
845
    (SecWebSocketProtocol, SEC_WEBSOCKET_PROTOCOL, b"sec-websocket-protocol");
846
847
    /// The |Sec-WebSocket-Version| header field is used in the WebSocket
848
    /// opening handshake.  It is sent from the client to the server to
849
    /// indicate the protocol version of the connection.  This enables
850
    /// servers to correctly interpret the opening handshake and subsequent
851
    /// data being sent from the data, and close the connection if the server
852
    /// cannot interpret that data in a safe manner.
853
    (SecWebSocketVersion, SEC_WEBSOCKET_VERSION, b"sec-websocket-version");
854
855
    /// Contains information about the software used by the origin server to
856
    /// handle the request.
857
    ///
858
    /// Overly long and detailed Server values should be avoided as they
859
    /// potentially reveal internal implementation details that might make it
860
    /// (slightly) easier for attackers to find and exploit known security
861
    /// holes.
862
    (Server, SERVER, b"server");
863
864
    /// Used to send cookies from the server to the user agent.
865
    (SetCookie, SET_COOKIE, b"set-cookie");
866
867
    /// Tells the client to communicate with HTTPS instead of using HTTP.
868
    (StrictTransportSecurity, STRICT_TRANSPORT_SECURITY, b"strict-transport-security");
869
870
    /// Informs the server of transfer encodings willing to be accepted as part
871
    /// of the response.
872
    ///
873
    /// See also the Transfer-Encoding response header for more details on
874
    /// transfer encodings. Note that chunked is always acceptable for HTTP/1.1
875
    /// recipients and you that don't have to specify "chunked" using the TE
876
    /// header. However, it is useful for setting if the client is accepting
877
    /// trailer fields in a chunked transfer coding using the "trailers" value.
878
    (Te, TE, b"te");
879
880
    /// Allows the sender to include additional fields at the end of chunked
881
    /// messages.
882
    (Trailer, TRAILER, b"trailer");
883
884
    /// Specifies the form of encoding used to safely transfer the entity to the
885
    /// client.
886
    ///
887
    /// `transfer-encoding` is a hop-by-hop header, that is applying to a
888
    /// message between two nodes, not to a resource itself. Each segment of a
889
    /// multi-node connection can use different `transfer-encoding` values. If
890
    /// you want to compress data over the whole connection, use the end-to-end
891
    /// header `content-encoding` header instead.
892
    ///
893
    /// When present on a response to a `HEAD` request that has no body, it
894
    /// indicates the value that would have applied to the corresponding `GET`
895
    /// message.
896
    (TransferEncoding, TRANSFER_ENCODING, b"transfer-encoding");
897
898
    /// Contains a string that allows identifying the requesting client's
899
    /// software.
900
    (UserAgent, USER_AGENT, b"user-agent");
901
902
    /// Used as part of the exchange to upgrade the protocol.
903
    (Upgrade, UPGRADE, b"upgrade");
904
905
    /// Sends a signal to the server expressing the client’s preference for an
906
    /// encrypted and authenticated response.
907
    (UpgradeInsecureRequests, UPGRADE_INSECURE_REQUESTS, b"upgrade-insecure-requests");
908
909
    /// Determines how to match future requests with cached responses.
910
    ///
911
    /// The `vary` HTTP response header determines how to match future request
912
    /// headers to decide whether a cached response can be used rather than
913
    /// requesting a fresh one from the origin server. It is used by the server
914
    /// to indicate which headers it used when selecting a representation of a
915
    /// resource in a content negotiation algorithm.
916
    ///
917
    /// The `vary` header should be set on a 304 Not Modified response exactly
918
    /// like it would have been set on an equivalent 200 OK response.
919
    (Vary, VARY, b"vary");
920
921
    /// Added by proxies to track routing.
922
    ///
923
    /// The `via` general header is added by proxies, both forward and reverse
924
    /// proxies, and can appear in the request headers and the response headers.
925
    /// It is used for tracking message forwards, avoiding request loops, and
926
    /// identifying the protocol capabilities of senders along the
927
    /// request/response chain.
928
    (Via, VIA, b"via");
929
930
    /// General HTTP header contains information about possible problems with
931
    /// the status of the message.
932
    ///
933
    /// More than one `warning` header may appear in a response. Warning header
934
    /// fields can in general be applied to any message, however some warn-codes
935
    /// are specific to caches and can only be applied to response messages.
936
    (Warning, WARNING, b"warning");
937
938
    /// Defines the authentication method that should be used to gain access to
939
    /// a resource.
940
    (WwwAuthenticate, WWW_AUTHENTICATE, b"www-authenticate");
941
942
    /// Marker used by the server to indicate that the MIME types advertised in
943
    /// the `content-type` headers should not be changed and be followed.
944
    ///
945
    /// This allows to opt-out of MIME type sniffing, or, in other words, it is
946
    /// a way to say that the webmasters knew what they were doing.
947
    ///
948
    /// This header was introduced by Microsoft in IE 8 as a way for webmasters
949
    /// to block content sniffing that was happening and could transform
950
    /// non-executable MIME types into executable MIME types. Since then, other
951
    /// browsers have introduced it, even if their MIME sniffing algorithms were
952
    /// less aggressive.
953
    ///
954
    /// Site security testers usually expect this header to be set.
955
    (XContentTypeOptions, X_CONTENT_TYPE_OPTIONS, b"x-content-type-options");
956
957
    /// Controls DNS prefetching.
958
    ///
959
    /// The `x-dns-prefetch-control` HTTP response header controls DNS
960
    /// prefetching, a feature by which browsers proactively perform domain name
961
    /// resolution on both links that the user may choose to follow as well as
962
    /// URLs for items referenced by the document, including images, CSS,
963
    /// JavaScript, and so forth.
964
    ///
965
    /// This prefetching is performed in the background, so that the DNS is
966
    /// likely to have been resolved by the time the referenced items are
967
    /// needed. This reduces latency when the user clicks a link.
968
    (XDnsPrefetchControl, X_DNS_PREFETCH_CONTROL, b"x-dns-prefetch-control");
969
970
    /// Indicates whether or not a browser should be allowed to render a page in
971
    /// a frame.
972
    ///
973
    /// Sites can use this to avoid clickjacking attacks, by ensuring that their
974
    /// content is not embedded into other sites.
975
    ///
976
    /// The added security is only provided if the user accessing the document
977
    /// is using a browser supporting `x-frame-options`.
978
    (XFrameOptions, X_FRAME_OPTIONS, b"x-frame-options");
979
980
    /// Stop pages from loading when an XSS attack is detected.
981
    ///
982
    /// The HTTP X-XSS-Protection response header is a feature of Internet
983
    /// Explorer, Chrome and Safari that stops pages from loading when they
984
    /// detect reflected cross-site scripting (XSS) attacks. Although these
985
    /// protections are largely unnecessary in modern browsers when sites
986
    /// implement a strong Content-Security-Policy that disables the use of
987
    /// inline JavaScript ('unsafe-inline'), they can still provide protections
988
    /// for users of older web browsers that don't yet support CSP.
989
    (XXssProtection, X_XSS_PROTECTION, b"x-xss-protection");
990
}
991
992
/// Valid header name characters
993
///
994
/// ```not_rust
995
///       field-name     = token
996
///       separators     = "(" | ")" | "<" | ">" | "@"
997
///                      | "," | ";" | ":" | "\" | <">
998
///                      | "/" | "[" | "]" | "?" | "="
999
///                      | "{" | "}" | SP | HT
1000
///       token          = 1*tchar
1001
///       tchar          = "!" / "#" / "$" / "%" / "&" / "'" / "*"
1002
///                      / "+" / "-" / "." / "^" / "_" / "`" / "|" / "~"
1003
///                      / DIGIT / ALPHA
1004
///                      ; any VCHAR, except delimiters
1005
/// ```
1006
// HEADER_CHARS maps every byte that is 128 or larger to 0 so everything that is
1007
// mapped by HEADER_CHARS, maps to a valid single-byte UTF-8 codepoint.
1008
#[rustfmt::skip]
1009
const HEADER_CHARS: [u8; 256] = [
1010
    //  0      1      2      3      4      5      6      7      8      9
1011
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, //   x
1012
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, //  1x
1013
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, //  2x
1014
        0,     0,     0,  b'!',     0,  b'#',  b'$',  b'%',  b'&', b'\'', //  3x
1015
        0,     0,  b'*',  b'+',     0,  b'-',  b'.',     0,  b'0',  b'1', //  4x
1016
     b'2',  b'3',  b'4',  b'5',  b'6',  b'7',  b'8',  b'9',     0,     0, //  5x
1017
        0,     0,     0,     0,     0,  b'a',  b'b',  b'c',  b'd',  b'e', //  6x
1018
     b'f',  b'g',  b'h',  b'i',  b'j',  b'k',  b'l',  b'm',  b'n',  b'o', //  7x
1019
     b'p',  b'q',  b'r',  b's',  b't',  b'u',  b'v',  b'w',  b'x',  b'y', //  8x
1020
     b'z',     0,     0,     0,  b'^',  b'_',  b'`',  b'a',  b'b',  b'c', //  9x
1021
     b'd',  b'e',  b'f',  b'g',  b'h',  b'i',  b'j',  b'k',  b'l',  b'm', // 10x
1022
     b'n',  b'o',  b'p',  b'q',  b'r',  b's',  b't',  b'u',  b'v',  b'w', // 11x
1023
     b'x',  b'y',  b'z',     0,  b'|',     0,  b'~',     0,     0,     0, // 12x
1024
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 13x
1025
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 14x
1026
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 15x
1027
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 16x
1028
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 17x
1029
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 18x
1030
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 19x
1031
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 20x
1032
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 21x
1033
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 22x
1034
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 23x
1035
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 24x
1036
        0,     0,     0,     0,     0,     0                              // 25x
1037
];
1038
1039
/// Valid header name characters for HTTP/2.0 and HTTP/3.0
1040
// HEADER_CHARS_H2 maps every byte that is 128 or larger to 0 so everything that is
1041
// mapped by HEADER_CHARS_H2, maps to a valid single-byte UTF-8 codepoint.
1042
#[rustfmt::skip]
1043
const HEADER_CHARS_H2: [u8; 256] = [
1044
    //  0      1      2      3      4      5      6      7      8      9
1045
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, //   x
1046
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, //  1x
1047
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, //  2x
1048
        0,     0,     0,  b'!',  b'"',  b'#',  b'$',  b'%',  b'&', b'\'', //  3x
1049
        0,     0,  b'*',  b'+',     0,  b'-',  b'.',     0,  b'0',  b'1', //  4x
1050
     b'2',  b'3',  b'4',  b'5',  b'6',  b'7',  b'8',  b'9',     0,     0, //  5x
1051
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, //  6x
1052
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, //  7x
1053
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, //  8x
1054
        0,     0,     0,     0,  b'^',  b'_',  b'`',  b'a',  b'b',  b'c', //  9x
1055
     b'd',  b'e',  b'f',  b'g',  b'h',  b'i',  b'j',  b'k',  b'l',  b'm', // 10x
1056
     b'n',  b'o',  b'p',  b'q',  b'r',  b's',  b't',  b'u',  b'v',  b'w', // 11x
1057
     b'x',  b'y',  b'z',     0,  b'|',     0,  b'~',     0,     0,     0, // 12x
1058
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 13x
1059
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 14x
1060
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 15x
1061
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 16x
1062
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 17x
1063
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 18x
1064
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 19x
1065
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 20x
1066
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 21x
1067
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 22x
1068
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 23x
1069
        0,     0,     0,     0,     0,     0,     0,     0,     0,     0, // 24x
1070
        0,     0,     0,     0,     0,     0                              // 25x
1071
];
1072
1073
7.35k
fn parse_hdr<'a>(
1074
7.35k
    data: &'a [u8],
1075
7.35k
    b: &'a mut [MaybeUninit<u8>; SCRATCH_BUF_SIZE],
1076
7.35k
    table: &[u8; 256],
1077
7.35k
) -> Result<HdrName<'a>, InvalidHeaderName> {
1078
7.35k
    match data.len() {
1079
1.00k
        0 => Err(InvalidHeaderName::new()),
1080
6.35k
        len @ 1..=SCRATCH_BUF_SIZE => {
1081
            // Read from data into the buffer - transforming using `table` as we go
1082
6.25k
            data.iter()
1083
6.25k
                .zip(b.iter_mut())
1084
104k
                .for_each(|(index, out)| *out = MaybeUninit::new(table[*index as usize]));
1085
            // Safety: len bytes of b were just initialized.
1086
6.25k
            let name: &'a [u8] = unsafe { slice_assume_init(&b[0..len]) };
1087
6.25k
            match StandardHeader::from_bytes(name) {
1088
252
                Some(sh) => Ok(sh.into()),
1089
                None => {
1090
6.00k
                    if name.contains(&0) {
1091
5.49k
                        Err(InvalidHeaderName::new())
1092
                    } else {
1093
506
                        Ok(HdrName::custom(name, true))
1094
                    }
1095
                }
1096
            }
1097
        }
1098
103
        SCRATCH_BUF_OVERFLOW..=super::MAX_HEADER_NAME_LEN => Ok(HdrName::custom(data, false)),
1099
13
        _ => Err(InvalidHeaderName::new()),
1100
    }
1101
7.35k
}
1102
1103
impl<'a> From<StandardHeader> for HdrName<'a> {
1104
252
    fn from(hdr: StandardHeader) -> HdrName<'a> {
1105
252
        HdrName {
1106
252
            inner: Repr::Standard(hdr),
1107
252
        }
1108
252
    }
1109
}
1110
1111
impl HeaderName {
1112
    /// Converts a slice of bytes to an HTTP header name.
1113
    ///
1114
    /// This function normalizes the input.
1115
7.35k
    pub fn from_bytes(src: &[u8]) -> Result<HeaderName, InvalidHeaderName> {
1116
7.35k
        let mut buf = uninit_u8_array();
1117
        // Precondition: HEADER_CHARS is a valid table for parse_hdr().
1118
7.35k
        match parse_hdr(src, &mut buf, &HEADER_CHARS)?.inner {
1119
252
            Repr::Standard(std) => Ok(std.into()),
1120
506
            Repr::Custom(MaybeLower { buf, lower: true }) => {
1121
506
                let buf = Bytes::copy_from_slice(buf);
1122
                // Safety: the invariant on MaybeLower ensures buf is valid UTF-8.
1123
506
                let val = unsafe { ByteStr::from_utf8_unchecked(buf) };
1124
506
                Ok(Custom(val).into())
1125
            }
1126
90
            Repr::Custom(MaybeLower { buf, lower: false }) => {
1127
                use bytes::BufMut;
1128
90
                let mut dst = BytesMut::with_capacity(buf.len());
1129
1130
253k
                for b in buf.iter() {
1131
                    // HEADER_CHARS maps all bytes to valid single-byte UTF-8
1132
253k
                    let b = HEADER_CHARS[*b as usize];
1133
1134
253k
                    if b == 0 {
1135
59
                        return Err(InvalidHeaderName::new());
1136
253k
                    }
1137
1138
253k
                    dst.put_u8(b);
1139
                }
1140
1141
                // Safety: the loop above maps all bytes in buf to valid single byte
1142
                // UTF-8 before copying them into dst. This means that dst (and hence
1143
                // dst.freeze()) is valid UTF-8.
1144
31
                let val = unsafe { ByteStr::from_utf8_unchecked(dst.freeze()) };
1145
1146
31
                Ok(Custom(val).into())
1147
            }
1148
        }
1149
7.35k
    }
1150
1151
    /// Converts a slice of bytes to an HTTP header name.
1152
    ///
1153
    /// This function expects the input to only contain lowercase characters.
1154
    /// This is useful when decoding HTTP/2.0 or HTTP/3.0 headers. Both
1155
    /// require that all headers be represented in lower case.
1156
    ///
1157
    /// # Examples
1158
    ///
1159
    /// ```
1160
    /// # use http::header::*;
1161
    ///
1162
    /// // Parsing a lower case header
1163
    /// let hdr = HeaderName::from_lowercase(b"content-length").unwrap();
1164
    /// assert_eq!(CONTENT_LENGTH, hdr);
1165
    ///
1166
    /// // Parsing a header that contains uppercase characters
1167
    /// assert!(HeaderName::from_lowercase(b"Content-Length").is_err());
1168
    /// ```
1169
0
    pub fn from_lowercase(src: &[u8]) -> Result<HeaderName, InvalidHeaderName> {
1170
0
        let mut buf = uninit_u8_array();
1171
        // Precondition: HEADER_CHARS_H2 is a valid table for parse_hdr()
1172
0
        match parse_hdr(src, &mut buf, &HEADER_CHARS_H2)?.inner {
1173
0
            Repr::Standard(std) => Ok(std.into()),
1174
0
            Repr::Custom(MaybeLower { buf, lower: true }) => {
1175
0
                let buf = Bytes::copy_from_slice(buf);
1176
                // Safety: the invariant on MaybeLower ensures buf is valid UTF-8.
1177
0
                let val = unsafe { ByteStr::from_utf8_unchecked(buf) };
1178
0
                Ok(Custom(val).into())
1179
            }
1180
0
            Repr::Custom(MaybeLower { buf, lower: false }) => {
1181
0
                for &b in buf.iter() {
1182
                    // HEADER_CHARS_H2 maps all bytes that are not valid single-byte
1183
                    // UTF-8 to 0 so this check returns an error for invalid UTF-8.
1184
0
                    if HEADER_CHARS_H2[b as usize] == 0 {
1185
0
                        return Err(InvalidHeaderName::new());
1186
0
                    }
1187
                }
1188
1189
0
                let buf = Bytes::copy_from_slice(buf);
1190
                // Safety: the loop above checks that each byte of buf (either
1191
                // version) is valid UTF-8.
1192
0
                let val = unsafe { ByteStr::from_utf8_unchecked(buf) };
1193
0
                Ok(Custom(val).into())
1194
            }
1195
        }
1196
0
    }
1197
1198
    /// Converts a static string to a HTTP header name.
1199
    ///
1200
    /// This function requires the static string to only contain lowercase
1201
    /// characters, numerals and symbols, as per the HTTP/2.0 specification
1202
    /// and header names internal representation within this library.
1203
    ///
1204
    /// # Panics
1205
    ///
1206
    /// This function panics when the static string is a invalid header.
1207
    ///
1208
    /// # Examples
1209
    ///
1210
    /// ```
1211
    /// # use http::header::*;
1212
    /// // Parsing a standard header
1213
    /// let hdr = HeaderName::from_static("content-length");
1214
    /// assert_eq!(CONTENT_LENGTH, hdr);
1215
    ///
1216
    /// // Parsing a custom header
1217
    /// let CUSTOM_HEADER: &'static str = "custom-header";
1218
    ///
1219
    /// let a = HeaderName::from_lowercase(b"custom-header").unwrap();
1220
    /// let b = HeaderName::from_static(CUSTOM_HEADER);
1221
    /// assert_eq!(a, b);
1222
    /// ```
1223
    ///
1224
    /// ```should_panic
1225
    /// # use http::header::*;
1226
    /// #
1227
    /// // Parsing a header that contains invalid symbols:
1228
    /// HeaderName::from_static("content{}{}length"); // This line panics!
1229
    ///
1230
    /// // Parsing a header that contains invalid uppercase characters.
1231
    /// let a = HeaderName::from_static("foobar");
1232
    /// let b = HeaderName::from_static("FOOBAR"); // This line panics!
1233
    /// ```
1234
0
    pub const fn from_static(src: &'static str) -> HeaderName {
1235
0
        let name_bytes = src.as_bytes();
1236
0
        if let Some(standard) = StandardHeader::from_bytes(name_bytes) {
1237
0
            return HeaderName {
1238
0
                inner: Repr::Standard(standard),
1239
0
            };
1240
0
        }
1241
1242
0
        if name_bytes.is_empty() || name_bytes.len() > super::MAX_HEADER_NAME_LEN || {
1243
0
            let mut i = 0;
1244
            loop {
1245
0
                if i >= name_bytes.len() {
1246
0
                    break false;
1247
0
                } else if HEADER_CHARS_H2[name_bytes[i] as usize] == 0 {
1248
0
                    break true;
1249
0
                }
1250
0
                i += 1;
1251
            }
1252
        } {
1253
            // Invalid header name
1254
0
            panic!("HeaderName::from_static with invalid bytes")
1255
0
        }
1256
1257
0
        HeaderName {
1258
0
            inner: Repr::Custom(Custom(ByteStr::from_static(src))),
1259
0
        }
1260
0
    }
1261
1262
    /// Returns a `str` representation of the header.
1263
    ///
1264
    /// The returned string will always be lower case.
1265
    #[inline]
1266
0
    pub fn as_str(&self) -> &str {
1267
0
        match self.inner {
1268
0
            Repr::Standard(v) => v.as_str(),
1269
0
            Repr::Custom(ref v) => &v.0,
1270
        }
1271
0
    }
1272
1273
0
    pub(super) fn into_bytes(self) -> Bytes {
1274
0
        self.inner.into()
1275
0
    }
1276
}
1277
1278
impl FromStr for HeaderName {
1279
    type Err = InvalidHeaderName;
1280
1281
0
    fn from_str(s: &str) -> Result<HeaderName, InvalidHeaderName> {
1282
0
        HeaderName::from_bytes(s.as_bytes()).map_err(|_| InvalidHeaderName { _priv: () })
1283
0
    }
1284
}
1285
1286
impl AsRef<str> for HeaderName {
1287
0
    fn as_ref(&self) -> &str {
1288
0
        self.as_str()
1289
0
    }
1290
}
1291
1292
impl AsRef<[u8]> for HeaderName {
1293
0
    fn as_ref(&self) -> &[u8] {
1294
0
        self.as_str().as_bytes()
1295
0
    }
1296
}
1297
1298
impl Borrow<str> for HeaderName {
1299
0
    fn borrow(&self) -> &str {
1300
0
        self.as_str()
1301
0
    }
1302
}
1303
1304
impl fmt::Debug for HeaderName {
1305
0
    fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
1306
0
        fmt::Debug::fmt(self.as_str(), fmt)
1307
0
    }
1308
}
1309
1310
impl fmt::Display for HeaderName {
1311
0
    fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
1312
0
        fmt::Display::fmt(self.as_str(), fmt)
1313
0
    }
1314
}
1315
1316
impl InvalidHeaderName {
1317
6.56k
    pub(super) fn new() -> InvalidHeaderName {
1318
6.56k
        InvalidHeaderName { _priv: () }
1319
6.56k
    }
1320
}
1321
1322
impl From<&HeaderName> for HeaderName {
1323
0
    fn from(src: &HeaderName) -> HeaderName {
1324
0
        src.clone()
1325
0
    }
1326
}
1327
1328
#[doc(hidden)]
1329
impl<T> From<Repr<T>> for Bytes
1330
where
1331
    T: Into<Bytes>,
1332
{
1333
0
    fn from(repr: Repr<T>) -> Bytes {
1334
0
        match repr {
1335
0
            Repr::Standard(header) => Bytes::from_static(header.as_str().as_bytes()),
1336
0
            Repr::Custom(header) => header.into(),
1337
        }
1338
0
    }
1339
}
1340
1341
impl From<Custom> for Bytes {
1342
    #[inline]
1343
0
    fn from(Custom(inner): Custom) -> Bytes {
1344
0
        Bytes::from(inner)
1345
0
    }
1346
}
1347
1348
impl TryFrom<&str> for HeaderName {
1349
    type Error = InvalidHeaderName;
1350
    #[inline]
1351
    fn try_from(s: &str) -> Result<Self, Self::Error> {
1352
        Self::from_bytes(s.as_bytes())
1353
    }
1354
}
1355
1356
impl TryFrom<&String> for HeaderName {
1357
    type Error = InvalidHeaderName;
1358
    #[inline]
1359
    fn try_from(s: &String) -> Result<Self, Self::Error> {
1360
        Self::from_bytes(s.as_bytes())
1361
    }
1362
}
1363
1364
impl TryFrom<&[u8]> for HeaderName {
1365
    type Error = InvalidHeaderName;
1366
    #[inline]
1367
7.35k
    fn try_from(s: &[u8]) -> Result<Self, Self::Error> {
1368
7.35k
        Self::from_bytes(s)
1369
7.35k
    }
1370
}
1371
1372
impl TryFrom<String> for HeaderName {
1373
    type Error = InvalidHeaderName;
1374
1375
    #[inline]
1376
    fn try_from(s: String) -> Result<Self, Self::Error> {
1377
        Self::from_bytes(s.as_bytes())
1378
    }
1379
}
1380
1381
impl TryFrom<Vec<u8>> for HeaderName {
1382
    type Error = InvalidHeaderName;
1383
1384
    #[inline]
1385
    fn try_from(vec: Vec<u8>) -> Result<Self, Self::Error> {
1386
        Self::from_bytes(&vec)
1387
    }
1388
}
1389
1390
#[doc(hidden)]
1391
impl From<StandardHeader> for HeaderName {
1392
252
    fn from(src: StandardHeader) -> HeaderName {
1393
252
        HeaderName {
1394
252
            inner: Repr::Standard(src),
1395
252
        }
1396
252
    }
1397
}
1398
1399
#[doc(hidden)]
1400
impl From<Custom> for HeaderName {
1401
537
    fn from(src: Custom) -> HeaderName {
1402
537
        HeaderName {
1403
537
            inner: Repr::Custom(src),
1404
537
        }
1405
537
    }
1406
}
1407
1408
impl PartialEq<&HeaderName> for HeaderName {
1409
    #[inline]
1410
    fn eq(&self, other: &&HeaderName) -> bool {
1411
        *self == **other
1412
    }
1413
}
1414
1415
impl PartialEq<HeaderName> for &HeaderName {
1416
    #[inline]
1417
    fn eq(&self, other: &HeaderName) -> bool {
1418
        *other == *self
1419
    }
1420
}
1421
1422
impl PartialEq<str> for HeaderName {
1423
    /// Performs a case-insensitive comparison of the string against the header
1424
    /// name
1425
    ///
1426
    /// # Examples
1427
    ///
1428
    /// ```
1429
    /// use http::header::CONTENT_LENGTH;
1430
    ///
1431
    /// assert_eq!(CONTENT_LENGTH, "content-length");
1432
    /// assert_eq!(CONTENT_LENGTH, "Content-Length");
1433
    /// assert_ne!(CONTENT_LENGTH, "content length");
1434
    /// ```
1435
    #[inline]
1436
    fn eq(&self, other: &str) -> bool {
1437
        eq_ignore_ascii_case(self.as_ref(), other.as_bytes())
1438
    }
1439
}
1440
1441
impl PartialEq<HeaderName> for str {
1442
    /// Performs a case-insensitive comparison of the string against the header
1443
    /// name
1444
    ///
1445
    /// # Examples
1446
    ///
1447
    /// ```
1448
    /// use http::header::CONTENT_LENGTH;
1449
    ///
1450
    /// assert_eq!(CONTENT_LENGTH, "content-length");
1451
    /// assert_eq!(CONTENT_LENGTH, "Content-Length");
1452
    /// assert_ne!(CONTENT_LENGTH, "content length");
1453
    /// ```
1454
    #[inline]
1455
    fn eq(&self, other: &HeaderName) -> bool {
1456
        *other == *self
1457
    }
1458
}
1459
1460
impl PartialEq<&str> for HeaderName {
1461
    /// Performs a case-insensitive comparison of the string against the header
1462
    /// name
1463
    #[inline]
1464
    fn eq(&self, other: &&str) -> bool {
1465
        *self == **other
1466
    }
1467
}
1468
1469
impl PartialEq<HeaderName> for &str {
1470
    /// Performs a case-insensitive comparison of the string against the header
1471
    /// name
1472
    #[inline]
1473
    fn eq(&self, other: &HeaderName) -> bool {
1474
        *other == *self
1475
    }
1476
}
1477
1478
impl fmt::Debug for InvalidHeaderName {
1479
0
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
1480
0
        f.debug_struct("InvalidHeaderName")
1481
            // skip _priv noise
1482
0
            .finish()
1483
0
    }
1484
}
1485
1486
impl fmt::Display for InvalidHeaderName {
1487
0
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1488
0
        f.write_str("invalid HTTP header name")
1489
0
    }
1490
}
1491
1492
impl Error for InvalidHeaderName {}
1493
1494
// ===== HdrName =====
1495
1496
impl<'a> HdrName<'a> {
1497
    // Precondition: if lower then buf is valid UTF-8
1498
596
    fn custom(buf: &'a [u8], lower: bool) -> HdrName<'a> {
1499
596
        HdrName {
1500
596
            // Invariant (on MaybeLower): follows from the precondition
1501
596
            inner: Repr::Custom(MaybeLower { buf, lower }),
1502
596
        }
1503
596
    }
1504
1505
    pub fn from_bytes<F, U>(hdr: &[u8], f: F) -> Result<U, InvalidHeaderName>
1506
    where
1507
        F: FnOnce(HdrName<'_>) -> U,
1508
    {
1509
        let mut buf = uninit_u8_array();
1510
        // Precondition: HEADER_CHARS is a valid table for parse_hdr().
1511
        let hdr = parse_hdr(hdr, &mut buf, &HEADER_CHARS)?;
1512
        Ok(f(hdr))
1513
    }
1514
1515
    pub fn from_static<F, U>(hdr: &'static str, f: F) -> U
1516
    where
1517
        F: FnOnce(HdrName<'_>) -> U,
1518
    {
1519
        let mut buf = uninit_u8_array();
1520
        let hdr =
1521
            // Precondition: HEADER_CHARS is a valid table for parse_hdr().
1522
            parse_hdr(hdr.as_bytes(), &mut buf, &HEADER_CHARS).expect("static str is invalid name");
1523
        f(hdr)
1524
    }
1525
}
1526
1527
#[doc(hidden)]
1528
impl<'a> From<HdrName<'a>> for HeaderName {
1529
0
    fn from(src: HdrName<'a>) -> HeaderName {
1530
0
        match src.inner {
1531
0
            Repr::Standard(s) => HeaderName {
1532
0
                inner: Repr::Standard(s),
1533
0
            },
1534
0
            Repr::Custom(maybe_lower) => {
1535
0
                if maybe_lower.lower {
1536
0
                    let buf = Bytes::copy_from_slice(maybe_lower.buf);
1537
                    // Safety: the invariant on MaybeLower ensures buf is valid UTF-8.
1538
0
                    let byte_str = unsafe { ByteStr::from_utf8_unchecked(buf) };
1539
1540
0
                    HeaderName {
1541
0
                        inner: Repr::Custom(Custom(byte_str)),
1542
0
                    }
1543
                } else {
1544
                    use bytes::BufMut;
1545
0
                    let mut dst = BytesMut::with_capacity(maybe_lower.buf.len());
1546
1547
0
                    for b in maybe_lower.buf.iter() {
1548
0
                        // HEADER_CHARS maps each byte to a valid single-byte UTF-8
1549
0
                        // codepoint.
1550
0
                        dst.put_u8(HEADER_CHARS[*b as usize]);
1551
0
                    }
1552
1553
                    // Safety: the loop above maps each byte of maybe_lower.buf to a
1554
                    // valid single-byte UTF-8 codepoint before copying it into dst.
1555
                    // dst (and hence dst.freeze()) is thus valid UTF-8.
1556
0
                    let buf = unsafe { ByteStr::from_utf8_unchecked(dst.freeze()) };
1557
1558
0
                    HeaderName {
1559
0
                        inner: Repr::Custom(Custom(buf)),
1560
0
                    }
1561
                }
1562
            }
1563
        }
1564
0
    }
1565
}
1566
1567
#[doc(hidden)]
1568
impl<'a> PartialEq<HdrName<'a>> for HeaderName {
1569
    #[inline]
1570
    fn eq(&self, other: &HdrName<'a>) -> bool {
1571
        match self.inner {
1572
            Repr::Standard(a) => match other.inner {
1573
                Repr::Standard(b) => a == b,
1574
                _ => false,
1575
            },
1576
            Repr::Custom(Custom(ref a)) => match other.inner {
1577
                Repr::Custom(ref b) => {
1578
                    if b.lower {
1579
                        a.as_bytes() == b.buf
1580
                    } else {
1581
                        eq_ignore_ascii_case(a.as_bytes(), b.buf)
1582
                    }
1583
                }
1584
                _ => false,
1585
            },
1586
        }
1587
    }
1588
}
1589
1590
// ===== Custom =====
1591
1592
impl Hash for Custom {
1593
    #[inline]
1594
437
    fn hash<H: Hasher>(&self, hasher: &mut H) {
1595
437
        hasher.write(self.0.as_bytes())
1596
437
    }
<http::header::name::Custom as core::hash::Hash>::hash::<http::header::map::FnvHasher>
Line
Count
Source
1594
437
    fn hash<H: Hasher>(&self, hasher: &mut H) {
1595
437
        hasher.write(self.0.as_bytes())
1596
437
    }
Unexecuted instantiation: <http::header::name::Custom as core::hash::Hash>::hash::<std::hash::random::DefaultHasher>
1597
}
1598
1599
// ===== MaybeLower =====
1600
1601
impl<'a> Hash for MaybeLower<'a> {
1602
    #[inline]
1603
    fn hash<H: Hasher>(&self, hasher: &mut H) {
1604
        if self.lower {
1605
            hasher.write(self.buf);
1606
        } else {
1607
            for &b in self.buf {
1608
                hasher.write(&[HEADER_CHARS[b as usize]]);
1609
            }
1610
        }
1611
    }
1612
}
1613
1614
// Assumes that the left hand side is already lower case
1615
#[inline]
1616
fn eq_ignore_ascii_case(lower: &[u8], s: &[u8]) -> bool {
1617
    if lower.len() != s.len() {
1618
        return false;
1619
    }
1620
1621
    lower
1622
        .iter()
1623
        .zip(s)
1624
        .all(|(a, b)| *a == HEADER_CHARS[*b as usize])
1625
}
1626
1627
// Utility functions for MaybeUninit<>. These are drawn from unstable API's on
1628
// MaybeUninit<> itself.
1629
const SCRATCH_BUF_SIZE: usize = 64;
1630
const SCRATCH_BUF_OVERFLOW: usize = SCRATCH_BUF_SIZE + 1;
1631
1632
7.35k
fn uninit_u8_array() -> [MaybeUninit<u8>; SCRATCH_BUF_SIZE] {
1633
7.35k
    let arr = MaybeUninit::<[MaybeUninit<u8>; SCRATCH_BUF_SIZE]>::uninit();
1634
    // Safety: assume_init() is claiming that an array of MaybeUninit<>
1635
    // has been initialized, but MaybeUninit<>'s do not require initialization.
1636
7.35k
    unsafe { arr.assume_init() }
1637
7.35k
}
1638
1639
// Assuming all the elements are initialized, get a slice of them.
1640
//
1641
// Safety: All elements of `slice` must be initialized to prevent
1642
// undefined behavior.
1643
6.25k
unsafe fn slice_assume_init<T>(slice: &[MaybeUninit<T>]) -> &[T] {
1644
6.25k
    &*(slice as *const [MaybeUninit<T>] as *const [T])
1645
6.25k
}
1646
1647
#[cfg(test)]
1648
mod tests {
1649
    use self::StandardHeader::Vary;
1650
    use super::*;
1651
1652
    #[test]
1653
    fn test_bounds() {
1654
        fn check_bounds<T: Sync + Send>() {}
1655
        check_bounds::<HeaderName>();
1656
    }
1657
1658
    #[test]
1659
    fn test_parse_invalid_headers() {
1660
        for i in 0..128 {
1661
            let hdr = vec![1u8; i];
1662
            assert!(
1663
                HeaderName::from_bytes(&hdr).is_err(),
1664
                "{} invalid header chars did not fail",
1665
                i
1666
            );
1667
        }
1668
    }
1669
1670
    const ONE_TOO_LONG: &[u8] = &[b'a'; super::super::MAX_HEADER_NAME_LEN + 1];
1671
1672
    #[test]
1673
    fn test_invalid_name_lengths() {
1674
        assert!(
1675
            HeaderName::from_bytes(&[]).is_err(),
1676
            "zero-length header name is an error",
1677
        );
1678
1679
        let long = &ONE_TOO_LONG[0..super::super::MAX_HEADER_NAME_LEN];
1680
1681
        let long_str = std::str::from_utf8(long).unwrap();
1682
        assert_eq!(HeaderName::from_static(long_str), long_str); // shouldn't panic!
1683
1684
        assert!(
1685
            HeaderName::from_bytes(long).is_ok(),
1686
            "max header name length is ok",
1687
        );
1688
        assert!(
1689
            HeaderName::from_bytes(ONE_TOO_LONG).is_err(),
1690
            "longer than max header name length is an error",
1691
        );
1692
    }
1693
1694
    #[test]
1695
    #[should_panic]
1696
    fn test_static_invalid_name_lengths() {
1697
        // Safety: ONE_TOO_LONG contains only the UTF-8 safe, single-byte codepoint b'a'.
1698
        let _ = HeaderName::from_static(unsafe { std::str::from_utf8_unchecked(ONE_TOO_LONG) });
1699
    }
1700
1701
    #[test]
1702
    fn test_from_hdr_name() {
1703
        use self::StandardHeader::Vary;
1704
1705
        let name = HeaderName::from(HdrName {
1706
            inner: Repr::Standard(Vary),
1707
        });
1708
1709
        assert_eq!(name.inner, Repr::Standard(Vary));
1710
1711
        let name = HeaderName::from(HdrName {
1712
            inner: Repr::Custom(MaybeLower {
1713
                buf: b"hello-world",
1714
                lower: true,
1715
            }),
1716
        });
1717
1718
        assert_eq!(
1719
            name.inner,
1720
            Repr::Custom(Custom(ByteStr::from_static("hello-world")))
1721
        );
1722
1723
        let name = HeaderName::from(HdrName {
1724
            inner: Repr::Custom(MaybeLower {
1725
                buf: b"Hello-World",
1726
                lower: false,
1727
            }),
1728
        });
1729
1730
        assert_eq!(
1731
            name.inner,
1732
            Repr::Custom(Custom(ByteStr::from_static("hello-world")))
1733
        );
1734
    }
1735
1736
    #[test]
1737
    fn test_eq_hdr_name() {
1738
        use self::StandardHeader::Vary;
1739
1740
        let a = HeaderName {
1741
            inner: Repr::Standard(Vary),
1742
        };
1743
        let b = HdrName {
1744
            inner: Repr::Standard(Vary),
1745
        };
1746
1747
        assert_eq!(a, b);
1748
1749
        let a = HeaderName {
1750
            inner: Repr::Custom(Custom(ByteStr::from_static("vaary"))),
1751
        };
1752
        assert_ne!(a, b);
1753
1754
        let b = HdrName {
1755
            inner: Repr::Custom(MaybeLower {
1756
                buf: b"vaary",
1757
                lower: true,
1758
            }),
1759
        };
1760
1761
        assert_eq!(a, b);
1762
1763
        let b = HdrName {
1764
            inner: Repr::Custom(MaybeLower {
1765
                buf: b"vaary",
1766
                lower: false,
1767
            }),
1768
        };
1769
1770
        assert_eq!(a, b);
1771
1772
        let b = HdrName {
1773
            inner: Repr::Custom(MaybeLower {
1774
                buf: b"VAARY",
1775
                lower: false,
1776
            }),
1777
        };
1778
1779
        assert_eq!(a, b);
1780
1781
        let a = HeaderName {
1782
            inner: Repr::Standard(Vary),
1783
        };
1784
        assert_ne!(a, b);
1785
    }
1786
1787
    #[test]
1788
    fn test_from_static_std() {
1789
        let a = HeaderName {
1790
            inner: Repr::Standard(Vary),
1791
        };
1792
1793
        let b = HeaderName::from_static("vary");
1794
        assert_eq!(a, b);
1795
1796
        let b = HeaderName::from_static("vaary");
1797
        assert_ne!(a, b);
1798
    }
1799
1800
    #[test]
1801
    #[should_panic]
1802
    fn test_from_static_std_uppercase() {
1803
        HeaderName::from_static("Vary");
1804
    }
1805
1806
    #[test]
1807
    #[should_panic]
1808
    fn test_from_static_std_symbol() {
1809
        HeaderName::from_static("vary{}");
1810
    }
1811
1812
    // MaybeLower { lower: true }
1813
    #[test]
1814
    fn test_from_static_custom_short() {
1815
        let a = HeaderName {
1816
            inner: Repr::Custom(Custom(ByteStr::from_static("customheader"))),
1817
        };
1818
        let b = HeaderName::from_static("customheader");
1819
        assert_eq!(a, b);
1820
    }
1821
1822
    #[test]
1823
    #[should_panic]
1824
    fn test_from_static_custom_short_uppercase() {
1825
        HeaderName::from_static("custom header");
1826
    }
1827
1828
    #[test]
1829
    #[should_panic]
1830
    fn test_from_static_custom_short_symbol() {
1831
        HeaderName::from_static("CustomHeader");
1832
    }
1833
1834
    // MaybeLower { lower: false }
1835
    #[test]
1836
    fn test_from_static_custom_long() {
1837
        let a = HeaderName {
1838
            inner: Repr::Custom(Custom(ByteStr::from_static(
1839
                "longer-than-63--thisheaderislongerthansixtythreecharactersandthushandleddifferent",
1840
            ))),
1841
        };
1842
        let b = HeaderName::from_static(
1843
            "longer-than-63--thisheaderislongerthansixtythreecharactersandthushandleddifferent",
1844
        );
1845
        assert_eq!(a, b);
1846
    }
1847
1848
    #[test]
1849
    #[should_panic]
1850
    fn test_from_static_custom_long_uppercase() {
1851
        HeaderName::from_static(
1852
            "Longer-Than-63--ThisHeaderIsLongerThanSixtyThreeCharactersAndThusHandledDifferent",
1853
        );
1854
    }
1855
1856
    #[test]
1857
    #[should_panic]
1858
    fn test_from_static_custom_long_symbol() {
1859
        HeaderName::from_static(
1860
            "longer-than-63--thisheader{}{}{}{}islongerthansixtythreecharactersandthushandleddifferent"
1861
        );
1862
    }
1863
1864
    #[test]
1865
    fn test_from_static_custom_single_char() {
1866
        let a = HeaderName {
1867
            inner: Repr::Custom(Custom(ByteStr::from_static("a"))),
1868
        };
1869
        let b = HeaderName::from_static("a");
1870
        assert_eq!(a, b);
1871
    }
1872
1873
    #[test]
1874
    #[should_panic]
1875
    fn test_from_static_empty() {
1876
        HeaderName::from_static("");
1877
    }
1878
1879
    #[test]
1880
    fn test_all_tokens() {
1881
        HeaderName::from_static("!#$%&'*+-.^_`|~0123456789abcdefghijklmnopqrstuvwxyz");
1882
    }
1883
1884
    #[test]
1885
    fn test_from_lowercase() {
1886
        HeaderName::from_lowercase(&[0; 10]).unwrap_err();
1887
        HeaderName::from_lowercase(&[b'A'; 10]).unwrap_err();
1888
        HeaderName::from_lowercase(&[0x1; 10]).unwrap_err();
1889
        HeaderName::from_lowercase(&[0xFF; 10]).unwrap_err();
1890
        //HeaderName::from_lowercase(&[0; 100]).unwrap_err();
1891
        HeaderName::from_lowercase(&[b'A'; 100]).unwrap_err();
1892
        HeaderName::from_lowercase(&[0x1; 100]).unwrap_err();
1893
        HeaderName::from_lowercase(&[0xFF; 100]).unwrap_err();
1894
    }
1895
}