bitwarden_core/client/
builder.rs1#[cfg(feature = "internal")]
2use std::sync::RwLock;
3use std::sync::{Arc, OnceLock};
4
5use bitwarden_api_base::new_http_client_builder;
6use bitwarden_crypto::KeyStore;
7use bitwarden_state::registry::StateRegistry;
8use reqwest::header::{self, HeaderValue};
9
10#[cfg(feature = "internal")]
11use crate::key_management::state_bridge::StateBridge;
12use crate::{
13 auth::auth_tokens::{NoopTokenHandler, TokenHandler},
14 client::{
15 client::Client,
16 client_settings::{ClientName, ClientSettings, HostPlatformInfo},
17 internal::{ApiConfigurations, InternalClient},
18 },
19};
20
21pub struct ClientBuilder {
23 settings: Option<ClientSettings>,
24 token_handler: Arc<dyn TokenHandler>,
25 state_registry: Option<StateRegistry>,
26 middleware: Vec<Arc<dyn reqwest_middleware::Middleware>>,
27}
28
29impl ClientBuilder {
30 pub fn new() -> Self {
32 Self {
33 settings: None,
34 token_handler: Arc::new(NoopTokenHandler),
35 state_registry: None,
36 middleware: Vec::new(),
37 }
38 }
39
40 pub fn with_settings(mut self, settings: ClientSettings) -> Self {
42 self.settings = Some(settings);
43 self
44 }
45
46 pub fn with_token_handler(mut self, token_handler: Arc<dyn TokenHandler>) -> Self {
48 self.token_handler = token_handler;
49 self
50 }
51
52 pub fn with_middleware(
54 mut self,
55 middleware: Vec<Arc<dyn reqwest_middleware::Middleware>>,
56 ) -> Self {
57 self.middleware = middleware;
58 self
59 }
60
61 pub fn with_state(mut self, state_registry: StateRegistry) -> Self {
64 self.state_registry = Some(state_registry);
65 self
66 }
67
68 pub fn build(self) -> Client {
70 let settings = self.settings.unwrap_or_default();
71
72 let external_http_client = new_http_client_builder()
73 .build()
74 .expect("External HTTP Client build should not fail");
75
76 let headers = build_default_headers(&HostPlatformInfo::from(&settings));
77
78 let key_store = KeyStore::default();
79 let state_registry = self
80 .state_registry
81 .unwrap_or_else(StateRegistry::new_with_memory_db);
82
83 let identity_http_client = new_http_client_builder()
85 .default_headers(headers.clone())
86 .build()
87 .expect("Bw HTTP Client build should not fail");
88 let identity = bitwarden_api_identity::Configuration {
89 base_path: settings.identity_url,
90 client: identity_http_client.into(),
91 };
92
93 let auth_middleware = self.token_handler.initialize_middleware(
95 &state_registry,
96 identity.clone(),
97 key_store.clone(),
98 );
99
100 #[cfg(not(target_arch = "wasm32"))]
105 let api_http_client = if self.middleware.is_empty() {
106 new_http_client_builder()
107 .default_headers(headers)
108 .build()
109 .expect("Bw HTTP Client build should not fail")
110 } else {
111 new_http_client_builder()
112 .default_headers(headers)
113 .redirect(reqwest::redirect::Policy::none())
114 .build()
115 .expect("Bw HTTP Client (no redirect) build should not fail")
116 };
117
118 #[cfg(target_arch = "wasm32")]
119 let api_http_client = new_http_client_builder()
120 .default_headers(headers)
121 .build()
122 .expect("Bw HTTP Client build should not fail");
123
124 let mut middleware_builder = reqwest_middleware::ClientBuilder::new(api_http_client);
126 for mw in self.middleware {
127 middleware_builder = middleware_builder.with_arc(mw);
128 }
129 let bw_http_client = middleware_builder.with_arc(auth_middleware).build();
130 let api = bitwarden_api_api::Configuration {
131 base_path: settings.api_url,
132 client: bw_http_client,
133 };
134
135 Client {
136 internal: Arc::new(InternalClient {
137 user_id: OnceLock::new(),
138 token_handler: self.token_handler,
139 api_configurations: ApiConfigurations::new(identity, api, settings.device_type),
140 external_http_client,
141 key_store,
142 #[cfg(feature = "internal")]
143 security_state: RwLock::new(None),
144 #[cfg(feature = "internal")]
145 state_bridge: StateBridge::new(),
146 state_registry,
147 }),
148 }
149 }
150}
151
152impl Default for ClientBuilder {
153 fn default() -> Self {
154 Self::new()
155 }
156}
157
158pub(crate) fn build_default_headers(info: &HostPlatformInfo) -> header::HeaderMap {
160 let mut headers = header::HeaderMap::new();
161
162 if let Some(device_identifier) = &info.device_identifier {
165 headers.append(
166 "Device-Identifier",
167 HeaderValue::from_str(device_identifier)
168 .expect("Device identifier should be a valid header value"),
169 );
170 }
171
172 if let Some(client_type) = Into::<Option<ClientName>>::into(info.device_type) {
173 headers.append(
174 "Bitwarden-Client-Name",
175 HeaderValue::from_str(&client_type.to_string())
176 .expect("All ASCII strings are valid header values"),
177 );
178 }
179
180 if let Some(version) = &info.bitwarden_client_version {
181 headers.append(
182 "Bitwarden-Client-Version",
183 HeaderValue::from_str(version).expect("Version should be a valid header value"),
184 );
185 }
186
187 if let Some(package_type) = &info.bitwarden_package_type {
188 headers.append(
189 "Bitwarden-Package-Type",
190 HeaderValue::from_str(package_type)
191 .expect("Package type should be a valid header value"),
192 );
193 }
194
195 headers.append(
198 "Device-Type",
199 HeaderValue::from_str(&(info.device_type as u8).to_string())
200 .expect("All numbers are valid ASCII"),
201 );
202
203 headers.append(
204 reqwest::header::USER_AGENT,
205 HeaderValue::from_str(&info.user_agent).expect("User agent should be a valid header value"),
206 );
207
208 headers
209}
210
211#[cfg(test)]
212mod tests {
213 use super::*;
214
215 #[test]
216 fn test_client_builder_default_builds() {
217 let _client = ClientBuilder::new().build();
218 }
219
220 #[test]
221 fn test_client_builder_with_settings_builds() {
222 let settings = ClientSettings::default();
223 let _client = ClientBuilder::new().with_settings(settings).build();
224 }
225
226 #[test]
227 fn test_client_builder_with_token_handler_builds() {
228 let handler: Arc<dyn TokenHandler> = Arc::new(NoopTokenHandler);
229 let _client = ClientBuilder::new().with_token_handler(handler).build();
230 }
231
232 #[test]
233 fn test_client_builder_chain_order_independence() {
234 let _a = ClientBuilder::new()
235 .with_settings(ClientSettings::default())
236 .with_token_handler(Arc::new(NoopTokenHandler) as Arc<dyn TokenHandler>)
237 .build();
238 let _b = ClientBuilder::new()
239 .with_token_handler(Arc::new(NoopTokenHandler) as Arc<dyn TokenHandler>)
240 .with_settings(ClientSettings::default())
241 .build();
242 }
243
244 #[test]
245 fn test_client_builder_with_state_builds() {
246 use bitwarden_state::registry::StateRegistry;
247 let registry = StateRegistry::new_with_memory_db();
248 let _client = ClientBuilder::new().with_state(registry).build();
249 }
250
251 #[test]
252 fn test_client_builder_with_state_in_chain() {
253 use bitwarden_state::registry::StateRegistry;
254 let registry = StateRegistry::new_with_memory_db();
255 let _client = ClientBuilder::new()
256 .with_settings(ClientSettings::default())
257 .with_state(registry)
258 .build();
259 }
260
261 #[test]
262 fn test_client_builder_with_middleware_compiles() {
263 struct StubMiddleware;
264
265 #[async_trait::async_trait]
266 impl reqwest_middleware::Middleware for StubMiddleware {
267 async fn handle(
268 &self,
269 req: reqwest::Request,
270 extensions: &mut http::Extensions,
271 next: reqwest_middleware::Next<'_>,
272 ) -> reqwest_middleware::Result<reqwest::Response> {
273 next.run(req, extensions).await
274 }
275 }
276
277 let arc_middleware: Arc<dyn reqwest_middleware::Middleware> = Arc::new(StubMiddleware);
278 let _client = ClientBuilder::new()
279 .with_middleware(vec![arc_middleware])
280 .build();
281 }
282}