tonic/
response.rs
1use http::Extensions;
2
3use crate::metadata::MetadataMap;
4
5#[derive(Debug)]
7pub struct Response<T> {
8 metadata: MetadataMap,
9 message: T,
10 extensions: Extensions,
11}
12
13impl<T> Response<T> {
14 pub fn new(message: T) -> Self {
27 Response {
28 metadata: MetadataMap::new(),
29 message,
30 extensions: Extensions::new(),
31 }
32 }
33
34 pub fn get_ref(&self) -> &T {
36 &self.message
37 }
38
39 pub fn get_mut(&mut self) -> &mut T {
41 &mut self.message
42 }
43
44 pub fn metadata(&self) -> &MetadataMap {
46 &self.metadata
47 }
48
49 pub fn metadata_mut(&mut self) -> &mut MetadataMap {
51 &mut self.metadata
52 }
53
54 pub fn into_inner(self) -> T {
56 self.message
57 }
58
59 pub fn into_parts(self) -> (MetadataMap, T, Extensions) {
61 (self.metadata, self.message, self.extensions)
62 }
63
64 pub fn from_parts(metadata: MetadataMap, message: T, extensions: Extensions) -> Self {
66 Self {
67 metadata,
68 message,
69 extensions,
70 }
71 }
72
73 pub(crate) fn from_http(res: http::Response<T>) -> Self {
74 let (head, message) = res.into_parts();
75 Response {
76 metadata: MetadataMap::from_headers(head.headers),
77 message,
78 extensions: head.extensions,
79 }
80 }
81
82 pub(crate) fn into_http(self) -> http::Response<T> {
83 let mut res = http::Response::new(self.message);
84
85 *res.version_mut() = http::Version::HTTP_2;
86 *res.headers_mut() = self.metadata.into_sanitized_headers();
87 *res.extensions_mut() = self.extensions;
88
89 res
90 }
91
92 #[doc(hidden)]
93 pub fn map<F, U>(self, f: F) -> Response<U>
94 where
95 F: FnOnce(T) -> U,
96 {
97 let message = f(self.message);
98 Response {
99 metadata: self.metadata,
100 message,
101 extensions: self.extensions,
102 }
103 }
104
105 pub fn extensions(&self) -> &Extensions {
107 &self.extensions
108 }
109
110 pub fn extensions_mut(&mut self) -> &mut Extensions {
112 &mut self.extensions
113 }
114
115 #[cfg(feature = "gzip")]
124 pub fn disable_compression(&mut self) {
125 self.extensions_mut()
126 .insert(crate::codec::compression::SingleMessageCompressionOverride::Disable);
127 }
128}
129
130#[cfg(test)]
131mod tests {
132 use super::*;
133 use crate::metadata::{MetadataKey, MetadataValue};
134
135 #[test]
136 fn reserved_headers_are_excluded() {
137 let mut r = Response::new(1);
138
139 for header in &MetadataMap::GRPC_RESERVED_HEADERS {
140 r.metadata_mut().insert(
141 MetadataKey::unchecked_from_header_name(header.clone()),
142 MetadataValue::from_static("invalid"),
143 );
144 }
145
146 let http_response = r.into_http();
147 assert!(http_response.headers().is_empty());
148 }
149}