tower_http/cors/
expose_headers.rs1use std::{array, fmt};
2
3use http::{
4 header::{self, HeaderName, HeaderValue},
5 request::Parts as RequestParts,
6};
7
8use super::{separated_by_commas, Any, WILDCARD};
9
10#[derive(Clone, Default)]
17#[must_use]
18pub struct ExposeHeaders(ExposeHeadersInner);
19
20impl ExposeHeaders {
21 pub fn any() -> Self {
27 Self(ExposeHeadersInner::Const(Some(WILDCARD)))
28 }
29
30 pub fn list<I>(headers: I) -> Self
36 where
37 I: IntoIterator<Item = HeaderName>,
38 {
39 Self(ExposeHeadersInner::Const(separated_by_commas(
40 headers.into_iter().map(Into::into),
41 )))
42 }
43
44 #[allow(clippy::borrow_interior_mutable_const)]
45 pub(super) fn is_wildcard(&self) -> bool {
46 matches!(&self.0, ExposeHeadersInner::Const(Some(v)) if v == WILDCARD)
47 }
48
49 pub(super) fn to_header(&self, _parts: &RequestParts) -> Option<(HeaderName, HeaderValue)> {
50 let expose_headers = match &self.0 {
51 ExposeHeadersInner::Const(v) => v.clone()?,
52 };
53
54 Some((header::ACCESS_CONTROL_EXPOSE_HEADERS, expose_headers))
55 }
56}
57
58impl fmt::Debug for ExposeHeaders {
59 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
60 match &self.0 {
61 ExposeHeadersInner::Const(inner) => f.debug_tuple("Const").field(inner).finish(),
62 }
63 }
64}
65
66impl From<Any> for ExposeHeaders {
67 fn from(_: Any) -> Self {
68 Self::any()
69 }
70}
71
72impl<const N: usize> From<[HeaderName; N]> for ExposeHeaders {
73 fn from(arr: [HeaderName; N]) -> Self {
74 #[allow(deprecated)] Self::list(array::IntoIter::new(arr))
76 }
77}
78
79impl From<Vec<HeaderName>> for ExposeHeaders {
80 fn from(vec: Vec<HeaderName>) -> Self {
81 Self::list(vec)
82 }
83}
84
85#[derive(Clone)]
86enum ExposeHeadersInner {
87 Const(Option<HeaderValue>),
88}
89
90impl Default for ExposeHeadersInner {
91 fn default() -> Self {
92 ExposeHeadersInner::Const(None)
93 }
94}