fox-send/app/fileManager.js

283 lines
7.3 KiB
JavaScript
Raw Normal View History

2018-08-07 22:40:17 +00:00
/* global DEFAULTS LIMITS */
import FileSender from './fileSender';
import FileReceiver from './fileReceiver';
2018-07-31 18:09:18 +00:00
import { copyToClipboard, delay, openLinksInNewTab, percent } from './utils';
import * as metrics from './metrics';
2018-07-31 18:09:18 +00:00
import Archive from './archive';
2018-08-03 19:24:41 +00:00
import { bytes } from './utils';
2018-10-25 02:32:53 +00:00
import okDialog from './ui/okDialog';
2018-10-26 01:55:11 +00:00
import copyDialog from './ui/copyDialog';
export default function(state, emitter) {
let lastRender = 0;
2017-09-13 19:01:55 +00:00
let updateTitle = false;
function render() {
emitter.emit('render');
}
async function checkFiles() {
2018-08-07 22:40:17 +00:00
const changes = await state.user.syncFileList();
const rerender = changes.incoming || changes.downloadCount;
if (rerender) {
render();
}
}
2017-09-13 19:01:55 +00:00
function updateProgress() {
if (updateTitle) {
emitter.emit('DOMTitleChange', percent(state.transfer.progressRatio));
}
render();
}
emitter.on('DOMContentLoaded', () => {
document.addEventListener('blur', () => (updateTitle = true));
document.addEventListener('focus', () => {
updateTitle = false;
emitter.emit('DOMTitleChange', 'Firefox Send');
});
checkFiles();
});
2018-06-25 21:01:08 +00:00
emitter.on('navigate', checkFiles);
emitter.on('render', () => {
lastRender = Date.now();
});
emitter.on('login', email => {
state.user.login(email);
2018-08-07 22:40:17 +00:00
});
emitter.on('logout', () => {
state.user.logout();
2018-09-24 22:25:08 +00:00
state.timeLimit = DEFAULTS.EXPIRE_SECONDS;
state.downloadCount = 1;
2018-08-31 21:20:15 +00:00
emitter.emit('pushState', '/');
2018-08-07 22:40:17 +00:00
});
2017-11-30 21:41:09 +00:00
emitter.on('changeLimit', async ({ file, value }) => {
2018-08-31 17:59:26 +00:00
const ok = await file.changeLimit(value, state.user);
if (!ok) {
// TODO
return;
}
2018-01-24 18:23:13 +00:00
state.storage.writeFile(file);
2017-11-30 21:41:09 +00:00
metrics.changedDownloadLimit(file);
});
2018-10-25 02:07:10 +00:00
emitter.on('removeUpload', file => {
state.archive.remove(file);
if (state.archive.numFiles === 0) {
state.archive = null;
}
2018-08-03 19:24:41 +00:00
render();
2018-07-31 18:09:18 +00:00
});
emitter.on('delete', async ({ file, location }) => {
try {
metrics.deletedUpload({
size: file.size,
time: file.time,
speed: file.speed,
type: file.type,
ttl: file.expiresAt - Date.now(),
location
});
state.storage.remove(file.id);
2018-01-24 18:23:13 +00:00
await file.del();
} catch (e) {
state.raven.captureException(e);
}
2018-10-25 02:07:10 +00:00
render();
});
emitter.on('cancel', () => {
state.transfer.cancel();
});
2018-07-31 18:09:18 +00:00
emitter.on('addFiles', async ({ files }) => {
2018-08-07 22:40:17 +00:00
const maxSize = state.user.maxSize;
state.archive = state.archive || new Archive();
try {
state.archive.addFiles(files, maxSize);
} catch (e) {
2018-09-07 17:53:40 +00:00
state.modal = okDialog(
2018-08-07 22:40:17 +00:00
state.translate(e.message, {
size: bytes(maxSize),
count: LIMITS.MAX_FILES_PER_ARCHIVE
})
);
2018-10-25 02:32:53 +00:00
if (state.archive.numFiles === 0) {
state.archive = null;
}
2018-07-31 18:09:18 +00:00
}
render();
});
2018-08-31 21:20:15 +00:00
emitter.on('upload', async ({ type, dlimit, password }) => {
2018-08-03 19:24:41 +00:00
if (!state.archive) return;
2018-08-07 22:40:17 +00:00
if (state.storage.files.length >= LIMITS.MAX_ARCHIVES_PER_USER) {
2018-09-07 17:53:40 +00:00
state.modal = okDialog(
2018-08-07 22:40:17 +00:00
state.translate('tooManyArchives', {
count: LIMITS.MAX_ARCHIVES_PER_USER
})
);
2018-09-07 17:53:40 +00:00
return render();
2018-08-07 22:40:17 +00:00
}
2018-08-03 19:24:41 +00:00
const size = state.archive.size;
2018-08-07 22:40:17 +00:00
if (!state.timeLimit) state.timeLimit = DEFAULTS.EXPIRE_SECONDS;
2018-08-31 21:20:15 +00:00
const sender = new FileSender();
2018-08-08 18:07:09 +00:00
2017-09-13 19:01:55 +00:00
sender.on('progress', updateProgress);
sender.on('encrypting', render);
2018-07-31 18:09:18 +00:00
sender.on('complete', render);
state.transfer = sender;
state.uploading = true;
render();
2018-01-24 18:23:13 +00:00
const links = openLinksInNewTab();
await delay(200);
try {
metrics.startedUpload({ size, type });
2018-07-31 18:09:18 +00:00
2018-08-31 21:20:15 +00:00
const ownedFile = await sender.upload(
state.archive,
state.timeLimit,
dlimit,
state.user.bearerToken
);
2018-02-02 18:15:17 +00:00
ownedFile.type = type;
2018-01-24 18:23:13 +00:00
state.storage.totalUploads += 1;
metrics.completedUpload(ownedFile);
state.storage.addFile(ownedFile);
2018-08-31 21:20:15 +00:00
// TODO integrate password into /upload request
2018-07-31 18:09:18 +00:00
if (password) {
emitter.emit('password', { password, file: ownedFile });
}
2018-10-29 16:52:24 +00:00
state.modal = copyDialog(ownedFile.name, ownedFile.url);
} catch (err) {
if (err.message === '0') {
//cancelled. do nothing
metrics.cancelledUpload({ size, type });
render();
} else {
// eslint-disable-next-line no-console
console.error(err);
state.raven.captureException(err);
metrics.stoppedUpload({ size, type, err });
emitter.emit('pushState', '/error');
}
} finally {
openLinksInNewTab(links, false);
2018-08-08 18:07:09 +00:00
state.archive = null;
2018-07-31 18:09:18 +00:00
state.password = '';
state.uploading = false;
state.transfer = null;
2018-10-25 02:07:10 +00:00
render();
}
});
2018-01-24 18:23:13 +00:00
emitter.on('password', async ({ password, file }) => {
try {
2018-02-16 20:56:53 +00:00
state.settingPassword = true;
render();
2018-01-24 18:23:13 +00:00
await file.setPassword(password);
state.storage.writeFile(file);
metrics.addedPassword({ size: file.size });
2018-02-16 20:56:53 +00:00
await delay(1000);
2018-01-24 18:23:13 +00:00
} catch (err) {
2018-03-02 05:36:45 +00:00
// eslint-disable-next-line no-console
2018-01-24 18:23:13 +00:00
console.error(err);
2018-02-21 20:35:52 +00:00
state.passwordSetError = err;
} finally {
state.settingPassword = false;
}
render();
});
2018-01-24 18:23:13 +00:00
emitter.on('getMetadata', async () => {
const file = state.fileInfo;
2018-07-06 22:49:50 +00:00
2018-01-24 18:23:13 +00:00
const receiver = new FileReceiver(file);
try {
2018-01-24 18:23:13 +00:00
await receiver.getMetadata();
state.transfer = receiver;
} catch (e) {
2018-08-07 22:40:17 +00:00
if (e.message === '401' || e.message === '404') {
file.password = null;
2018-01-24 18:23:13 +00:00
if (!file.requiresPassword) {
return emitter.emit('pushState', '/404');
}
}
}
2018-07-05 19:40:49 +00:00
render();
});
emitter.on('download', async file => {
state.transfer.on('progress', updateProgress);
state.transfer.on('decrypting', render);
2018-07-31 18:09:18 +00:00
state.transfer.on('complete', render);
const links = openLinksInNewTab();
const size = file.size;
try {
const start = Date.now();
metrics.startedDownload({ size: file.size, ttl: file.ttl });
2018-07-31 18:29:26 +00:00
const dl = state.transfer.download({
stream: state.capabilities.streamDownload
});
render();
await dl;
const time = Date.now() - start;
const speed = size / (time / 1000);
state.storage.totalDownloads += 1;
metrics.completedDownload({ size, time, speed });
} catch (err) {
2018-01-24 18:23:13 +00:00
if (err.message === '0') {
// download cancelled
state.transfer.reset();
render();
} else {
// eslint-disable-next-line no-console
console.error(err);
state.transfer = null;
const location = err.message === '404' ? '/404' : '/error';
if (location === '/error') {
state.raven.captureException(err);
metrics.stoppedDownload({ size, err });
}
emitter.emit('pushState', location);
}
} finally {
openLinksInNewTab(links, false);
}
});
emitter.on('copy', ({ url, location }) => {
copyToClipboard(url);
metrics.copiedLink({ location });
});
2018-01-24 18:23:13 +00:00
setInterval(() => {
// poll for updates of the download counts
// TODO something for the share page: || state.route === '/share/:id'
if (state.route === '/') {
checkFiles();
}
}, 2 * 60 * 1000);
setInterval(() => {
// poll for rerendering the file list countdown timers
if (
state.route === '/' &&
state.storage.files.length > 0 &&
Date.now() - lastRender > 30000
) {
render();
}
}, 60000);
}