a73694deb4
## Ticket 1572 Always check for availability before use. ## Changes - Consolidated the keys into one place for easier tracking. - It'll also be easier to code using constant autocomplete. - Cleaned up the wrapper to be as close as the original. - Updated existing code to use the wrapper (even if they already handled the availability) to encourage future code to just use the wrapper.
138 lines
4.3 KiB
JavaScript
138 lines
4.3 KiB
JavaScript
// @flow
|
|
|
|
/**
|
|
* This serves a bridge between tabs using localStorage to indicate whether an
|
|
* upload is currently in progress (locked) or removed.
|
|
*
|
|
* An alternative is to sync the redux's 'publish::currentUploads' through the
|
|
* wallet's sync process, but let's not pollute the wallet for now.
|
|
*/
|
|
|
|
import { v4 as uuid } from 'uuid';
|
|
import { isLocalStorageAvailable, LocalStorage, LS } from 'util/storage';
|
|
import { doUpdateUploadRemove, doUpdateUploadProgress } from 'redux/actions/publish';
|
|
|
|
const localStorageAvailable = isLocalStorageAvailable();
|
|
|
|
let gTabId: string = '';
|
|
|
|
function getTabId() {
|
|
if (!gTabId) {
|
|
// We want to maximize bootup speed, so only initialize
|
|
// the tab ID on first use instead when declared.
|
|
gTabId = uuid();
|
|
}
|
|
return gTabId;
|
|
}
|
|
|
|
// ****************************************************************************
|
|
// Locked
|
|
// ****************************************************************************
|
|
|
|
function getLockedUploads() {
|
|
if (localStorageAvailable) {
|
|
const storedValue = LocalStorage.getItem(LS.TUS_LOCKED_UPLOADS);
|
|
return storedValue ? JSON.parse(storedValue) : {};
|
|
}
|
|
return {};
|
|
}
|
|
|
|
export function tusIsSessionLocked(guid: string) {
|
|
const lockedUploads = getLockedUploads();
|
|
return lockedUploads[guid] && lockedUploads[guid] !== getTabId();
|
|
}
|
|
|
|
export function tusLockAndNotify(guid: string) {
|
|
const lockedUploads = getLockedUploads();
|
|
if (!lockedUploads[guid] && localStorageAvailable) {
|
|
lockedUploads[guid] = getTabId();
|
|
LocalStorage.setItem(LS.TUS_LOCKED_UPLOADS, JSON.stringify(lockedUploads));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* tusUnlockAndNotify
|
|
*
|
|
* @param guid The upload session to unlock and notify other tabs of.
|
|
* Passing 'undefined' will clear all sessions locked by this tab.
|
|
*/
|
|
export function tusUnlockAndNotify(guid?: string) {
|
|
if (!localStorageAvailable) return;
|
|
|
|
const lockedUploads = getLockedUploads();
|
|
|
|
if (guid) {
|
|
delete lockedUploads[guid];
|
|
} else {
|
|
const ourTabId = getTabId();
|
|
const lockedUploadsEntries = Object.entries(lockedUploads);
|
|
lockedUploadsEntries.forEach(([lockedGuid, tabId]) => {
|
|
if (tabId === ourTabId) {
|
|
delete lockedUploads[lockedGuid];
|
|
}
|
|
});
|
|
}
|
|
|
|
if (Object.keys(lockedUploads).length > 0) {
|
|
LocalStorage.setItem(LS.TUS_LOCKED_UPLOADS, JSON.stringify(lockedUploads));
|
|
} else {
|
|
LocalStorage.removeItem(LS.TUS_LOCKED_UPLOADS);
|
|
}
|
|
}
|
|
|
|
// ****************************************************************************
|
|
// Removed
|
|
// ****************************************************************************
|
|
|
|
function getRemovedUploads() {
|
|
if (localStorageAvailable) {
|
|
const storedValue = LocalStorage.getItem(LS.TUS_REMOVED_UPLOADS);
|
|
return storedValue ? storedValue.split(',') : [];
|
|
}
|
|
return [];
|
|
}
|
|
|
|
export function tusRemoveAndNotify(guid: string) {
|
|
if (!localStorageAvailable) return;
|
|
const removedUploads = getRemovedUploads();
|
|
removedUploads.push(guid);
|
|
LocalStorage.setItem(LS.TUS_REMOVED_UPLOADS, removedUploads.join(','));
|
|
}
|
|
|
|
export function tusClearRemovedUploads() {
|
|
if (!localStorageAvailable) return;
|
|
LocalStorage.removeItem(LS.TUS_REMOVED_UPLOADS);
|
|
}
|
|
|
|
export function tusClearLockedUploads() {
|
|
if (!localStorageAvailable) return;
|
|
LocalStorage.removeItem(LS.TUS_LOCKED_UPLOADS);
|
|
LocalStorage.setItem(LS.TUS_REFRESH_LOCK, String(Math.random()));
|
|
}
|
|
|
|
// ****************************************************************************
|
|
// Respond to changes from other tabs.
|
|
// ****************************************************************************
|
|
|
|
export function tusHandleTabUpdates(storageKey: string) {
|
|
switch (storageKey) {
|
|
case LS.TUS_LOCKED_UPLOADS:
|
|
// The locked IDs are in localStorage, but related GUI is unaware.
|
|
// Send a redux update to force an update.
|
|
window.store.dispatch(doUpdateUploadProgress({ guid: 'force--update' }));
|
|
break;
|
|
|
|
case LS.TUS_REFRESH_LOCK:
|
|
window.store.dispatch(doUpdateUploadProgress({ guid: 'refresh--lock' }));
|
|
break;
|
|
|
|
case LS.TUS_REMOVED_UPLOADS:
|
|
// The other tab's store has removed this upload, so it's safe to do the
|
|
// same without affecting rehydration.
|
|
if (localStorageAvailable) {
|
|
const removedUploads = getRemovedUploads();
|
|
removedUploads.forEach((guid) => window.store.dispatch(doUpdateUploadRemove(guid)));
|
|
}
|
|
break;
|
|
}
|
|
}
|