2023-11-09 03:48:17 +01:00
|
|
|
class PersistentStorage {
|
|
|
|
constructor() {
|
|
|
|
if (!('indexedDB' in window)) {
|
|
|
|
PersistentStorage.logBrowserNotCapable();
|
|
|
|
return;
|
|
|
|
}
|
2024-02-22 15:28:29 +01:00
|
|
|
const DBOpenRequest = window.indexedDB.open('pairdrop_store', 6);
|
2023-11-09 03:48:17 +01:00
|
|
|
DBOpenRequest.onerror = e => {
|
|
|
|
PersistentStorage.logBrowserNotCapable();
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.error('Error initializing database:', e);
|
2023-11-09 03:48:17 +01:00
|
|
|
};
|
|
|
|
DBOpenRequest.onsuccess = _ => {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug('Database initialised.');
|
2023-11-09 03:48:17 +01:00
|
|
|
};
|
2023-11-24 17:32:40 +01:00
|
|
|
DBOpenRequest.onupgradeneeded = async e => {
|
2023-11-09 03:48:17 +01:00
|
|
|
const db = e.target.result;
|
|
|
|
const txn = e.target.transaction;
|
|
|
|
|
2024-02-05 15:42:27 +01:00
|
|
|
db.onerror = e => Logger.error('Error loading database:', e);
|
2023-11-09 03:48:17 +01:00
|
|
|
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug(`Upgrading IndexedDB database from version ${e.oldVersion} to version ${e.newVersion}`);
|
2023-11-09 03:48:17 +01:00
|
|
|
|
|
|
|
if (e.oldVersion === 0) {
|
|
|
|
// initiate v1
|
|
|
|
db.createObjectStore('keyval');
|
|
|
|
let roomSecretsObjectStore1 = db.createObjectStore('room_secrets', {autoIncrement: true});
|
|
|
|
roomSecretsObjectStore1.createIndex('secret', 'secret', { unique: true });
|
|
|
|
}
|
|
|
|
if (e.oldVersion <= 1) {
|
|
|
|
// migrate to v2
|
|
|
|
db.createObjectStore('share_target_files');
|
|
|
|
}
|
|
|
|
if (e.oldVersion <= 2) {
|
|
|
|
// migrate to v3
|
|
|
|
db.deleteObjectStore('share_target_files');
|
|
|
|
db.createObjectStore('share_target_files', {autoIncrement: true});
|
|
|
|
}
|
|
|
|
if (e.oldVersion <= 3) {
|
|
|
|
// migrate to v4
|
|
|
|
let roomSecretsObjectStore4 = txn.objectStore('room_secrets');
|
|
|
|
roomSecretsObjectStore4.createIndex('display_name', 'display_name');
|
|
|
|
roomSecretsObjectStore4.createIndex('auto_accept', 'auto_accept');
|
|
|
|
}
|
2023-11-24 17:32:40 +01:00
|
|
|
if (e.oldVersion <= 4) {
|
|
|
|
// migrate to v5
|
|
|
|
const editedDisplayNameOld = await PersistentStorage.get('editedDisplayName');
|
|
|
|
if (editedDisplayNameOld) {
|
|
|
|
await PersistentStorage.set('edited_display_name', editedDisplayNameOld);
|
|
|
|
await PersistentStorage.delete('editedDisplayName');
|
|
|
|
}
|
|
|
|
}
|
2024-02-22 15:28:29 +01:00
|
|
|
if (e.oldVersion <= 5) {
|
|
|
|
// migrate to v6
|
|
|
|
let roomSecretsObjectStore5 = txn.objectStore('room_secrets');
|
|
|
|
roomSecretsObjectStore5.createIndex('ws_domain', 'ws_domain');
|
|
|
|
// add current ws_domain to existing peer secret entries once the config has loaded
|
|
|
|
Events.on('config-loaded', _ => PersistentStorage.addCurrentWsDomainToAllRoomSecrets(), { once: true });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static getCurrentWsDomain() {
|
|
|
|
return window._config && window._config.signalingServer
|
|
|
|
? window._config.signalingServer
|
|
|
|
: location.host + location.pathname;
|
|
|
|
}
|
|
|
|
|
|
|
|
static async addCurrentWsDomainToAllRoomSecrets() {
|
|
|
|
const wsServerDomain = this.getCurrentWsDomain();
|
|
|
|
|
|
|
|
const roomSecrets = await PersistentStorage.getAllRoomSecrets(false);
|
|
|
|
for (let i = 0; i < roomSecrets.length; i++) {
|
|
|
|
await PersistentStorage.updateRoomSecret(roomSecrets[i], null, null, null, null, wsServerDomain);
|
2023-11-09 03:48:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static logBrowserNotCapable() {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.log("This browser does not support IndexedDB. Paired devices will be gone after the browser is closed.");
|
2023-11-09 03:48:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static set(key, value) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const DBOpenRequest = window.indexedDB.open('pairdrop_store');
|
|
|
|
DBOpenRequest.onsuccess = e => {
|
|
|
|
const db = e.target.result;
|
|
|
|
const transaction = db.transaction('keyval', 'readwrite');
|
|
|
|
const objectStore = transaction.objectStore('keyval');
|
|
|
|
const objectStoreRequest = objectStore.put(value, key);
|
|
|
|
objectStoreRequest.onsuccess = _ => {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug(`Request successful. Added key-pair: ${key} - ${value}`);
|
2023-11-09 03:48:17 +01:00
|
|
|
resolve(value);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
DBOpenRequest.onerror = e => {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
static get(key) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const DBOpenRequest = window.indexedDB.open('pairdrop_store');
|
|
|
|
DBOpenRequest.onsuccess = e => {
|
|
|
|
const db = e.target.result;
|
|
|
|
const transaction = db.transaction('keyval', 'readonly');
|
|
|
|
const objectStore = transaction.objectStore('keyval');
|
|
|
|
const objectStoreRequest = objectStore.get(key);
|
|
|
|
objectStoreRequest.onsuccess = _ => {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug(`Request successful. Retrieved key-pair: ${key} - ${objectStoreRequest.result}`);
|
2023-11-09 03:48:17 +01:00
|
|
|
resolve(objectStoreRequest.result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DBOpenRequest.onerror = e => {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static delete(key) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const DBOpenRequest = window.indexedDB.open('pairdrop_store');
|
|
|
|
DBOpenRequest.onsuccess = e => {
|
|
|
|
const db = e.target.result;
|
|
|
|
const transaction = db.transaction('keyval', 'readwrite');
|
|
|
|
const objectStore = transaction.objectStore('keyval');
|
|
|
|
const objectStoreRequest = objectStore.delete(key);
|
|
|
|
objectStoreRequest.onsuccess = _ => {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug(`Request successful. Deleted key: ${key}`);
|
2023-11-09 03:48:17 +01:00
|
|
|
resolve();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
DBOpenRequest.onerror = e => {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
static addRoomSecret(roomSecret, displayName, deviceName) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const DBOpenRequest = window.indexedDB.open('pairdrop_store');
|
|
|
|
DBOpenRequest.onsuccess = e => {
|
|
|
|
const db = e.target.result;
|
|
|
|
const transaction = db.transaction('room_secrets', 'readwrite');
|
|
|
|
const objectStore = transaction.objectStore('room_secrets');
|
|
|
|
const objectStoreRequest = objectStore.add({
|
|
|
|
'secret': roomSecret,
|
|
|
|
'display_name': displayName,
|
|
|
|
'device_name': deviceName,
|
2024-02-22 15:28:29 +01:00
|
|
|
'auto_accept': false,
|
|
|
|
'ws_domain': PersistentStorage.getCurrentWsDomain()
|
2023-11-09 03:48:17 +01:00
|
|
|
});
|
|
|
|
objectStoreRequest.onsuccess = e => {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug(`Request successful. RoomSecret added: ${e.target.result}`);
|
2023-11-09 03:48:17 +01:00
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DBOpenRequest.onerror = e => {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-02-22 15:28:29 +01:00
|
|
|
static async getAllRoomSecretsCount(currentWsDomainOnly = true) {
|
|
|
|
return (await PersistentStorage.getAllRoomSecrets(currentWsDomainOnly)).length;
|
|
|
|
}
|
|
|
|
|
|
|
|
static async getAllRoomSecrets(currentWsDomainOnly = true) {
|
|
|
|
let secrets = [];
|
2023-11-09 03:48:17 +01:00
|
|
|
try {
|
2024-02-22 15:28:29 +01:00
|
|
|
const roomSecrets = await this.getAllRoomSecretEntries(currentWsDomainOnly);
|
|
|
|
|
|
|
|
secrets = roomSecrets.map(roomSecret => roomSecret.secret);
|
|
|
|
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug(`Request successful. Retrieved ${secrets.length} room_secrets`);
|
2024-02-22 15:28:29 +01:00
|
|
|
}
|
|
|
|
catch (e) {
|
|
|
|
console.debug(e)
|
2023-11-09 03:48:17 +01:00
|
|
|
this.logBrowserNotCapable();
|
|
|
|
}
|
2024-02-22 15:28:29 +01:00
|
|
|
|
|
|
|
return secrets;
|
2023-11-09 03:48:17 +01:00
|
|
|
}
|
|
|
|
|
2024-02-22 15:28:29 +01:00
|
|
|
static getAllRoomSecretEntries(currentWsDomainOnly = true) {
|
2023-11-09 03:48:17 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const DBOpenRequest = window.indexedDB.open('pairdrop_store');
|
|
|
|
DBOpenRequest.onsuccess = (e) => {
|
|
|
|
const db = e.target.result;
|
|
|
|
const transaction = db.transaction('room_secrets', 'readonly');
|
|
|
|
const objectStore = transaction.objectStore('room_secrets');
|
|
|
|
const objectStoreRequest = objectStore.getAll();
|
|
|
|
objectStoreRequest.onsuccess = e => {
|
2024-02-22 15:28:29 +01:00
|
|
|
let roomSecrets = e.target.result;
|
|
|
|
let roomSecretEntries = [];
|
|
|
|
|
|
|
|
for (let i = 0; i < roomSecrets.length; i++) {
|
|
|
|
const currentWsDomainDiffers = roomSecrets[i].ws_domain !== PersistentStorage.getCurrentWsDomain();
|
|
|
|
|
|
|
|
// if the saved ws domain differs from the current ws domain and only peers for the current ws domain should be returned -> skip this entry
|
|
|
|
if (currentWsDomainOnly && currentWsDomainDiffers) continue;
|
|
|
|
|
|
|
|
roomSecretEntries.push(roomSecrets[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve(roomSecretEntries);
|
2023-11-09 03:48:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
DBOpenRequest.onerror = (e) => {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static getRoomSecretEntry(roomSecret) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const DBOpenRequest = window.indexedDB.open('pairdrop_store');
|
|
|
|
DBOpenRequest.onsuccess = e => {
|
|
|
|
const db = e.target.result;
|
|
|
|
const transaction = db.transaction('room_secrets', 'readonly');
|
|
|
|
const objectStore = transaction.objectStore('room_secrets');
|
|
|
|
const objectStoreRequestKey = objectStore.index("secret").getKey(roomSecret);
|
|
|
|
objectStoreRequestKey.onsuccess = e => {
|
|
|
|
const key = e.target.result;
|
|
|
|
if (!key) {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug(`Nothing to retrieve. Entry for room_secret not existing: ${roomSecret}`);
|
2023-11-09 03:48:17 +01:00
|
|
|
resolve();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const objectStoreRequestRetrieval = objectStore.get(key);
|
|
|
|
objectStoreRequestRetrieval.onsuccess = e => {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug(`Request successful. Retrieved entry for room_secret: ${key}`);
|
2023-11-09 03:48:17 +01:00
|
|
|
resolve({
|
|
|
|
"entry": e.target.result,
|
|
|
|
"key": key
|
|
|
|
});
|
|
|
|
}
|
|
|
|
objectStoreRequestRetrieval.onerror = (e) => {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
DBOpenRequest.onerror = (e) => {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static deleteRoomSecret(roomSecret) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const DBOpenRequest = window.indexedDB.open('pairdrop_store');
|
|
|
|
DBOpenRequest.onsuccess = (e) => {
|
|
|
|
const db = e.target.result;
|
|
|
|
const transaction = db.transaction('room_secrets', 'readwrite');
|
|
|
|
const objectStore = transaction.objectStore('room_secrets');
|
|
|
|
const objectStoreRequestKey = objectStore.index("secret").getKey(roomSecret);
|
|
|
|
objectStoreRequestKey.onsuccess = e => {
|
|
|
|
if (!e.target.result) {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug(`Nothing to delete. room_secret not existing: ${roomSecret}`);
|
2023-11-09 03:48:17 +01:00
|
|
|
resolve();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const key = e.target.result;
|
|
|
|
const objectStoreRequestDeletion = objectStore.delete(key);
|
|
|
|
objectStoreRequestDeletion.onsuccess = _ => {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug(`Request successful. Deleted room_secret: ${key}`);
|
2023-11-09 03:48:17 +01:00
|
|
|
resolve(roomSecret);
|
|
|
|
}
|
|
|
|
objectStoreRequestDeletion.onerror = e => {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
DBOpenRequest.onerror = e => {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
static clearRoomSecrets() {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const DBOpenRequest = window.indexedDB.open('pairdrop_store');
|
|
|
|
DBOpenRequest.onsuccess = (e) => {
|
|
|
|
const db = e.target.result;
|
|
|
|
const transaction = db.transaction('room_secrets', 'readwrite');
|
|
|
|
const objectStore = transaction.objectStore('room_secrets');
|
|
|
|
const objectStoreRequest = objectStore.clear();
|
|
|
|
objectStoreRequest.onsuccess = _ => {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug('Request successful. All room_secrets cleared');
|
2023-11-09 03:48:17 +01:00
|
|
|
resolve();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
DBOpenRequest.onerror = e => {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-02-04 18:05:11 +01:00
|
|
|
static updateRoomSecretDisplayName(roomSecret, displayName) {
|
|
|
|
return this.updateRoomSecret(roomSecret, null, displayName, null);
|
2023-11-09 03:48:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static updateRoomSecretAutoAccept(roomSecret, autoAccept) {
|
2024-02-04 18:05:11 +01:00
|
|
|
return this.updateRoomSecret(roomSecret, null, null, null, autoAccept);
|
2023-11-09 03:48:17 +01:00
|
|
|
}
|
|
|
|
|
2024-02-22 15:28:29 +01:00
|
|
|
static updateRoomSecret(roomSecret, updatedRoomSecret = null, updatedDisplayName = null, updatedDeviceName = null, updatedAutoAccept = null, wsDomain = null) {
|
2023-11-09 03:48:17 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const DBOpenRequest = window.indexedDB.open('pairdrop_store');
|
|
|
|
DBOpenRequest.onsuccess = e => {
|
|
|
|
const db = e.target.result;
|
|
|
|
this.getRoomSecretEntry(roomSecret)
|
|
|
|
.then(roomSecretEntry => {
|
|
|
|
if (!roomSecretEntry) {
|
|
|
|
resolve(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const transaction = db.transaction('room_secrets', 'readwrite');
|
|
|
|
const objectStore = transaction.objectStore('room_secrets');
|
|
|
|
// Do not use `updatedRoomSecret ?? roomSecretEntry.entry.secret` to ensure compatibility with older browsers
|
|
|
|
const updatedRoomSecretEntry = {
|
2024-02-04 18:05:11 +01:00
|
|
|
'secret': updatedRoomSecret !== null ? updatedRoomSecret : roomSecretEntry.entry.secret,
|
|
|
|
'display_name': updatedDisplayName !== null ? updatedDisplayName : roomSecretEntry.entry.display_name,
|
|
|
|
'device_name': updatedDeviceName !== null ? updatedDeviceName : roomSecretEntry.entry.device_name,
|
2024-02-22 15:28:29 +01:00
|
|
|
'auto_accept': updatedAutoAccept !== null ? updatedAutoAccept : roomSecretEntry.entry.auto_accept,
|
|
|
|
'ws_domain': wsDomain !== null ? wsDomain : roomSecretEntry.entry.ws_domain
|
2023-11-09 03:48:17 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
const objectStoreRequestUpdate = objectStore.put(updatedRoomSecretEntry, roomSecretEntry.key);
|
|
|
|
|
|
|
|
objectStoreRequestUpdate.onsuccess = e => {
|
2024-02-05 15:42:27 +01:00
|
|
|
Logger.debug(`Request successful. Updated room_secret: ${roomSecretEntry.key}`);
|
2023-11-09 03:48:17 +01:00
|
|
|
resolve({
|
|
|
|
"entry": updatedRoomSecretEntry,
|
|
|
|
"key": roomSecretEntry.key
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
objectStoreRequestUpdate.onerror = (e) => {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.catch(e => reject(e));
|
|
|
|
};
|
|
|
|
|
|
|
|
DBOpenRequest.onerror = e => reject(e);
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|