1use std::{
16 collections::{HashMap, HashSet},
17 sync::Arc,
18};
19
20use gloo_utils::format::JsValueSerdeExt;
21use indexed_db_futures::{
22 database::{Database, VersionChangeEvent},
23 error::Error,
24 future::OpenDbRequest,
25 object_store::ObjectStore,
26 prelude::*,
27 transaction::{Transaction, TransactionMode},
28};
29use js_sys::Date as JsDate;
30use matrix_sdk_base::{
31 deserialized_responses::SyncOrStrippedState, store::migration_helpers::RoomInfoV1,
32 StateStoreDataKey,
33};
34use matrix_sdk_store_encryption::StoreCipher;
35use ruma::{
36 events::{
37 room::{
38 create::RoomCreateEventContent,
39 member::{StrippedRoomMemberEvent, SyncRoomMemberEvent},
40 },
41 StateEventType,
42 },
43 serde::Raw,
44};
45use serde::{Deserialize, Serialize};
46use serde_json::value::{RawValue as RawJsonValue, Value as JsonValue};
47use wasm_bindgen::JsValue;
48
49use super::{
50 deserialize_value, encode_key, encode_to_range, keys, serialize_value, Result, RoomMember,
51 ALL_STORES,
52};
53use crate::IndexeddbStateStoreError;
54
55const CURRENT_DB_VERSION: u32 = 14;
56const CURRENT_META_DB_VERSION: u32 = 2;
57
58#[derive(Clone, Debug, PartialEq, Eq)]
61pub enum MigrationConflictStrategy {
62 Drop,
64 Raise,
68 BackupAndDrop,
70}
71
72#[derive(Clone, Serialize, Deserialize)]
73struct StoreKeyWrapper(Vec<u8>);
74
75mod old_keys {
76 pub const SESSION: &str = "session";
77 pub const SYNC_TOKEN: &str = "sync_token";
78 pub const MEMBERS: &str = "members";
79 pub const STRIPPED_MEMBERS: &str = "stripped_members";
80 pub const JOINED_USER_IDS: &str = "joined_user_ids";
81 pub const INVITED_USER_IDS: &str = "invited_user_ids";
82 pub const STRIPPED_JOINED_USER_IDS: &str = "stripped_joined_user_ids";
83 pub const STRIPPED_INVITED_USER_IDS: &str = "stripped_invited_user_ids";
84 pub const STRIPPED_ROOM_INFOS: &str = "stripped_room_infos";
85 pub const MEDIA: &str = "media";
86}
87
88pub async fn upgrade_meta_db(
89 meta_name: &str,
90 passphrase: Option<&str>,
91) -> Result<(Database, Option<Arc<StoreCipher>>)> {
92 let db_req: OpenDbRequest = Database::open(meta_name)
94 .with_version(CURRENT_META_DB_VERSION)
95 .with_on_upgrade_needed(
96 |evt: VersionChangeEvent, tx: &Transaction<'_>| -> Result<(), Error> {
97 let db = tx.db();
98 let old_version = evt.old_version() as u32;
99
100 if old_version < 1 {
101 db.create_object_store(keys::INTERNAL_STATE).build()?;
102 }
103
104 if old_version < 2 {
105 db.create_object_store(keys::BACKUPS_META).build()?;
106 }
107
108 Ok(())
109 },
110 )
111 .build()?;
112
113 let meta_db: Database = db_req.await?;
114
115 let store_cipher = if let Some(passphrase) = passphrase {
116 let tx: Transaction<'_> = meta_db
117 .transaction(keys::INTERNAL_STATE)
118 .with_mode(TransactionMode::Readwrite)
119 .build()?;
120 let ob = tx.object_store(keys::INTERNAL_STATE)?;
121
122 let cipher = if let Some(StoreKeyWrapper(inner)) = ob
123 .get(&JsValue::from_str(keys::STORE_KEY))
124 .await?
125 .map(|v: JsValue| v.into_serde())
126 .transpose()?
127 {
128 StoreCipher::import(passphrase, &inner)?
129 } else {
130 let cipher = StoreCipher::new()?;
131 #[cfg(not(test))]
132 let export = cipher.export(passphrase)?;
133 #[cfg(test)]
134 let export = cipher._insecure_export_fast_for_testing(passphrase)?;
135 ob.put(&StoreKeyWrapper(export)).with_key(keys::STORE_KEY.to_owned()).serde()?.await?;
136 cipher
137 };
138
139 tx.commit().await?;
140 Some(Arc::new(cipher))
141 } else {
142 None
143 };
144
145 Ok((meta_db, store_cipher))
146}
147
148#[derive(Debug, Clone, Default)]
150pub struct OngoingMigration {
151 drop_stores: HashSet<&'static str>,
153 create_stores: HashSet<&'static str>,
155 data: HashMap<&'static str, Vec<(JsValue, JsValue)>>,
157}
158
159pub async fn upgrade_inner_db(
160 name: &str,
161 store_cipher: Option<&StoreCipher>,
162 migration_strategy: MigrationConflictStrategy,
163 meta_db: &Database,
164) -> Result<Database> {
165 let mut db = Database::open(name).await?;
166
167 let mut old_version = db.version() as u32;
171
172 if old_version < CURRENT_DB_VERSION {
173 let has_store_cipher = store_cipher.is_some();
181
182 if old_version == 1 && db.object_store_names().next().is_none() {
186 old_version = 0;
187 }
188
189 if old_version == 0 {
192 let migration = OngoingMigration {
193 create_stores: ALL_STORES.iter().copied().collect(),
194 ..Default::default()
195 };
196 db = apply_migration(db, CURRENT_DB_VERSION, migration).await?;
197 } else if old_version < 2 && has_store_cipher {
198 match migration_strategy {
199 MigrationConflictStrategy::BackupAndDrop => {
200 backup_v1(&db, meta_db).await?;
201 }
202 MigrationConflictStrategy::Drop => {}
203 MigrationConflictStrategy::Raise => {
204 return Err(IndexeddbStateStoreError::MigrationConflict {
205 name: name.to_owned(),
206 old_version,
207 new_version: CURRENT_DB_VERSION,
208 });
209 }
210 }
211
212 let migration = OngoingMigration {
213 drop_stores: V1_STORES.iter().copied().collect(),
214 create_stores: ALL_STORES.iter().copied().collect(),
215 ..Default::default()
216 };
217 db = apply_migration(db, CURRENT_DB_VERSION, migration).await?;
218 } else {
219 if old_version < 3 {
220 db = migrate_to_v3(db, store_cipher).await?;
221 }
222 if old_version < 4 {
223 db = migrate_to_v4(db, store_cipher).await?;
224 }
225 if old_version < 5 {
226 db = migrate_to_v5(db, store_cipher).await?;
227 }
228 if old_version < 6 {
229 db = migrate_to_v6(db, store_cipher).await?;
230 }
231 if old_version < 7 {
232 db = migrate_to_v7(db, store_cipher).await?;
233 }
234 if old_version < 8 {
235 db = migrate_to_v8(db, store_cipher).await?;
236 }
237 if old_version < 9 {
238 db = migrate_to_v9(db).await?;
239 }
240 if old_version < 10 {
241 db = migrate_to_v10(db).await?;
242 }
243 if old_version < 11 {
244 db = migrate_to_v11(db).await?;
245 }
246 if old_version < 12 {
247 db = migrate_to_v12(db).await?;
248 }
249 if old_version < 13 {
250 db = migrate_to_v13(db).await?;
251 }
252 if old_version < 14 {
253 db = migrate_to_v14(db).await?;
254 }
255 }
256
257 db.close();
258
259 let db_req: OpenDbRequest = Database::open(name)
260 .with_version(CURRENT_DB_VERSION)
261 .with_on_upgrade_needed(
262 move |evt: VersionChangeEvent, _: &Transaction<'_>| -> Result<(), Error> {
263 panic!(
267 "Opening database that was not fully upgraded: \
268 DB version: {}; latest version: {CURRENT_DB_VERSION}",
269 evt.old_version()
270 )
271 },
272 )
273 .build()?;
274 db = db_req.await?;
275 }
276
277 Ok(db)
278}
279
280async fn apply_migration(
283 db: Database,
284 version: u32,
285 migration: OngoingMigration,
286) -> Result<Database> {
287 let name = db.name();
288 db.close();
289
290 let db_req: OpenDbRequest = Database::open(&name)
291 .with_version(version)
292 .with_on_upgrade_needed(
293 move |_: VersionChangeEvent, tx: &Transaction<'_>| -> Result<(), Error> {
294 for store in &migration.drop_stores {
296 tx.db().delete_object_store(store)?;
297 }
298 for store in &migration.create_stores {
299 tx.db().create_object_store(store).build()?;
300 }
301
302 Ok(())
303 },
304 )
305 .build()?;
306
307 let db = db_req.await?;
308
309 if !migration.data.is_empty() {
311 let stores: Vec<_> = migration.data.keys().copied().collect();
312 let tx = db.transaction(stores).with_mode(TransactionMode::Readwrite).build()?;
313
314 for (name, data) in migration.data {
315 let store = tx.object_store(name)?;
316 for (key, value) in data {
317 store.put(&value).with_key(key).await?;
318 }
319 }
320
321 tx.commit().await?;
322 }
323
324 Ok(db)
325}
326
327pub const V1_STORES: &[&str] = &[
328 old_keys::SESSION,
329 keys::ACCOUNT_DATA,
330 old_keys::MEMBERS,
331 keys::PROFILES,
332 keys::DISPLAY_NAMES,
333 old_keys::JOINED_USER_IDS,
334 old_keys::INVITED_USER_IDS,
335 keys::ROOM_STATE,
336 keys::ROOM_INFOS,
337 keys::PRESENCE,
338 keys::ROOM_ACCOUNT_DATA,
339 old_keys::STRIPPED_ROOM_INFOS,
340 old_keys::STRIPPED_MEMBERS,
341 keys::STRIPPED_ROOM_STATE,
342 old_keys::STRIPPED_JOINED_USER_IDS,
343 old_keys::STRIPPED_INVITED_USER_IDS,
344 keys::ROOM_USER_RECEIPTS,
345 keys::ROOM_EVENT_RECEIPTS,
346 old_keys::MEDIA,
347 keys::CUSTOM,
348 old_keys::SYNC_TOKEN,
349];
350
351async fn backup_v1(source: &Database, meta: &Database) -> Result<()> {
352 let now = JsDate::now();
353 let backup_name = format!("backup-{}-{now}", source.name());
354
355 let db_req: OpenDbRequest = Database::open(&backup_name)
356 .with_version(source.version())
357 .with_on_upgrade_needed(
358 move |_: VersionChangeEvent, tx: &Transaction<'_>| -> Result<(), Error> {
359 let db = tx.db();
361 for name in V1_STORES {
362 db.create_object_store(name).build()?;
363 }
364 Ok(())
365 },
366 )
367 .build()?;
368 let target = db_req.await?;
369
370 for name in V1_STORES {
371 let source_tx = source.transaction(*name).with_mode(TransactionMode::Readonly).build()?;
372 let source_obj = source_tx.object_store(name)?;
373 let Some(mut curs) = source_obj.open_cursor().await? else {
374 continue;
375 };
376
377 let mut data = vec![];
378 while let Some(value) = curs.next_record::<JsValue>().await? {
379 if let Some(key) = curs.key::<JsValue>()? {
380 data.push((key, value));
381 }
382 }
383
384 let target_tx = target.transaction(*name).with_mode(TransactionMode::Readwrite).build()?;
385 let target_obj = target_tx.object_store(name)?;
386
387 for (key, value) in data {
388 target_obj.put(value).with_key(key).await?;
389 }
390
391 target_tx.commit().await?;
392 }
393
394 let tx = meta.transaction(keys::BACKUPS_META).with_mode(TransactionMode::Readwrite).build()?;
395 let backup_store = tx.object_store(keys::BACKUPS_META)?;
396 backup_store.put(&backup_name).with_key(now).await?;
397
398 tx.commit().await?;
399
400 target.close();
401
402 Ok(())
403}
404
405async fn v3_fix_store(store: &ObjectStore<'_>, store_cipher: Option<&StoreCipher>) -> Result<()> {
406 fn maybe_fix_json(raw_json: &RawJsonValue) -> Result<Option<JsonValue>> {
407 let json = raw_json.get();
408
409 if json.contains(r#""content":null"#) {
410 let mut value: JsonValue = serde_json::from_str(json)?;
411 if let Some(content) = value.get_mut("content") {
412 if matches!(content, JsonValue::Null) {
413 *content = JsonValue::Object(Default::default());
414 return Ok(Some(value));
415 }
416 }
417 }
418
419 Ok(None)
420 }
421
422 let cursor = store.open_cursor().await?;
423
424 if let Some(mut cursor) = cursor {
425 while let Some(value) = cursor.next_record().await? {
426 let raw_json: Box<RawJsonValue> = deserialize_value(store_cipher, &value)?;
427
428 if let Some(fixed_json) = maybe_fix_json(&raw_json)? {
429 cursor.update(&serialize_value(store_cipher, &fixed_json)?).await?;
430 }
431 }
432 }
433
434 Ok(())
435}
436
437async fn migrate_to_v3(db: Database, store_cipher: Option<&StoreCipher>) -> Result<Database> {
439 let tx = db
440 .transaction([keys::ROOM_STATE, keys::ROOM_INFOS])
441 .with_mode(TransactionMode::Readwrite)
442 .build()?;
443
444 v3_fix_store(&tx.object_store(keys::ROOM_STATE)?, store_cipher).await?;
445 v3_fix_store(&tx.object_store(keys::ROOM_INFOS)?, store_cipher).await?;
446
447 tx.commit().await?;
448
449 let name = db.name();
450 db.close();
451
452 Ok(Database::open(&name).with_version(3u32).await?)
454}
455
456async fn migrate_to_v4(db: Database, store_cipher: Option<&StoreCipher>) -> Result<Database> {
458 let tx = db
459 .transaction([old_keys::SYNC_TOKEN, old_keys::SESSION])
460 .with_mode(TransactionMode::Readonly)
461 .build()?;
462 let mut values = Vec::new();
463
464 let sync_token_store = tx.object_store(old_keys::SYNC_TOKEN)?;
466 let sync_token = sync_token_store.get(&JsValue::from_str(old_keys::SYNC_TOKEN)).await?;
467
468 if let Some(sync_token) = sync_token {
469 values.push((
470 encode_key(store_cipher, StateStoreDataKey::SYNC_TOKEN, StateStoreDataKey::SYNC_TOKEN),
471 sync_token,
472 ));
473 }
474
475 let session_store = tx.object_store(old_keys::SESSION)?;
477 let range = encode_to_range(store_cipher, StateStoreDataKey::FILTER, StateStoreDataKey::FILTER);
478 if let Some(mut cursor) = session_store.open_cursor().with_query(&range).await? {
479 while let Some(value) = cursor.next_record().await? {
480 let Some(key) = cursor.key()? else {
481 break;
482 };
483 values.push((key, value));
484 }
485 }
486
487 tx.commit().await?;
488
489 let mut data = HashMap::new();
490 if !values.is_empty() {
491 data.insert(keys::KV, values);
492 }
493
494 let migration = OngoingMigration {
495 drop_stores: [old_keys::SYNC_TOKEN, old_keys::SESSION].into_iter().collect(),
496 create_stores: [keys::KV].into_iter().collect(),
497 data,
498 };
499 apply_migration(db, 4, migration).await
500}
501
502async fn migrate_to_v5(db: Database, store_cipher: Option<&StoreCipher>) -> Result<Database> {
504 let tx = db
505 .transaction([
506 old_keys::MEMBERS,
507 old_keys::STRIPPED_MEMBERS,
508 keys::ROOM_STATE,
509 keys::STRIPPED_ROOM_STATE,
510 keys::ROOM_INFOS,
511 old_keys::STRIPPED_ROOM_INFOS,
512 ])
513 .with_mode(TransactionMode::Readwrite)
514 .build()?;
515
516 let members_store = tx.object_store(old_keys::MEMBERS)?;
517 let state_store = tx.object_store(keys::ROOM_STATE)?;
518 let room_infos = tx
519 .object_store(keys::ROOM_INFOS)?
520 .get_all()
521 .await?
522 .filter_map(Result::ok)
523 .filter_map(|f| deserialize_value::<RoomInfoV1>(store_cipher, &f).ok())
524 .collect::<Vec<_>>();
525
526 for room_info in room_infos {
527 let room_id = room_info.room_id();
528 let range = encode_to_range(store_cipher, old_keys::MEMBERS, room_id);
529 for result in members_store.get_all().with_query(&range).await? {
530 let value = result?;
531 let raw_member_event =
532 deserialize_value::<Raw<SyncRoomMemberEvent>>(store_cipher, &value)?;
533 let state_key = raw_member_event.get_field::<String>("state_key")?.unwrap_or_default();
534 let key = encode_key(
535 store_cipher,
536 keys::ROOM_STATE,
537 (room_id, StateEventType::RoomMember, state_key),
538 );
539
540 state_store.add(&value).with_key(key).build()?.await?;
541 }
542 }
543
544 let stripped_members_store = tx.object_store(old_keys::STRIPPED_MEMBERS)?;
545 let stripped_state_store = tx.object_store(keys::STRIPPED_ROOM_STATE)?;
546 let stripped_room_infos = tx
547 .object_store(old_keys::STRIPPED_ROOM_INFOS)?
548 .get_all()
549 .await?
550 .filter_map(Result::ok)
551 .filter_map(|f| deserialize_value::<RoomInfoV1>(store_cipher, &f).ok())
552 .collect::<Vec<_>>();
553
554 for room_info in stripped_room_infos {
555 let room_id = room_info.room_id();
556 let range = encode_to_range(store_cipher, old_keys::STRIPPED_MEMBERS, room_id);
557 for result in stripped_members_store.get_all().with_query(&range).await? {
558 let value = result?;
559 let raw_member_event =
560 deserialize_value::<Raw<StrippedRoomMemberEvent>>(store_cipher, &value)?;
561 let state_key = raw_member_event.get_field::<String>("state_key")?.unwrap_or_default();
562 let key = encode_key(
563 store_cipher,
564 keys::STRIPPED_ROOM_STATE,
565 (room_id, StateEventType::RoomMember, state_key),
566 );
567
568 stripped_state_store.add(&value).with_key(key).build()?.await?;
569 }
570 }
571
572 tx.commit().await?;
573
574 let migration = OngoingMigration {
575 drop_stores: [old_keys::MEMBERS, old_keys::STRIPPED_MEMBERS].into_iter().collect(),
576 create_stores: Default::default(),
577 data: Default::default(),
578 };
579 apply_migration(db, 5, migration).await
580}
581
582async fn migrate_to_v6(db: Database, store_cipher: Option<&StoreCipher>) -> Result<Database> {
584 let tx = db
587 .transaction([
588 keys::ROOM_STATE,
589 keys::ROOM_INFOS,
590 keys::STRIPPED_ROOM_STATE,
591 old_keys::STRIPPED_ROOM_INFOS,
592 ])
593 .with_mode(TransactionMode::Readonly)
594 .build()?;
595
596 let state_store = tx.object_store(keys::ROOM_STATE)?;
597 let room_infos = tx
598 .object_store(keys::ROOM_INFOS)?
599 .get_all()
600 .await?
601 .filter_map(Result::ok)
602 .filter_map(|f| deserialize_value::<RoomInfoV1>(store_cipher, &f).ok())
603 .collect::<Vec<_>>();
604 let mut values = Vec::new();
605
606 for room_info in room_infos {
607 let room_id = room_info.room_id();
608 let range =
609 encode_to_range(store_cipher, keys::ROOM_STATE, (room_id, StateEventType::RoomMember));
610 for result in state_store.get_all().with_query(&range).await? {
611 let value = result?;
612 let member_event = deserialize_value::<Raw<SyncRoomMemberEvent>>(store_cipher, &value)?
613 .deserialize()?;
614 let key = encode_key(store_cipher, keys::USER_IDS, (room_id, member_event.state_key()));
615 let value = serialize_value(store_cipher, &RoomMember::from(&member_event))?;
616
617 values.push((key, value));
618 }
619 }
620
621 let stripped_state_store = tx.object_store(keys::STRIPPED_ROOM_STATE)?;
622 let stripped_room_infos = tx
623 .object_store(old_keys::STRIPPED_ROOM_INFOS)?
624 .get_all()
625 .await?
626 .filter_map(Result::ok)
627 .filter_map(|f| deserialize_value::<RoomInfoV1>(store_cipher, &f).ok())
628 .collect::<Vec<_>>();
629 let mut stripped_values = Vec::new();
630
631 for room_info in stripped_room_infos {
632 let room_id = room_info.room_id();
633 let range = encode_to_range(
634 store_cipher,
635 keys::STRIPPED_ROOM_STATE,
636 (room_id, StateEventType::RoomMember),
637 );
638 for result in stripped_state_store.get_all().with_query(&range).await? {
639 let value = result?;
640 let stripped_member_event =
641 deserialize_value::<Raw<StrippedRoomMemberEvent>>(store_cipher, &value)?
642 .deserialize()?;
643 let key = encode_key(
644 store_cipher,
645 keys::STRIPPED_USER_IDS,
646 (room_id, &stripped_member_event.state_key),
647 );
648 let value = serialize_value(store_cipher, &RoomMember::from(&stripped_member_event))?;
649
650 stripped_values.push((key, value));
651 }
652 }
653
654 tx.commit().await?;
655
656 let mut data = HashMap::new();
657 if !values.is_empty() {
658 data.insert(keys::USER_IDS, values);
659 }
660 if !stripped_values.is_empty() {
661 data.insert(keys::STRIPPED_USER_IDS, stripped_values);
662 }
663
664 let migration = OngoingMigration {
665 drop_stores: HashSet::from_iter([
666 old_keys::JOINED_USER_IDS,
667 old_keys::INVITED_USER_IDS,
668 old_keys::STRIPPED_JOINED_USER_IDS,
669 old_keys::STRIPPED_INVITED_USER_IDS,
670 ]),
671 create_stores: HashSet::from_iter([keys::USER_IDS, keys::STRIPPED_USER_IDS]),
672 data,
673 };
674 apply_migration(db, 6, migration).await
675}
676
677async fn migrate_to_v7(db: Database, store_cipher: Option<&StoreCipher>) -> Result<Database> {
680 let tx = db
681 .transaction([old_keys::STRIPPED_ROOM_INFOS])
682 .with_mode(TransactionMode::Readonly)
683 .build()?;
684
685 let room_infos = tx
686 .object_store(old_keys::STRIPPED_ROOM_INFOS)?
687 .get_all()
688 .await?
689 .filter_map(Result::ok)
690 .filter_map(|value| {
691 deserialize_value::<RoomInfoV1>(store_cipher, &value)
692 .ok()
693 .map(|info| (encode_key(store_cipher, keys::ROOM_INFOS, info.room_id()), value))
694 })
695 .collect::<Vec<_>>();
696
697 tx.commit().await?;
698
699 let mut data = HashMap::new();
700 if !room_infos.is_empty() {
701 data.insert(keys::ROOM_INFOS, room_infos);
702 }
703
704 let migration = OngoingMigration {
705 drop_stores: HashSet::from_iter([old_keys::STRIPPED_ROOM_INFOS]),
706 data,
707 ..Default::default()
708 };
709 apply_migration(db, 7, migration).await
710}
711
712async fn migrate_to_v8(db: Database, store_cipher: Option<&StoreCipher>) -> Result<Database> {
714 let tx = db
715 .transaction([keys::ROOM_STATE, keys::STRIPPED_ROOM_STATE, keys::ROOM_INFOS])
716 .with_mode(TransactionMode::Readwrite)
717 .build()?;
718
719 let room_state_store = tx.object_store(keys::ROOM_STATE)?;
720 let stripped_room_state_store = tx.object_store(keys::STRIPPED_ROOM_STATE)?;
721 let room_infos_store = tx.object_store(keys::ROOM_INFOS)?;
722
723 let room_infos_v1 = room_infos_store
724 .get_all()
725 .build()?
726 .await?
727 .map(|value| deserialize_value::<RoomInfoV1>(store_cipher, &value?))
728 .collect::<Result<Vec<_>, _>>()?;
729
730 for room_info_v1 in room_infos_v1 {
731 let create = if let Some(event) = stripped_room_state_store
732 .get(&encode_key(
733 store_cipher,
734 keys::STRIPPED_ROOM_STATE,
735 (room_info_v1.room_id(), &StateEventType::RoomCreate, ""),
736 ))
737 .await?
738 .map(|f| deserialize_value(store_cipher, &f))
739 .transpose()?
740 {
741 Some(SyncOrStrippedState::<RoomCreateEventContent>::Stripped(event))
742 } else {
743 room_state_store
744 .get(&encode_key(
745 store_cipher,
746 keys::ROOM_STATE,
747 (room_info_v1.room_id(), &StateEventType::RoomCreate, ""),
748 ))
749 .await?
750 .map(|f| deserialize_value(store_cipher, &f))
751 .transpose()?
752 .map(SyncOrStrippedState::<RoomCreateEventContent>::Sync)
753 };
754
755 let room_info = room_info_v1.migrate(create.as_ref());
756 room_infos_store
757 .put(&serialize_value(store_cipher, &room_info)?)
758 .with_key(encode_key(store_cipher, keys::ROOM_INFOS, room_info.room_id()))
759 .build()?;
760 }
761
762 tx.commit().await?;
763
764 let name = db.name();
765 db.close();
766
767 Ok(Database::open(&name).with_version(8u32).build()?.await?)
769}
770
771async fn migrate_to_v9(db: Database) -> Result<Database> {
773 let migration = OngoingMigration {
774 drop_stores: [].into(),
775 create_stores: [keys::ROOM_SEND_QUEUE].into_iter().collect(),
776 data: Default::default(),
777 };
778 apply_migration(db, 9, migration).await
779}
780
781async fn migrate_to_v10(db: Database) -> Result<Database> {
783 let migration = OngoingMigration {
784 drop_stores: [].into(),
785 create_stores: [keys::DEPENDENT_SEND_QUEUE].into_iter().collect(),
786 data: Default::default(),
787 };
788 apply_migration(db, 10, migration).await
789}
790
791async fn migrate_to_v11(db: Database) -> Result<Database> {
793 let migration = OngoingMigration {
794 drop_stores: [old_keys::MEDIA].into(),
795 create_stores: Default::default(),
796 data: Default::default(),
797 };
798 apply_migration(db, 11, migration).await
799}
800
801async fn migrate_to_v12(db: Database) -> Result<Database> {
804 let store_keys = [keys::DEPENDENT_SEND_QUEUE, keys::ROOM_SEND_QUEUE];
805 let tx = db.transaction(store_keys).with_mode(TransactionMode::Readwrite).build()?;
806
807 for store_name in store_keys {
808 let store = tx.object_store(store_name)?;
809 store.clear()?;
810 }
811
812 tx.commit().await?;
813
814 let name = db.name();
815 db.close();
816
817 Ok(Database::open(&name).with_version(12u32).build()?.await?)
819}
820
821async fn migrate_to_v13(db: Database) -> Result<Database> {
823 let migration = OngoingMigration {
824 drop_stores: [].into(),
825 create_stores: [keys::THREAD_SUBSCRIPTIONS].into_iter().collect(),
826 data: Default::default(),
827 };
828 apply_migration(db, 13, migration).await
829}
830
831async fn migrate_to_v14(db: Database) -> Result<Database> {
835 let migration = OngoingMigration {
836 drop_stores: [keys::THREAD_SUBSCRIPTIONS].into_iter().collect(),
837 create_stores: [keys::THREAD_SUBSCRIPTIONS].into_iter().collect(),
838 data: Default::default(),
839 };
840 apply_migration(db, 14, migration).await
841}
842
843#[cfg(all(test, target_family = "wasm"))]
844mod tests {
845 wasm_bindgen_test::wasm_bindgen_test_configure!(run_in_browser);
846
847 use assert_matches::assert_matches;
848 use assert_matches2::assert_let;
849 use indexed_db_futures::{
850 database::{Database, VersionChangeEvent},
851 error::Error,
852 object_store::ObjectStore,
853 prelude::*,
854 transaction::{Transaction, TransactionMode},
855 };
856 use matrix_sdk_base::{
857 deserialized_responses::RawMemberEvent,
858 store::{RoomLoadSettings, StateStoreExt},
859 sync::UnreadNotificationsCount,
860 RoomMemberships, RoomState, StateStore, StateStoreDataKey, StoreError,
861 };
862 use matrix_sdk_test::{async_test, test_json};
863 use ruma::{
864 events::{
865 room::{
866 create::RoomCreateEventContent,
867 member::{StrippedRoomMemberEvent, SyncRoomMemberEvent},
868 },
869 AnySyncStateEvent, StateEventType,
870 },
871 owned_user_id, room_id,
872 serde::Raw,
873 server_name, user_id, EventId, MilliSecondsSinceUnixEpoch, OwnedUserId, RoomId, UserId,
874 };
875 use serde_json::json;
876 use uuid::Uuid;
877 use wasm_bindgen::JsValue;
878
879 use super::{old_keys, MigrationConflictStrategy, CURRENT_DB_VERSION, CURRENT_META_DB_VERSION};
880 use crate::{
881 serializer::safe_encode::traits::SafeEncode,
882 state_store::{encode_key, keys, serialize_value, Result},
883 IndexeddbStateStore, IndexeddbStateStoreError,
884 };
885
886 const CUSTOM_DATA_KEY: &[u8] = b"custom_data_key";
887 const CUSTOM_DATA: &[u8] = b"some_custom_data";
888
889 pub async fn create_fake_db(name: &str, version: u32) -> Result<Database> {
890 Database::open(name)
891 .with_version(version)
892 .with_on_upgrade_needed(
893 move |_: VersionChangeEvent, tx: &Transaction<'_>| -> Result<(), Error> {
894 let db = tx.db();
895
896 let common_stores = &[
898 keys::ACCOUNT_DATA,
899 keys::PROFILES,
900 keys::DISPLAY_NAMES,
901 keys::ROOM_STATE,
902 keys::ROOM_INFOS,
903 keys::PRESENCE,
904 keys::ROOM_ACCOUNT_DATA,
905 keys::STRIPPED_ROOM_STATE,
906 keys::ROOM_USER_RECEIPTS,
907 keys::ROOM_EVENT_RECEIPTS,
908 keys::CUSTOM,
909 ];
910
911 for name in common_stores {
912 db.create_object_store(name).build()?;
913 }
914
915 if version < 4 {
916 for name in [old_keys::SYNC_TOKEN, old_keys::SESSION] {
917 db.create_object_store(name).build()?;
918 }
919 }
920 if version >= 4 {
921 db.create_object_store(keys::KV).build()?;
922 }
923 if version < 5 {
924 for name in [old_keys::MEMBERS, old_keys::STRIPPED_MEMBERS] {
925 db.create_object_store(name).build()?;
926 }
927 }
928 if version < 6 {
929 for name in [
930 old_keys::INVITED_USER_IDS,
931 old_keys::JOINED_USER_IDS,
932 old_keys::STRIPPED_INVITED_USER_IDS,
933 old_keys::STRIPPED_JOINED_USER_IDS,
934 ] {
935 db.create_object_store(name).build()?;
936 }
937 }
938 if version >= 6 {
939 for name in [keys::USER_IDS, keys::STRIPPED_USER_IDS] {
940 db.create_object_store(name).build()?;
941 }
942 }
943 if version < 7 {
944 db.create_object_store(old_keys::STRIPPED_ROOM_INFOS).build()?;
945 }
946 if version < 11 {
947 db.create_object_store(old_keys::MEDIA).build()?;
948 }
949
950 Ok(())
951 },
952 )
953 .build()?
954 .await
955 .map_err(Into::into)
956 }
957
958 fn room_info_v1_json(
959 room_id: &RoomId,
960 state: RoomState,
961 name: Option<&str>,
962 creator: Option<&UserId>,
963 ) -> serde_json::Value {
964 let name_content = match name {
966 Some(name) => json!({ "name": name }),
967 None => json!({ "name": null }),
968 };
969 let create_content = match creator {
971 Some(creator) => RoomCreateEventContent::new_v1(creator.to_owned()),
972 None => RoomCreateEventContent::new_v11(),
973 };
974
975 json!({
976 "room_id": room_id,
977 "room_type": state,
978 "notification_counts": UnreadNotificationsCount::default(),
979 "summary": {
980 "heroes": [],
981 "joined_member_count": 0,
982 "invited_member_count": 0,
983 },
984 "members_synced": false,
985 "base_info": {
986 "dm_targets": [],
987 "max_power_level": 100,
988 "name": {
989 "Original": {
990 "content": name_content,
991 },
992 },
993 "create": {
994 "Original": {
995 "content": create_content,
996 }
997 }
998 },
999 })
1000 }
1001
1002 #[async_test]
1003 pub async fn test_new_store() -> Result<()> {
1004 let name = format!("new-store-no-cipher-{}", Uuid::new_v4().as_hyphenated().to_string());
1005
1006 let store = IndexeddbStateStore::builder().name(name).build().await?;
1008 assert_eq!(store.has_backups().await?, false);
1010 assert_eq!(store.get_custom_value(CUSTOM_DATA_KEY).await?, None);
1012
1013 assert_eq!(store.version(), CURRENT_DB_VERSION);
1015 assert_eq!(store.meta_version(), CURRENT_META_DB_VERSION);
1016
1017 Ok(())
1018 }
1019
1020 #[async_test]
1021 pub async fn test_migrating_v1_to_v2_plain() -> Result<()> {
1022 let name = format!("migrating-v2-no-cipher-{}", Uuid::new_v4().as_hyphenated().to_string());
1023
1024 {
1026 let db = create_fake_db(&name, 1).await?;
1027 let tx = db.transaction(keys::CUSTOM).with_mode(TransactionMode::Readwrite).build()?;
1028 let custom = tx.object_store(keys::CUSTOM)?;
1029 let jskey = JsValue::from_str(
1030 core::str::from_utf8(CUSTOM_DATA_KEY).map_err(StoreError::Codec)?,
1031 );
1032 custom.put(&serialize_value(None, &CUSTOM_DATA)?).with_key(jskey).await?;
1033 tx.commit().await?;
1034 db.close();
1035 }
1036
1037 let store = IndexeddbStateStore::builder().name(name).build().await?;
1039 assert_eq!(store.has_backups().await?, false);
1041 assert_let!(Some(stored_data) = store.get_custom_value(CUSTOM_DATA_KEY).await?);
1043 assert_eq!(stored_data, CUSTOM_DATA);
1044
1045 assert_eq!(store.version(), CURRENT_DB_VERSION);
1047 assert_eq!(store.meta_version(), CURRENT_META_DB_VERSION);
1048
1049 Ok(())
1050 }
1051
1052 #[async_test]
1053 pub async fn test_migrating_v1_to_v2_with_pw() -> Result<()> {
1054 let name =
1055 format!("migrating-v2-with-cipher-{}", Uuid::new_v4().as_hyphenated().to_string());
1056 let passphrase = "somepassphrase".to_owned();
1057
1058 {
1060 let db = create_fake_db(&name, 1).await?;
1061 let tx = db.transaction(keys::CUSTOM).with_mode(TransactionMode::Readwrite).build()?;
1062 let custom = tx.object_store(keys::CUSTOM)?;
1063 let jskey = JsValue::from_str(
1064 core::str::from_utf8(CUSTOM_DATA_KEY).map_err(StoreError::Codec)?,
1065 );
1066 custom.put(&serialize_value(None, &CUSTOM_DATA)?).with_key(jskey).await?;
1067 tx.commit().await?;
1068 db.close();
1069 }
1070
1071 let store =
1073 IndexeddbStateStore::builder().name(name).passphrase(passphrase).build().await?;
1074 assert_eq!(store.has_backups().await?, true);
1076 assert!(store.latest_backup().await?.is_some(), "No backup_found");
1077 assert_eq!(store.get_custom_value(CUSTOM_DATA_KEY).await?, None);
1079
1080 assert_eq!(store.version(), CURRENT_DB_VERSION);
1082 assert_eq!(store.meta_version(), CURRENT_META_DB_VERSION);
1083
1084 Ok(())
1085 }
1086
1087 #[async_test]
1088 pub async fn test_migrating_v1_to_v2_with_pw_drops() -> Result<()> {
1089 let name = format!(
1090 "migrating-v2-with-cipher-drops-{}",
1091 Uuid::new_v4().as_hyphenated().to_string()
1092 );
1093 let passphrase = "some-other-passphrase".to_owned();
1094
1095 {
1097 let db = create_fake_db(&name, 1).await?;
1098 let tx = db.transaction(keys::CUSTOM).with_mode(TransactionMode::Readwrite).build()?;
1099 let custom = tx.object_store(keys::CUSTOM)?;
1100 let jskey = JsValue::from_str(
1101 core::str::from_utf8(CUSTOM_DATA_KEY).map_err(StoreError::Codec)?,
1102 );
1103 custom.put(&serialize_value(None, &CUSTOM_DATA)?).with_key(jskey).await?;
1104 tx.commit().await?;
1105 db.close();
1106 }
1107
1108 let store = IndexeddbStateStore::builder()
1110 .name(name)
1111 .passphrase(passphrase)
1112 .migration_conflict_strategy(MigrationConflictStrategy::Drop)
1113 .build()
1114 .await?;
1115 assert_eq!(store.has_backups().await?, false);
1117 assert_eq!(store.get_custom_value(CUSTOM_DATA_KEY).await?, None);
1119
1120 assert_eq!(store.version(), CURRENT_DB_VERSION);
1122 assert_eq!(store.meta_version(), CURRENT_META_DB_VERSION);
1123
1124 Ok(())
1125 }
1126
1127 #[async_test]
1128 pub async fn test_migrating_v1_to_v2_with_pw_raise() -> Result<()> {
1129 let name = format!(
1130 "migrating-v2-with-cipher-raises-{}",
1131 Uuid::new_v4().as_hyphenated().to_string()
1132 );
1133 let passphrase = "some-other-passphrase".to_owned();
1134
1135 {
1137 let db = create_fake_db(&name, 1).await?;
1138 let tx = db.transaction(keys::CUSTOM).with_mode(TransactionMode::Readwrite).build()?;
1139 let custom = tx.object_store(keys::CUSTOM)?;
1140 let jskey = JsValue::from_str(
1141 core::str::from_utf8(CUSTOM_DATA_KEY).map_err(StoreError::Codec)?,
1142 );
1143 custom.put(&serialize_value(None, &CUSTOM_DATA)?).with_key(jskey).await?;
1144 tx.commit().await?;
1145 db.close();
1146 }
1147
1148 let store_res = IndexeddbStateStore::builder()
1150 .name(name)
1151 .passphrase(passphrase)
1152 .migration_conflict_strategy(MigrationConflictStrategy::Raise)
1153 .build()
1154 .await;
1155
1156 assert_matches!(store_res, Err(IndexeddbStateStoreError::MigrationConflict { .. }));
1157
1158 Ok(())
1159 }
1160
1161 #[async_test]
1162 pub async fn test_migrating_to_v3() -> Result<()> {
1163 let name = format!("migrating-v3-{}", Uuid::new_v4().as_hyphenated().to_string());
1164
1165 let wrong_redacted_state_event = json!({
1167 "content": null,
1168 "event_id": "$wrongevent",
1169 "origin_server_ts": 1673887516047_u64,
1170 "sender": "@example:localhost",
1171 "state_key": "",
1172 "type": "m.room.topic",
1173 "unsigned": {
1174 "redacted_because": {
1175 "type": "m.room.redaction",
1176 "sender": "@example:localhost",
1177 "content": {},
1178 "redacts": "$wrongevent",
1179 "origin_server_ts": 1673893816047_u64,
1180 "unsigned": {},
1181 "event_id": "$redactionevent",
1182 },
1183 },
1184 });
1185 serde_json::from_value::<AnySyncStateEvent>(wrong_redacted_state_event.clone())
1186 .unwrap_err();
1187
1188 let room_id = room_id!("!some_room:localhost");
1189
1190 {
1192 let db = create_fake_db(&name, 2).await?;
1193 let tx =
1194 db.transaction(keys::ROOM_STATE).with_mode(TransactionMode::Readwrite).build()?;
1195 let state = tx.object_store(keys::ROOM_STATE)?;
1196 let key: JsValue = (room_id, StateEventType::RoomTopic, "").as_encoded_string().into();
1197 state.put(&serialize_value(None, &wrong_redacted_state_event)?).with_key(key).await?;
1198 tx.commit().await?;
1199 db.close();
1200 }
1201
1202 let store = IndexeddbStateStore::builder().name(name).build().await?;
1204 let event =
1205 store.get_state_event(room_id, StateEventType::RoomTopic, "").await.unwrap().unwrap();
1206 event.deserialize().unwrap();
1207
1208 assert_eq!(store.version(), CURRENT_DB_VERSION);
1210 assert_eq!(store.meta_version(), CURRENT_META_DB_VERSION);
1211
1212 Ok(())
1213 }
1214
1215 #[async_test]
1216 pub async fn test_migrating_to_v4() -> Result<()> {
1217 let name = format!("migrating-v4-{}", Uuid::new_v4().as_hyphenated().to_string());
1218
1219 let sync_token = "a_very_unique_string";
1220 let filter_1 = "filter_1";
1221 let filter_1_id = "filter_1_id";
1222 let filter_2 = "filter_2";
1223 let filter_2_id = "filter_2_id";
1224
1225 {
1227 let db = create_fake_db(&name, 3).await?;
1228 let tx = db
1229 .transaction([old_keys::SYNC_TOKEN, old_keys::SESSION])
1230 .with_mode(TransactionMode::Readwrite)
1231 .build()?;
1232
1233 let sync_token_store = tx.object_store(old_keys::SYNC_TOKEN)?;
1234 sync_token_store
1235 .put(&serialize_value(None, &sync_token)?)
1236 .with_key(JsValue::from_str(old_keys::SYNC_TOKEN))
1237 .build()?;
1238
1239 let session_store = tx.object_store(old_keys::SESSION)?;
1240 session_store
1241 .put(&serialize_value(None, &filter_1_id)?)
1242 .with_key(encode_key(
1243 None,
1244 StateStoreDataKey::FILTER,
1245 (StateStoreDataKey::FILTER, filter_1),
1246 ))
1247 .build()?;
1248 session_store
1249 .put(&serialize_value(None, &filter_2_id)?)
1250 .with_key(encode_key(
1251 None,
1252 StateStoreDataKey::FILTER,
1253 (StateStoreDataKey::FILTER, filter_2),
1254 ))
1255 .build()?;
1256
1257 tx.commit().await?;
1258 db.close();
1259 }
1260
1261 let store = IndexeddbStateStore::builder().name(name).build().await?;
1263
1264 let stored_sync_token = store
1265 .get_kv_data(StateStoreDataKey::SyncToken)
1266 .await?
1267 .unwrap()
1268 .into_sync_token()
1269 .unwrap();
1270 assert_eq!(stored_sync_token, sync_token);
1271
1272 let stored_filter_1_id = store
1273 .get_kv_data(StateStoreDataKey::Filter(filter_1))
1274 .await?
1275 .unwrap()
1276 .into_filter()
1277 .unwrap();
1278 assert_eq!(stored_filter_1_id, filter_1_id);
1279
1280 let stored_filter_2_id = store
1281 .get_kv_data(StateStoreDataKey::Filter(filter_2))
1282 .await?
1283 .unwrap()
1284 .into_filter()
1285 .unwrap();
1286 assert_eq!(stored_filter_2_id, filter_2_id);
1287
1288 Ok(())
1289 }
1290
1291 #[async_test]
1292 pub async fn test_migrating_to_v5() -> Result<()> {
1293 let name = format!("migrating-v5-{}", Uuid::new_v4().as_hyphenated().to_string());
1294
1295 let room_id = room_id!("!room:localhost");
1296 let member_event =
1297 Raw::new(&*test_json::MEMBER_INVITE).unwrap().cast_unchecked::<SyncRoomMemberEvent>();
1298 let user_id = user_id!("@invited:localhost");
1299
1300 let stripped_room_id = room_id!("!stripped_room:localhost");
1301 let stripped_member_event = Raw::new(&*test_json::MEMBER_STRIPPED)
1302 .unwrap()
1303 .cast_unchecked::<StrippedRoomMemberEvent>();
1304 let stripped_user_id = user_id!("@example:localhost");
1305
1306 {
1308 let db = create_fake_db(&name, 4).await?;
1309 let tx = db
1310 .transaction([
1311 old_keys::MEMBERS,
1312 keys::ROOM_INFOS,
1313 old_keys::STRIPPED_MEMBERS,
1314 old_keys::STRIPPED_ROOM_INFOS,
1315 ])
1316 .with_mode(TransactionMode::Readwrite)
1317 .build()?;
1318
1319 let members_store = tx.object_store(old_keys::MEMBERS)?;
1320 members_store
1321 .put(&serialize_value(None, &member_event)?)
1322 .with_key(encode_key(None, old_keys::MEMBERS, (room_id, user_id)))
1323 .build()?;
1324 let room_infos_store = tx.object_store(keys::ROOM_INFOS)?;
1325 let room_info = room_info_v1_json(room_id, RoomState::Joined, None, None);
1326 room_infos_store
1327 .put(&serialize_value(None, &room_info)?)
1328 .with_key(encode_key(None, keys::ROOM_INFOS, room_id))
1329 .build()?;
1330
1331 let stripped_members_store = tx.object_store(old_keys::STRIPPED_MEMBERS)?;
1332 stripped_members_store
1333 .put(&serialize_value(None, &stripped_member_event)?)
1334 .with_key(encode_key(
1335 None,
1336 old_keys::STRIPPED_MEMBERS,
1337 (stripped_room_id, stripped_user_id),
1338 ))
1339 .build()?;
1340 let stripped_room_infos_store = tx.object_store(old_keys::STRIPPED_ROOM_INFOS)?;
1341 let stripped_room_info =
1342 room_info_v1_json(stripped_room_id, RoomState::Invited, None, None);
1343 stripped_room_infos_store
1344 .put(&serialize_value(None, &stripped_room_info)?)
1345 .with_key(encode_key(None, old_keys::STRIPPED_ROOM_INFOS, stripped_room_id))
1346 .build()?;
1347
1348 tx.commit().await?;
1349 db.close();
1350 }
1351
1352 let store = IndexeddbStateStore::builder().name(name).build().await?;
1354
1355 assert_let!(
1356 Ok(Some(RawMemberEvent::Sync(stored_member_event))) =
1357 store.get_member_event(room_id, user_id).await
1358 );
1359 assert_eq!(stored_member_event.json().get(), member_event.json().get());
1360
1361 assert_let!(
1362 Ok(Some(RawMemberEvent::Stripped(stored_stripped_member_event))) =
1363 store.get_member_event(stripped_room_id, stripped_user_id).await
1364 );
1365 assert_eq!(stored_stripped_member_event.json().get(), stripped_member_event.json().get());
1366
1367 Ok(())
1368 }
1369
1370 #[async_test]
1371 pub async fn test_migrating_to_v6() -> Result<()> {
1372 let name = format!("migrating-v6-{}", Uuid::new_v4().as_hyphenated().to_string());
1373
1374 let room_id = room_id!("!room:localhost");
1375 let invite_member_event =
1376 Raw::new(&*test_json::MEMBER_INVITE).unwrap().cast_unchecked::<SyncRoomMemberEvent>();
1377 let invite_user_id = user_id!("@invited:localhost");
1378 let ban_member_event =
1379 Raw::new(&*test_json::MEMBER_BAN).unwrap().cast_unchecked::<SyncRoomMemberEvent>();
1380 let ban_user_id = user_id!("@banned:localhost");
1381
1382 let stripped_room_id = room_id!("!stripped_room:localhost");
1383 let stripped_member_event = Raw::new(&*test_json::MEMBER_STRIPPED)
1384 .unwrap()
1385 .cast_unchecked::<StrippedRoomMemberEvent>();
1386 let stripped_user_id = user_id!("@example:localhost");
1387
1388 {
1390 let db = create_fake_db(&name, 5).await?;
1391 let tx = db
1392 .transaction([
1393 keys::ROOM_STATE,
1394 keys::ROOM_INFOS,
1395 keys::STRIPPED_ROOM_STATE,
1396 old_keys::STRIPPED_ROOM_INFOS,
1397 old_keys::INVITED_USER_IDS,
1398 old_keys::JOINED_USER_IDS,
1399 old_keys::STRIPPED_INVITED_USER_IDS,
1400 old_keys::STRIPPED_JOINED_USER_IDS,
1401 ])
1402 .with_mode(TransactionMode::Readwrite)
1403 .build()?;
1404
1405 let state_store = tx.object_store(keys::ROOM_STATE)?;
1406 state_store
1407 .put(&serialize_value(None, &invite_member_event)?)
1408 .with_key(encode_key(
1409 None,
1410 keys::ROOM_STATE,
1411 (room_id, StateEventType::RoomMember, invite_user_id),
1412 ))
1413 .build()?;
1414 state_store
1415 .put(&serialize_value(None, &ban_member_event)?)
1416 .with_key(encode_key(
1417 None,
1418 keys::ROOM_STATE,
1419 (room_id, StateEventType::RoomMember, ban_user_id),
1420 ))
1421 .build()?;
1422 let room_infos_store = tx.object_store(keys::ROOM_INFOS)?;
1423 let room_info = room_info_v1_json(room_id, RoomState::Joined, None, None);
1424 room_infos_store
1425 .put(&serialize_value(None, &room_info)?)
1426 .with_key(encode_key(None, keys::ROOM_INFOS, room_id))
1427 .build()?;
1428
1429 let stripped_state_store = tx.object_store(keys::STRIPPED_ROOM_STATE)?;
1430 stripped_state_store
1431 .put(&serialize_value(None, &stripped_member_event)?)
1432 .with_key(encode_key(
1433 None,
1434 keys::STRIPPED_ROOM_STATE,
1435 (stripped_room_id, StateEventType::RoomMember, stripped_user_id),
1436 ))
1437 .build()?;
1438 let stripped_room_infos_store = tx.object_store(old_keys::STRIPPED_ROOM_INFOS)?;
1439 let stripped_room_info =
1440 room_info_v1_json(stripped_room_id, RoomState::Invited, None, None);
1441 stripped_room_infos_store
1442 .put(&serialize_value(None, &stripped_room_info)?)
1443 .with_key(encode_key(None, old_keys::STRIPPED_ROOM_INFOS, stripped_room_id))
1444 .build()?;
1445
1446 let joined_user_id = user_id!("@joined_user:localhost");
1448 tx.object_store(old_keys::JOINED_USER_IDS)?
1449 .put(&serialize_value(None, &joined_user_id)?)
1450 .with_key(encode_key(None, old_keys::JOINED_USER_IDS, (room_id, joined_user_id)))
1451 .build()?;
1452 let invited_user_id = user_id!("@invited_user:localhost");
1453 tx.object_store(old_keys::INVITED_USER_IDS)?
1454 .put(&serialize_value(None, &invited_user_id)?)
1455 .with_key(encode_key(None, old_keys::INVITED_USER_IDS, (room_id, invited_user_id)))
1456 .build()?;
1457 let stripped_joined_user_id = user_id!("@stripped_joined_user:localhost");
1458 tx.object_store(old_keys::STRIPPED_JOINED_USER_IDS)?
1459 .put(&serialize_value(None, &stripped_joined_user_id)?)
1460 .with_key(encode_key(
1461 None,
1462 old_keys::STRIPPED_JOINED_USER_IDS,
1463 (room_id, stripped_joined_user_id),
1464 ))
1465 .build()?;
1466 let stripped_invited_user_id = user_id!("@stripped_invited_user:localhost");
1467 tx.object_store(old_keys::STRIPPED_INVITED_USER_IDS)?
1468 .put(&serialize_value(None, &stripped_invited_user_id)?)
1469 .with_key(encode_key(
1470 None,
1471 old_keys::STRIPPED_INVITED_USER_IDS,
1472 (room_id, stripped_invited_user_id),
1473 ))
1474 .build()?;
1475
1476 tx.commit().await?;
1477 db.close();
1478 }
1479
1480 let store = IndexeddbStateStore::builder().name(name).build().await?;
1482
1483 assert_eq!(store.get_user_ids(room_id, RoomMemberships::JOIN).await.unwrap().len(), 0);
1484 assert_eq!(
1485 store.get_user_ids(room_id, RoomMemberships::INVITE).await.unwrap().as_slice(),
1486 [invite_user_id.to_owned()]
1487 );
1488 let user_ids = store.get_user_ids(room_id, RoomMemberships::empty()).await.unwrap();
1489 assert_eq!(user_ids.len(), 2);
1490 assert!(user_ids.contains(&invite_user_id.to_owned()));
1491 assert!(user_ids.contains(&ban_user_id.to_owned()));
1492
1493 assert_eq!(
1494 store.get_user_ids(stripped_room_id, RoomMemberships::JOIN).await.unwrap().as_slice(),
1495 [stripped_user_id.to_owned()]
1496 );
1497 assert_eq!(
1498 store.get_user_ids(stripped_room_id, RoomMemberships::INVITE).await.unwrap().len(),
1499 0
1500 );
1501 assert_eq!(
1502 store
1503 .get_user_ids(stripped_room_id, RoomMemberships::empty())
1504 .await
1505 .unwrap()
1506 .as_slice(),
1507 [stripped_user_id.to_owned()]
1508 );
1509
1510 Ok(())
1511 }
1512
1513 #[async_test]
1514 pub async fn test_migrating_to_v7() -> Result<()> {
1515 let name = format!("migrating-v7-{}", Uuid::new_v4().as_hyphenated().to_string());
1516
1517 let room_id = room_id!("!room:localhost");
1518 let stripped_room_id = room_id!("!stripped_room:localhost");
1519
1520 {
1522 let db = create_fake_db(&name, 6).await?;
1523 let tx = db
1524 .transaction([keys::ROOM_INFOS, old_keys::STRIPPED_ROOM_INFOS])
1525 .with_mode(TransactionMode::Readwrite)
1526 .build()?;
1527
1528 let room_infos_store = tx.object_store(keys::ROOM_INFOS)?;
1529 let room_info = room_info_v1_json(room_id, RoomState::Joined, None, None);
1530 room_infos_store
1531 .put(&serialize_value(None, &room_info)?)
1532 .with_key(encode_key(None, keys::ROOM_INFOS, room_id))
1533 .build()?;
1534
1535 let stripped_room_infos_store = tx.object_store(old_keys::STRIPPED_ROOM_INFOS)?;
1536 let stripped_room_info =
1537 room_info_v1_json(stripped_room_id, RoomState::Invited, None, None);
1538 stripped_room_infos_store
1539 .put(&serialize_value(None, &stripped_room_info)?)
1540 .with_key(encode_key(None, old_keys::STRIPPED_ROOM_INFOS, stripped_room_id))
1541 .build()?;
1542
1543 tx.commit().await?;
1544 db.close();
1545 }
1546
1547 let store = IndexeddbStateStore::builder().name(name).build().await?;
1549
1550 assert_eq!(store.get_room_infos(&RoomLoadSettings::default()).await.unwrap().len(), 2);
1551
1552 Ok(())
1553 }
1554
1555 fn add_room_v7(
1557 room_infos_store: &ObjectStore<'_>,
1558 room_state_store: &ObjectStore<'_>,
1559 room_id: &RoomId,
1560 name: Option<&str>,
1561 create_creator: Option<OwnedUserId>,
1562 create_sender: Option<&UserId>,
1563 ) -> Result<()> {
1564 let room_info_json =
1565 room_info_v1_json(room_id, RoomState::Joined, name, create_creator.as_deref());
1566
1567 room_infos_store
1568 .put(&serialize_value(None, &room_info_json)?)
1569 .with_key(encode_key(None, keys::ROOM_INFOS, room_id))
1570 .build()?;
1571
1572 let Some(create_sender) = create_sender else {
1574 return Ok(());
1575 };
1576
1577 let create_content = match create_creator {
1578 Some(creator) => RoomCreateEventContent::new_v1(creator),
1579 None => RoomCreateEventContent::new_v11(),
1580 };
1581
1582 let event_id = EventId::new(server_name!("dummy.local"));
1583 let create_event = json!({
1584 "content": create_content,
1585 "event_id": event_id,
1586 "sender": create_sender,
1587 "origin_server_ts": MilliSecondsSinceUnixEpoch::now(),
1588 "state_key": "",
1589 "type": "m.room.create",
1590 "unsigned": {},
1591 });
1592
1593 room_state_store
1594 .put(&serialize_value(None, &create_event)?)
1595 .with_key(encode_key(
1596 None,
1597 keys::ROOM_STATE,
1598 (room_id, &StateEventType::RoomCreate, ""),
1599 ))
1600 .build()?;
1601
1602 Ok(())
1603 }
1604
1605 #[async_test]
1606 pub async fn test_migrating_to_v8() -> Result<()> {
1607 let name = format!("migrating-v8-{}", Uuid::new_v4().as_hyphenated().to_string());
1608
1609 let room_a_id = room_id!("!room_a:dummy.local");
1611 let room_a_name = "Room A";
1612 let room_a_creator = owned_user_id!("@creator:dummy.local");
1613 let room_a_create_sender = owned_user_id!("@sender:dummy.local");
1616
1617 let room_b_id = room_id!("!room_b:dummy.local");
1619
1620 let room_c_id = room_id!("!room_c:dummy.local");
1622 let room_c_create_sender = owned_user_id!("@creator:dummy.local");
1623
1624 {
1626 let db = create_fake_db(&name, 6).await?;
1627 let tx = db
1628 .transaction([keys::ROOM_INFOS, keys::ROOM_STATE])
1629 .with_mode(TransactionMode::Readwrite)
1630 .build()?;
1631
1632 let room_infos_store = tx.object_store(keys::ROOM_INFOS)?;
1633 let room_state_store = tx.object_store(keys::ROOM_STATE)?;
1634
1635 add_room_v7(
1636 &room_infos_store,
1637 &room_state_store,
1638 room_a_id,
1639 Some(room_a_name),
1640 Some(room_a_creator),
1641 Some(&room_a_create_sender),
1642 )?;
1643 add_room_v7(&room_infos_store, &room_state_store, room_b_id, None, None, None)?;
1644 add_room_v7(
1645 &room_infos_store,
1646 &room_state_store,
1647 room_c_id,
1648 None,
1649 None,
1650 Some(&room_c_create_sender),
1651 )?;
1652
1653 tx.commit().await?;
1654 db.close();
1655 }
1656
1657 let store = IndexeddbStateStore::builder().name(name).build().await?;
1659
1660 let room_infos = store.get_room_infos(&RoomLoadSettings::default()).await?;
1662 assert_eq!(room_infos.len(), 3);
1663
1664 let room_a = room_infos.iter().find(|r| r.room_id() == room_a_id).unwrap();
1665 assert_eq!(room_a.name(), Some(room_a_name));
1666 assert_eq!(room_a.creators(), Some(vec![room_a_create_sender]));
1667
1668 let room_b = room_infos.iter().find(|r| r.room_id() == room_b_id).unwrap();
1669 assert_eq!(room_b.name(), None);
1670 assert_eq!(room_b.creators(), None);
1671
1672 let room_c = room_infos.iter().find(|r| r.room_id() == room_c_id).unwrap();
1673 assert_eq!(room_c.name(), None);
1674 assert_eq!(room_c.creators(), Some(vec![room_c_create_sender]));
1675
1676 Ok(())
1677 }
1678}