bitwarden_vault/folder/
create.rs1use bitwarden_api_api::models::FolderRequestModel;
2use bitwarden_core::{
3 ApiError, MissingFieldError,
4 key_management::{KeySlotIds, SymmetricKeySlotId},
5 require,
6};
7use bitwarden_crypto::{
8 CompositeEncryptable, CryptoError, IdentifyKey, KeyStoreContext, PrimitiveEncryptable,
9};
10use bitwarden_error::bitwarden_error;
11use bitwarden_state::repository::{RepositoryError, RepositoryOption};
12use serde::{Deserialize, Serialize};
13use thiserror::Error;
14#[cfg(feature = "wasm")]
15use tsify::Tsify;
16#[cfg(feature = "wasm")]
17use wasm_bindgen::prelude::*;
18
19use crate::{Folder, FolderView, FoldersClient, VaultParseError};
20
21#[derive(Serialize, Deserialize, Debug)]
23#[serde(rename_all = "camelCase")]
24#[cfg_attr(feature = "uniffi", derive(uniffi::Record))]
25#[cfg_attr(feature = "wasm", derive(Tsify), tsify(into_wasm_abi, from_wasm_abi))]
26pub struct FolderAddEditRequest {
27 pub name: String,
29}
30
31impl CompositeEncryptable<KeySlotIds, SymmetricKeySlotId, FolderRequestModel>
32 for FolderAddEditRequest
33{
34 fn encrypt_composite(
35 &self,
36 ctx: &mut KeyStoreContext<KeySlotIds>,
37 key: SymmetricKeySlotId,
38 ) -> Result<FolderRequestModel, CryptoError> {
39 Ok(FolderRequestModel {
40 name: self.name.encrypt(ctx, key)?.to_string(),
41 })
42 }
43}
44
45impl IdentifyKey<SymmetricKeySlotId> for FolderAddEditRequest {
46 fn key_identifier(&self) -> SymmetricKeySlotId {
47 SymmetricKeySlotId::User
48 }
49}
50
51#[allow(missing_docs)]
52#[bitwarden_error(flat)]
53#[derive(Debug, Error)]
54pub enum CreateFolderError {
55 #[error(transparent)]
56 Crypto(#[from] CryptoError),
57 #[error(transparent)]
58 Api(#[from] ApiError),
59 #[error(transparent)]
60 VaultParse(#[from] VaultParseError),
61 #[error(transparent)]
62 MissingField(#[from] MissingFieldError),
63 #[error(transparent)]
64 Repository(#[from] RepositoryError),
65}
66
67#[cfg_attr(feature = "wasm", wasm_bindgen)]
68impl FoldersClient {
69 pub async fn create(
71 &self,
72 request: FolderAddEditRequest,
73 ) -> Result<FolderView, CreateFolderError> {
74 let folder_request = self.key_store.encrypt(request)?;
75
76 let resp = self
77 .api_configurations
78 .api_client
79 .folders_api()
80 .post(Some(folder_request))
81 .await
82 .map_err(ApiError::from)?;
83
84 let folder: Folder = resp.try_into()?;
85
86 self.repository
87 .require()?
88 .set(require!(folder.id), folder.clone())
89 .await?;
90
91 Ok(self.key_store.decrypt(&folder)?)
92 }
93}
94
95#[cfg(test)]
96mod tests {
97 use std::sync::Arc;
98
99 use bitwarden_api_api::{apis::ApiClient, models::FolderResponseModel};
100 use bitwarden_core::{
101 client::ApiConfigurations, key_management::create_test_crypto_with_user_key,
102 };
103 use bitwarden_crypto::SymmetricCryptoKey;
104 use bitwarden_test::MemoryRepository;
105 use uuid::uuid;
106
107 use super::*;
108 use crate::FolderId;
109
110 fn create_client(api_client: ApiClient) -> FoldersClient {
111 FoldersClient {
112 key_store: create_test_crypto_with_user_key(
113 SymmetricCryptoKey::make_aes256_cbc_hmac_key(),
114 ),
115 api_configurations: Arc::new(ApiConfigurations::from_api_client(api_client)),
116 repository: Some(Arc::new(MemoryRepository::<Folder>::default())),
117 }
118 }
119
120 #[tokio::test]
121 async fn test_create_folder() {
122 let folder_id = FolderId::new(uuid!("25afb11c-9c95-4db5-8bac-c21cb204a3f1"));
123
124 let client = create_client(ApiClient::new_mocked(move |mock| {
125 mock.folders_api
126 .expect_post()
127 .returning(move |model| {
128 Ok(FolderResponseModel {
129 id: Some(folder_id.into()),
130 name: Some(model.unwrap().name),
131 revision_date: Some("2025-01-01T00:00:00Z".to_string()),
132 object: Some("folder".to_string()),
133 })
134 })
135 .once();
136 }));
137
138 let result = client
139 .create(FolderAddEditRequest {
140 name: "test".to_string(),
141 })
142 .await
143 .unwrap();
144
145 assert_eq!(
146 result,
147 FolderView {
148 id: Some(folder_id),
149 name: "test".to_string(),
150 revision_date: "2025-01-01T00:00:00Z".parse().unwrap(),
151 }
152 );
153
154 assert_eq!(
156 client
157 .key_store
158 .decrypt(
159 &client
160 .repository
161 .as_ref()
162 .unwrap()
163 .get(folder_id)
164 .await
165 .unwrap()
166 .unwrap()
167 )
168 .unwrap(),
169 result
170 );
171 }
172
173 #[tokio::test]
174 async fn test_create_folder_http_error() {
175 let client = create_client(ApiClient::new_mocked(move |mock| {
176 mock.folders_api
177 .expect_post()
178 .returning(move |_model| Err(std::io::Error::other("Simulated error").into()));
179 }));
180
181 let result = client
182 .create(FolderAddEditRequest {
183 name: "test".to_string(),
184 })
185 .await;
186
187 assert!(result.is_err());
188 assert!(matches!(result.unwrap_err(), CreateFolderError::Api(_)));
189 }
190}