mirror of https://github.com/fafhrd91/actix-web
Merge branch 'master' into h2c-auto
This commit is contained in:
commit
719cc13f2b
|
@ -5,6 +5,14 @@
|
||||||
- Implement `MessageBody` for `&mut B` where `B: MessageBody + Unpin`. [#2868]
|
- Implement `MessageBody` for `&mut B` where `B: MessageBody + Unpin`. [#2868]
|
||||||
- Implement `MessageBody` for `Pin<B>` where `B::Target: MessageBody`. [#2868]
|
- Implement `MessageBody` for `Pin<B>` where `B::Target: MessageBody`. [#2868]
|
||||||
- Automatic h2c detection via new service finalizer `HttpService::tcp_auto_h2c()`. [#2957]
|
- Automatic h2c detection via new service finalizer `HttpService::tcp_auto_h2c()`. [#2957]
|
||||||
|
- `HeaderMap::retain()` [#2955].
|
||||||
|
- Header name constants in `header` module. [#2956]
|
||||||
|
- `CROSS_ORIGIN_EMBEDDER_POLICY`
|
||||||
|
- `CROSS_ORIGIN_OPENER_POLICY`
|
||||||
|
- `PERMISSIONS_POLICY`
|
||||||
|
- `X_FORWARDED_FOR`
|
||||||
|
- `X_FORWARDED_HOST`
|
||||||
|
- `X_FORWARDED_PROTO`
|
||||||
|
|
||||||
### Performance
|
### Performance
|
||||||
- Improve overall performance of operations on `Extensions`. [#2890]
|
- Improve overall performance of operations on `Extensions`. [#2890]
|
||||||
|
@ -12,6 +20,8 @@
|
||||||
[#2868]: https://github.com/actix/actix-web/pull/2868
|
[#2868]: https://github.com/actix/actix-web/pull/2868
|
||||||
[#2890]: https://github.com/actix/actix-web/pull/2890
|
[#2890]: https://github.com/actix/actix-web/pull/2890
|
||||||
[#2957]: https://github.com/actix/actix-web/pull/2957
|
[#2957]: https://github.com/actix/actix-web/pull/2957
|
||||||
|
[#2955]: https://github.com/actix/actix-web/pull/2955
|
||||||
|
[#2956]: https://github.com/actix/actix-web/pull/2956
|
||||||
|
|
||||||
|
|
||||||
## 3.2.2 - 2022-09-11
|
## 3.2.2 - 2022-09-11
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
//! Common header names not defined in [`http`].
|
||||||
|
//!
|
||||||
|
//! Any headers added to this file will need to be re-exported from the list at `crate::headers`.
|
||||||
|
|
||||||
|
use http::header::HeaderName;
|
||||||
|
|
||||||
|
/// Response header that prevents a document from loading any cross-origin resources that don't
|
||||||
|
/// explicitly grant the document permission (using [CORP] or [CORS]).
|
||||||
|
///
|
||||||
|
/// [CORP]: https://developer.mozilla.org/en-US/docs/Web/HTTP/Cross-Origin_Resource_Policy_(CORP)
|
||||||
|
/// [CORS]: https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS
|
||||||
|
pub const CROSS_ORIGIN_EMBEDDER_POLICY: HeaderName =
|
||||||
|
HeaderName::from_static("cross-origin-embedder-policy");
|
||||||
|
|
||||||
|
/// Response header that allows you to ensure a top-level document does not share a browsing context
|
||||||
|
/// group with cross-origin documents.
|
||||||
|
pub const CROSS_ORIGIN_OPENER_POLICY: HeaderName =
|
||||||
|
HeaderName::from_static("cross-origin-opener-policy");
|
||||||
|
|
||||||
|
/// Response header that conveys a desire that the browser blocks no-cors cross-origin/cross-site
|
||||||
|
/// requests to the given resource.
|
||||||
|
pub const CROSS_ORIGIN_RESOURCE_POLICY: HeaderName =
|
||||||
|
HeaderName::from_static("cross-origin-resource-policy");
|
||||||
|
|
||||||
|
/// Response header that provides a mechanism to allow and deny the use of browser features in a
|
||||||
|
/// document or within any `<iframe>` elements in the document.
|
||||||
|
pub const PERMISSIONS_POLICY: HeaderName = HeaderName::from_static("permissions-policy");
|
||||||
|
|
||||||
|
/// Request header (de-facto standard) for identifying the originating IP address of a client
|
||||||
|
/// connecting to a web server through a proxy server.
|
||||||
|
pub const X_FORWARDED_FOR: HeaderName = HeaderName::from_static("x-forwarded-for");
|
||||||
|
|
||||||
|
/// Request header (de-facto standard) for identifying the original host requested by the client in
|
||||||
|
/// the `Host` HTTP request header.
|
||||||
|
pub const X_FORWARDED_HOST: HeaderName = HeaderName::from_static("x-forwarded-host");
|
||||||
|
|
||||||
|
/// Request header (de-facto standard) for identifying the protocol that a client used to connect to
|
||||||
|
/// your proxy or load balancer.
|
||||||
|
pub const X_FORWARDED_PROTO: HeaderName = HeaderName::from_static("x-forwarded-proto");
|
|
@ -150,9 +150,7 @@ impl HeaderMap {
|
||||||
/// assert_eq!(map.len(), 3);
|
/// assert_eq!(map.len(), 3);
|
||||||
/// ```
|
/// ```
|
||||||
pub fn len(&self) -> usize {
|
pub fn len(&self) -> usize {
|
||||||
self.inner
|
self.inner.values().map(|vals| vals.len()).sum()
|
||||||
.iter()
|
|
||||||
.fold(0, |acc, (_, values)| acc + values.len())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns the number of _keys_ stored in the map.
|
/// Returns the number of _keys_ stored in the map.
|
||||||
|
@ -552,6 +550,39 @@ impl HeaderMap {
|
||||||
Keys(self.inner.keys())
|
Keys(self.inner.keys())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Retains only the headers specified by the predicate.
|
||||||
|
///
|
||||||
|
/// In other words, removes all headers `(name, val)` for which `retain_fn(&name, &mut val)`
|
||||||
|
/// returns false.
|
||||||
|
///
|
||||||
|
/// The order in which headers are visited should be considered arbitrary.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
/// ```
|
||||||
|
/// # use actix_http::header::{self, HeaderMap, HeaderValue};
|
||||||
|
/// let mut map = HeaderMap::new();
|
||||||
|
///
|
||||||
|
/// map.append(header::HOST, HeaderValue::from_static("duck.com"));
|
||||||
|
/// map.append(header::SET_COOKIE, HeaderValue::from_static("one=1"));
|
||||||
|
/// map.append(header::SET_COOKIE, HeaderValue::from_static("two=2"));
|
||||||
|
///
|
||||||
|
/// map.retain(|name, val| val.as_bytes().starts_with(b"one"));
|
||||||
|
///
|
||||||
|
/// assert_eq!(map.len(), 1);
|
||||||
|
/// assert!(map.contains_key(&header::SET_COOKIE));
|
||||||
|
/// ```
|
||||||
|
pub fn retain<F>(&mut self, mut retain_fn: F)
|
||||||
|
where
|
||||||
|
F: FnMut(&HeaderName, &mut HeaderValue) -> bool,
|
||||||
|
{
|
||||||
|
self.inner.retain(|name, vals| {
|
||||||
|
vals.inner.retain(|val| retain_fn(name, val));
|
||||||
|
|
||||||
|
// invariant: make sure newly empty value lists are removed
|
||||||
|
!vals.is_empty()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
/// Clears the map, returning all name-value sets as an iterator.
|
/// Clears the map, returning all name-value sets as an iterator.
|
||||||
///
|
///
|
||||||
/// Header names will only be yielded for the first value in each set. All items that are
|
/// Header names will only be yielded for the first value in each set. All items that are
|
||||||
|
@ -943,6 +974,55 @@ mod tests {
|
||||||
assert!(map.is_empty());
|
assert!(map.is_empty());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn retain() {
|
||||||
|
let mut map = HeaderMap::new();
|
||||||
|
|
||||||
|
map.append(header::LOCATION, HeaderValue::from_static("/test"));
|
||||||
|
map.append(header::HOST, HeaderValue::from_static("duck.com"));
|
||||||
|
map.append(header::COOKIE, HeaderValue::from_static("one=1"));
|
||||||
|
map.append(header::COOKIE, HeaderValue::from_static("two=2"));
|
||||||
|
|
||||||
|
assert_eq!(map.len(), 4);
|
||||||
|
|
||||||
|
// by value
|
||||||
|
map.retain(|_, val| !val.as_bytes().contains(&b'/'));
|
||||||
|
assert_eq!(map.len(), 3);
|
||||||
|
|
||||||
|
// by name
|
||||||
|
map.retain(|name, _| name.as_str() != "cookie");
|
||||||
|
assert_eq!(map.len(), 1);
|
||||||
|
|
||||||
|
// keep but mutate value
|
||||||
|
map.retain(|_, val| {
|
||||||
|
*val = HeaderValue::from_static("replaced");
|
||||||
|
true
|
||||||
|
});
|
||||||
|
assert_eq!(map.len(), 1);
|
||||||
|
assert_eq!(map.get("host").unwrap(), "replaced");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn retain_removes_empty_value_lists() {
|
||||||
|
let mut map = HeaderMap::with_capacity(3);
|
||||||
|
|
||||||
|
map.append(header::HOST, HeaderValue::from_static("duck.com"));
|
||||||
|
map.append(header::HOST, HeaderValue::from_static("duck.com"));
|
||||||
|
|
||||||
|
assert_eq!(map.len(), 2);
|
||||||
|
assert_eq!(map.len_keys(), 1);
|
||||||
|
assert_eq!(map.inner.len(), 1);
|
||||||
|
assert_eq!(map.capacity(), 3);
|
||||||
|
|
||||||
|
// remove everything
|
||||||
|
map.retain(|_n, _v| false);
|
||||||
|
|
||||||
|
assert_eq!(map.len(), 0);
|
||||||
|
assert_eq!(map.len_keys(), 0);
|
||||||
|
assert_eq!(map.inner.len(), 0);
|
||||||
|
assert_eq!(map.capacity(), 3);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn entries_into_iter() {
|
fn entries_into_iter() {
|
||||||
let mut map = HeaderMap::new();
|
let mut map = HeaderMap::new();
|
||||||
|
|
|
@ -1,14 +1,18 @@
|
||||||
//! Pre-defined `HeaderName`s, traits for parsing and conversion, and other header utility methods.
|
//! Pre-defined `HeaderName`s, traits for parsing and conversion, and other header utility methods.
|
||||||
|
|
||||||
|
// declaring new header consts will yield this error
|
||||||
|
#![allow(clippy::declare_interior_mutable_const)]
|
||||||
|
|
||||||
use percent_encoding::{AsciiSet, CONTROLS};
|
use percent_encoding::{AsciiSet, CONTROLS};
|
||||||
|
|
||||||
// re-export from http except header map related items
|
// re-export from http except header map related items
|
||||||
pub use http::header::{
|
pub use ::http::header::{
|
||||||
HeaderName, HeaderValue, InvalidHeaderName, InvalidHeaderValue, ToStrError,
|
HeaderName, HeaderValue, InvalidHeaderName, InvalidHeaderValue, ToStrError,
|
||||||
};
|
};
|
||||||
|
|
||||||
// re-export const header names
|
// re-export const header names, list is explicit so that any updates to `common` module do not
|
||||||
pub use http::header::{
|
// conflict with this set
|
||||||
|
pub use ::http::header::{
|
||||||
ACCEPT, ACCEPT_CHARSET, ACCEPT_ENCODING, ACCEPT_LANGUAGE, ACCEPT_RANGES,
|
ACCEPT, ACCEPT_CHARSET, ACCEPT_ENCODING, ACCEPT_LANGUAGE, ACCEPT_RANGES,
|
||||||
ACCESS_CONTROL_ALLOW_CREDENTIALS, ACCESS_CONTROL_ALLOW_HEADERS,
|
ACCESS_CONTROL_ALLOW_CREDENTIALS, ACCESS_CONTROL_ALLOW_HEADERS,
|
||||||
ACCESS_CONTROL_ALLOW_METHODS, ACCESS_CONTROL_ALLOW_ORIGIN, ACCESS_CONTROL_EXPOSE_HEADERS,
|
ACCESS_CONTROL_ALLOW_METHODS, ACCESS_CONTROL_ALLOW_ORIGIN, ACCESS_CONTROL_EXPOSE_HEADERS,
|
||||||
|
@ -30,22 +34,29 @@ pub use http::header::{
|
||||||
use crate::{error::ParseError, HttpMessage};
|
use crate::{error::ParseError, HttpMessage};
|
||||||
|
|
||||||
mod as_name;
|
mod as_name;
|
||||||
|
mod common;
|
||||||
mod into_pair;
|
mod into_pair;
|
||||||
mod into_value;
|
mod into_value;
|
||||||
pub mod map;
|
pub mod map;
|
||||||
mod shared;
|
mod shared;
|
||||||
mod utils;
|
mod utils;
|
||||||
|
|
||||||
pub use self::as_name::AsHeaderName;
|
pub use self::{
|
||||||
pub use self::into_pair::TryIntoHeaderPair;
|
as_name::AsHeaderName,
|
||||||
pub use self::into_value::TryIntoHeaderValue;
|
into_pair::TryIntoHeaderPair,
|
||||||
pub use self::map::HeaderMap;
|
into_value::TryIntoHeaderValue,
|
||||||
pub use self::shared::{
|
map::HeaderMap,
|
||||||
parse_extended_value, q, Charset, ContentEncoding, ExtendedValue, HttpDate, LanguageTag,
|
shared::{
|
||||||
Quality, QualityItem,
|
parse_extended_value, q, Charset, ContentEncoding, ExtendedValue, HttpDate,
|
||||||
|
LanguageTag, Quality, QualityItem,
|
||||||
|
},
|
||||||
|
utils::{fmt_comma_delimited, from_comma_delimited, from_one_raw_str, http_percent_encode},
|
||||||
};
|
};
|
||||||
pub use self::utils::{
|
|
||||||
fmt_comma_delimited, from_comma_delimited, from_one_raw_str, http_percent_encode,
|
// re-export list is explicit so that any updates to `http` do not conflict with this set
|
||||||
|
pub use self::common::{
|
||||||
|
CROSS_ORIGIN_EMBEDDER_POLICY, CROSS_ORIGIN_OPENER_POLICY, CROSS_ORIGIN_RESOURCE_POLICY,
|
||||||
|
PERMISSIONS_POLICY, X_FORWARDED_FOR, X_FORWARDED_HOST, X_FORWARDED_PROTO,
|
||||||
};
|
};
|
||||||
|
|
||||||
/// An interface for types that already represent a valid header.
|
/// An interface for types that already represent a valid header.
|
||||||
|
|
Loading…
Reference in New Issue