mirror of
https://github.com/mickael-kerjean/filestash.git
synced 2025-11-01 10:56:31 +08:00
660 lines
28 KiB
JavaScript
660 lines
28 KiB
JavaScript
import { createElement, createFragment, createRender } from "../../lib/skeleton/index.js";
|
|
import { toHref } from "../../lib/skeleton/router.js";
|
|
import rxjs, { effect, onClick } from "../../lib/rx.js";
|
|
import { forwardURLParams } from "../../lib/path.js";
|
|
import { animate, slideYOut } from "../../lib/animate.js";
|
|
import { loadCSS } from "../../helpers/loader.js";
|
|
import { qs } from "../../lib/dom.js";
|
|
import { AjaxError } from "../../lib/error.js";
|
|
import assert from "../../lib/assert.js";
|
|
import { currentPath, isNativeFileUpload } from "./helper.js";
|
|
import { getPermission, calculatePermission } from "./model_acl.js";
|
|
import { mkdir, save } from "./model_virtual_layer.js";
|
|
import t from "../../locales/index.js";
|
|
|
|
const workers$ = new rxjs.BehaviorSubject({ tasks: [], size: null });
|
|
const ABORT_ERROR = new AjaxError("aborted", null, "ABORTED");
|
|
|
|
export default async function(render) {
|
|
if (!document.querySelector(`[is="component_upload_queue"]`)) {
|
|
const $queue = createElement(`<div is="component_upload_queue"></div>`);
|
|
document.body.appendChild($queue);
|
|
componentUploadQueue(createRender($queue), { workers$ });
|
|
}
|
|
|
|
effect(getPermission().pipe(
|
|
rxjs.filter(() => calculatePermission(currentPath(), "new-file")),
|
|
rxjs.tap(() => {
|
|
const $page = createFragment(`
|
|
<div is="component_filezone"></div>
|
|
<div is="component_upload_fab"></div>
|
|
`);
|
|
componentFilezone(createRender(assert.type($page.children[0], HTMLElement)), { workers$ });
|
|
componentUploadFAB(createRender(assert.type($page.children[1], HTMLElement)), { workers$ });
|
|
render($page);
|
|
}),
|
|
));
|
|
}
|
|
|
|
export function init() {
|
|
return loadCSS(import.meta.url, "./ctrl_upload.css");
|
|
}
|
|
|
|
function componentUploadFAB(render, { workers$ }) {
|
|
const $page = createElement(`
|
|
<div class="component_mobilefileupload no-select">
|
|
<form>
|
|
<input type="file" name="file" id="mobilefileupload" multiple />
|
|
<label for="mobilefileupload" title="${t("Upload")}">
|
|
<img
|
|
class="component_icon"
|
|
draggable="false"
|
|
alt="upload"
|
|
src=""
|
|
/>
|
|
</label>
|
|
</form>
|
|
</div>
|
|
`);
|
|
effect(rxjs.fromEvent(qs($page, `input[type="file"]`), "change").pipe(
|
|
rxjs.tap(async(e) => workers$.next(await processFiles(e.target.files))),
|
|
));
|
|
render($page);
|
|
}
|
|
|
|
function componentFilezone(render, { workers$ }) {
|
|
const selector = `[data-bind="filemanager-children"]`;
|
|
const $target = assert.type(qs(document.body, selector), HTMLElement);
|
|
|
|
$target.ondragenter = (e) => {
|
|
if (!isNativeFileUpload(e)) return;
|
|
$target.classList.add("dropzone");
|
|
};
|
|
$target.ondrop = async(e) => {
|
|
if (!isNativeFileUpload(e)) return;
|
|
$target.classList.remove("dropzone");
|
|
e.preventDefault();
|
|
const loadID = setTimeout(() => render(createElement("<div>LOADING</div>")), 2000);
|
|
if (e.dataTransfer.items instanceof window.DataTransferItemList) {
|
|
workers$.next(await processItems(e.dataTransfer.items));
|
|
} else if (e.dataTransfer.files instanceof window.FileList) {
|
|
workers$.next(await processFiles(e.dataTransfer.files));
|
|
} else {
|
|
assert.fail("NOT_IMPLEMENTED - unknown entry type in ctrl_upload.js");
|
|
}
|
|
clearTimeout(loadID);
|
|
render(createFragment(""));
|
|
};
|
|
$target.ondragleave = (e) => {
|
|
if (!isNativeFileUpload(e)) return;
|
|
if (!(e.relatedTarget === null || // eg: drag outside the window
|
|
!e.relatedTarget.closest(selector) // eg: drag on the breadcrumb, ...
|
|
)) return;
|
|
$target.classList.remove("dropzone");
|
|
};
|
|
$target.ondragover = (e) => e.preventDefault();
|
|
}
|
|
|
|
const MAX_WORKERS = 4;
|
|
|
|
function componentUploadQueue(render, { workers$ }) {
|
|
const $page = createElement(`
|
|
<div class="component_upload hidden">
|
|
<h2 class="no-select">${t("Current Upload")}
|
|
<div class="count_block">
|
|
<span class="completed">0</span>
|
|
<span class="grandTotal">0</span>
|
|
</div>
|
|
<img class="component_icon" draggable="false" src="" alt="close">
|
|
</h2>
|
|
<h3 class="no-select"> <span></span></h3>
|
|
<div class="stats_content"></div>
|
|
</div>
|
|
`);
|
|
render($page);
|
|
const $content = qs($page, ".stats_content");
|
|
const $file = createElement(`
|
|
<div class="file_row todo_color">
|
|
<div class="file_path ellipsis"><span class="path"></span><span class="speed no-select"></span></div>
|
|
<div class="file_state no-select"></div>
|
|
<div class="file_control no-select"></div>
|
|
</div>
|
|
`);
|
|
const updateTotal = {
|
|
reset: () => {
|
|
qs($page, ".grandTotal").innerText = 0;
|
|
qs($page, ".completed").innerText = 0;
|
|
},
|
|
addToTotal: (n) => {
|
|
const $total = qs($page, ".grandTotal");
|
|
$total.innerText = parseInt($total.innerText) + n;
|
|
},
|
|
incrementCompleted: () => {
|
|
const $completed = qs($page, ".completed");
|
|
$completed.innerText = parseInt($completed.innerText) + 1;
|
|
},
|
|
};
|
|
|
|
// feature1: close the queue
|
|
onClick(qs($page, `img[alt="close"]`)).pipe(rxjs.tap(async() => {
|
|
const cleanup = await animate($page, { time: 200, keyframes: slideYOut(50) });
|
|
$content.innerHTML = "";
|
|
$page.classList.add("hidden");
|
|
updateTotal.reset();
|
|
cleanup();
|
|
})).subscribe();
|
|
|
|
// feature2: setup the task queue in the dom
|
|
workers$.subscribe(({ tasks }) => {
|
|
if (tasks.length === 0) return;
|
|
updateTotal.addToTotal(tasks.length);
|
|
const $fragment = document.createDocumentFragment();
|
|
for (let i = 0; i<tasks.length; i++) {
|
|
const $task = assert.type($file.cloneNode(true), HTMLElement);
|
|
$fragment.appendChild($task);
|
|
$task.setAttribute("data-path", tasks[i]["path"]);
|
|
$task.firstElementChild.firstElementChild.textContent = tasks[i]["path"]; // qs($todo, ".file_path span.path")
|
|
$task.firstElementChild.firstElementChild.setAttribute("title", tasks[i]["path"]);
|
|
$task.firstElementChild.nextElementSibling.classList.add("file_state_todo"); // qs($todo, ".file_state")
|
|
$task.firstElementChild.nextElementSibling.textContent = t("Waiting");
|
|
}
|
|
$page.classList.remove("hidden");
|
|
$content.appendChild($fragment);
|
|
});
|
|
|
|
// feature3: process tasks
|
|
const ICON = {
|
|
STOP: "",
|
|
RETRY: "",
|
|
};
|
|
const $iconStop = createElement(`<img class="component_icon" draggable="false" src="${ICON.STOP}" alt="stop" title="${t("Aborted")}">`);
|
|
const $iconRetry = createElement(`<img class="component_icon" draggable="false" src="${ICON.RETRY}" alt="retry">`);
|
|
const $close = qs($page, `img[alt="close"]`);
|
|
const updateDOMTaskProgress = ($task, text) => $task.firstElementChild.nextElementSibling.textContent = text;
|
|
const updateDOMTaskSpeed = ($task, text) => $task.firstElementChild.firstElementChild.nextElementSibling.textContent = formatSpeed(text);
|
|
const updateDOMGlobalSpeed = (function(workersSpeed) {
|
|
let last = 0;
|
|
return (nworker, currentWorkerSpeed) => {
|
|
workersSpeed[nworker] = currentWorkerSpeed;
|
|
if (new Date().getTime() - last <= 500) return;
|
|
last = new Date().getTime();
|
|
const speed = workersSpeed.reduce((acc, el) => acc + el, 0);
|
|
const $speed = assert.type($page.firstElementChild?.nextElementSibling?.firstElementChild, HTMLElement);
|
|
$speed.textContent = formatSpeed(speed);
|
|
};
|
|
}(new Array(MAX_WORKERS).fill(0)));
|
|
const updateDOMGlobalTitle = ($page, text) => $page.firstElementChild.nextElementSibling.childNodes[0].textContent = text;
|
|
const updateDOMWithStatus = ($task, { status, exec, nworker }) => {
|
|
const executeMutation = (status) => {
|
|
switch (status) {
|
|
case "todo":
|
|
updateDOMGlobalTitle($page, t("Running") + "...");
|
|
break;
|
|
case "doing":
|
|
const $stop = assert.type($iconStop.cloneNode(true), HTMLElement);
|
|
updateDOMTaskProgress($task, formatPercent(0));
|
|
$task.classList.remove("error_color");
|
|
$task.classList.add("todo_color");
|
|
$task.firstElementChild.nextElementSibling.nextElementSibling.replaceChildren($stop);
|
|
$stop.onclick = () => {
|
|
exec.cancel();
|
|
$task.firstElementChild.nextElementSibling.nextElementSibling.classList.add("hidden");
|
|
};
|
|
$close.addEventListener("click", exec.cancel);
|
|
break;
|
|
case "done":
|
|
updateDOMGlobalTitle($page, t("Done"));
|
|
updateDOMTaskProgress($task, t("Done"));
|
|
updateDOMGlobalSpeed(nworker, 0);
|
|
updateDOMTaskSpeed($task, 0);
|
|
$task.removeAttribute("data-path");
|
|
$task.classList.remove("todo_color");
|
|
$task.firstElementChild.nextElementSibling.nextElementSibling.classList.add("hidden");
|
|
$close.removeEventListener("click", exec.cancel);
|
|
break;
|
|
case "error":
|
|
const $retry = assert.type($iconRetry.cloneNode(true), HTMLElement);
|
|
updateDOMGlobalTitle($page, t("Error"));
|
|
updateDOMTaskProgress($task, t("Error"));
|
|
updateDOMGlobalSpeed(nworker, 0);
|
|
updateDOMTaskSpeed($task, 0);
|
|
|
|
$task.removeAttribute("data-path");
|
|
$task.classList.remove("todo_color");
|
|
$task.classList.add("error_color");
|
|
$task.firstElementChild.nextElementSibling.nextElementSibling.firstElementChild.remove();
|
|
$task.firstElementChild.nextElementSibling.nextElementSibling.appendChild($retry);
|
|
$retry.onclick = async() => {
|
|
executeMutation("todo");
|
|
executeMutation("doing");
|
|
try {
|
|
await exec.retry();
|
|
executeMutation("done");
|
|
} catch (err) {
|
|
executeMutation("error");
|
|
}
|
|
};
|
|
$close.removeEventListener("click", exec.cancel);
|
|
break;
|
|
default:
|
|
assert.fail(`UNEXPECTED_STATUS status="${status}" path="${$task.getAttribute("path")}"`);
|
|
}
|
|
};
|
|
executeMutation(status);
|
|
};
|
|
|
|
let tasks = [];
|
|
const reservations = new Array(MAX_WORKERS).fill(false);
|
|
const processWorkerQueue = async(nworker) => {
|
|
while (tasks.length > 0) {
|
|
updateDOMGlobalTitle($page, t("Running")+"...");
|
|
const task = nextTask(tasks);
|
|
if (!task) {
|
|
await new Promise((resolve) => setTimeout(resolve, 1000));
|
|
continue;
|
|
}
|
|
const $task = qs($page, `[data-path="${task.path}"]`);
|
|
const exec = task.exec({
|
|
progress: (progress) => updateDOMTaskProgress($task, formatPercent(progress)),
|
|
speed: (speed) => {
|
|
updateDOMTaskSpeed($task, speed);
|
|
updateDOMGlobalSpeed(nworker, speed);
|
|
},
|
|
});
|
|
updateDOMWithStatus($task, { exec, status: "doing", nworker });
|
|
try {
|
|
await exec.run(task);
|
|
updateDOMWithStatus($task, { exec, status: "done", nworker });
|
|
} catch (err) {
|
|
updateDOMWithStatus($task, { exec, status: "error", nworker });
|
|
}
|
|
updateTotal.incrementCompleted();
|
|
task.done = true;
|
|
|
|
if (tasks.length === 0 && // no remaining tasks
|
|
reservations.filter((t) => t === true).length === 1 // only for the last remaining job
|
|
) updateDOMGlobalTitle($page, t("Done"));
|
|
}
|
|
};
|
|
const nextTask = (tasks) => {
|
|
for (let i=0; i<tasks.length; i++) {
|
|
const possibleTask = tasks[i];
|
|
if (!possibleTask.ready()) continue;
|
|
tasks.splice(i, 1);
|
|
return possibleTask;
|
|
}
|
|
return null;
|
|
};
|
|
const noFailureAllowed = (fn) => fn().catch(() => noFailureAllowed(fn));
|
|
workers$.subscribe(async({ tasks: newTasks }) => {
|
|
tasks = tasks.concat(newTasks); // add new tasks to the pool
|
|
while (true) {
|
|
const nworker = reservations.indexOf(false);
|
|
if (nworker === -1) break; // the pool of workers is already to its max
|
|
reservations[nworker] = true;
|
|
noFailureAllowed(processWorkerQueue.bind(null, nworker)).then(() => reservations[nworker] = false);
|
|
}
|
|
});
|
|
}
|
|
|
|
class IExecutor {
|
|
contructor() {}
|
|
cancel() { throw new Error("NOT_IMPLEMENTED"); }
|
|
retry() { throw new Error("NOT_IMPLEMENTED"); }
|
|
run() { throw new Error("NOT_IMPLEMENTED"); }
|
|
}
|
|
|
|
function workerImplFile({ progress, speed }) {
|
|
return new class Worker extends IExecutor {
|
|
constructor() {
|
|
super();
|
|
this.xhr = null;
|
|
}
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
cancel() {
|
|
assert.type(this.xhr, XMLHttpRequest).abort();
|
|
}
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
async run({ file, path, virtual }) {
|
|
const _file = await file();
|
|
const executeJob = (firstRun) => this.prepareJob({ file: _file, path, virtual, firstRun });
|
|
this.retry = () => executeJob(false);
|
|
return executeJob(true);
|
|
}
|
|
|
|
async prepareJob({ file, path, virtual, firstRun }) {
|
|
if (firstRun === false) virtual.before();
|
|
const chunkSize = (window.CONFIG["upload_chunk_size"] || 0) *1024*1024;
|
|
const numberOfChunks = Math.ceil(file.size / chunkSize);
|
|
|
|
// Case1: basic upload
|
|
if (chunkSize === 0 || numberOfChunks === 0 || numberOfChunks === 1) {
|
|
try {
|
|
await executeHttp.call(this, toHref(`/api/files/cat?path=${encodeURIComponent(path)}`), {
|
|
method: "POST",
|
|
headers: {},
|
|
body: file,
|
|
progress,
|
|
speed,
|
|
});
|
|
virtual.afterSuccess();
|
|
} catch (err) {
|
|
virtual.afterError();
|
|
if (err === ABORT_ERROR) return;
|
|
throw err;
|
|
}
|
|
return;
|
|
}
|
|
|
|
// Case2: chunked upload => TUS: https://www.ietf.org/archive/id/draft-tus-httpbis-resumable-uploads-protocol-00.html
|
|
try {
|
|
let resp = await executeHttp.call(this, toHref(`/api/files/cat?path=${encodeURIComponent(path)}&proto=tus`), {
|
|
method: "POST",
|
|
headers: { "Upload-Length": file.size },
|
|
body: null,
|
|
progress: (n) => progress(n),
|
|
speed,
|
|
});
|
|
const url = resp.headers.location;
|
|
if (!url.startsWith(toHref("/api/files/cat?"))) {
|
|
throw new Error("Internal Error");
|
|
}
|
|
for (let i=0; i<numberOfChunks; i++) {
|
|
const offset = chunkSize * i;
|
|
resp = await executeHttp.call(this, url, {
|
|
method: "PATCH",
|
|
headers: { "Upload-Offset": offset },
|
|
body: file.slice(offset, offset + chunkSize),
|
|
progress: (p) => {
|
|
const chunksAlreadyDownloaded = i * chunkSize;
|
|
const currentChunkDownloaded = p / 100 * (
|
|
i !== numberOfChunks - 1 ? chunkSize : (file.size % chunkSize) || chunkSize
|
|
);
|
|
progress(Math.floor(100 * (chunksAlreadyDownloaded + currentChunkDownloaded) / file.size));
|
|
},
|
|
speed,
|
|
});
|
|
}
|
|
virtual.afterSuccess();
|
|
} catch (err) {
|
|
virtual.afterError();
|
|
if (err === ABORT_ERROR) return;
|
|
throw err;
|
|
}
|
|
}
|
|
}();
|
|
}
|
|
|
|
function workerImplDirectory({ progress }) {
|
|
return new class Worker extends IExecutor {
|
|
constructor() {
|
|
super();
|
|
this.xhr = null;
|
|
}
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
cancel() {
|
|
assert.type(this.xhr, XMLHttpRequest).abort();
|
|
}
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
async run({ virtual, path }) {
|
|
const executeJob = (firstRun) => this.prepareJob({ virtual, path, firstRun });
|
|
this.retry = () => executeJob(false);
|
|
return executeJob(true);
|
|
}
|
|
|
|
async prepareJob({ virtual, path, firstRun }) {
|
|
if (firstRun === false) virtual.before();
|
|
let percent = 0;
|
|
const id = setInterval(() => {
|
|
percent += 10;
|
|
if (percent >= 100) {
|
|
clearInterval(id);
|
|
return;
|
|
}
|
|
progress(percent);
|
|
}, 100);
|
|
try {
|
|
await executeHttp.call(this, toHref(`/api/files/mkdir?path=${encodeURIComponent(path)}`), {
|
|
method: "POST",
|
|
headers: {},
|
|
body: null,
|
|
progress,
|
|
speed: () => {},
|
|
});
|
|
clearInterval(id);
|
|
progress(100);
|
|
virtual.afterSuccess();
|
|
} catch (err) {
|
|
clearInterval(id);
|
|
virtual.afterError();
|
|
if (err === ABORT_ERROR) return;
|
|
throw err;
|
|
}
|
|
}
|
|
}();
|
|
}
|
|
|
|
function executeHttp(url, { method, headers, body, progress, speed }) {
|
|
const xhr = new XMLHttpRequest();
|
|
const prevProgress = [];
|
|
this.xhr = xhr;
|
|
return new Promise((resolve, reject) => {
|
|
xhr.open(method, forwardURLParams(url, ["share"]));
|
|
xhr.setRequestHeader("X-Requested-With", "XmlHttpRequest");
|
|
xhr.withCredentials = true;
|
|
for (const key in headers) {
|
|
xhr.setRequestHeader(key, headers[key]);
|
|
}
|
|
xhr.upload.onprogress = (e) => {
|
|
if (!e.lengthComputable) return;
|
|
const percent = Math.floor(100 * e.loaded / e.total);
|
|
progress(percent);
|
|
if (prevProgress.length === 0) {
|
|
prevProgress.push(e);
|
|
return;
|
|
}
|
|
prevProgress.push(e);
|
|
|
|
const calculateTime = (p1, pm1) => (p1.timeStamp - pm1.timeStamp)/1000;
|
|
const calculateBytes = (p1, pm1) => p1.loaded - pm1.loaded;
|
|
let avgSpeed = 0;
|
|
for (let i=1; i<prevProgress.length; i++) {
|
|
const p1 = prevProgress[i];
|
|
const pm1 = prevProgress[i-1];
|
|
avgSpeed += calculateBytes(p1, pm1) / calculateTime(p1, pm1);
|
|
}
|
|
avgSpeed = avgSpeed / (prevProgress.length - 1);
|
|
speed(avgSpeed);
|
|
if (e.timeStamp - prevProgress[0].timeStamp > 5000) {
|
|
prevProgress.shift();
|
|
}
|
|
};
|
|
xhr.upload.onabort = () => reject(ABORT_ERROR);
|
|
xhr.onerror = (e) => reject(new AjaxError("failed", e, "FAILED"));
|
|
xhr.onload = () => {
|
|
if ([200, 201, 204].indexOf(xhr.status) === -1) {
|
|
reject(new Error(xhr.statusText));
|
|
return;
|
|
}
|
|
progress(100);
|
|
resolve({
|
|
status: xhr.status,
|
|
headers: xhr.getAllResponseHeaders()
|
|
.split("\r\n")
|
|
.reduce((acc, el) => {
|
|
const tmp = el.split(": ");
|
|
if (typeof tmp[0] === "string" && typeof tmp[1] === "string") {
|
|
acc[tmp[0]] = tmp[1];
|
|
}
|
|
return acc;
|
|
}, {})
|
|
});
|
|
};
|
|
xhr.send(body);
|
|
});
|
|
}
|
|
|
|
async function processFiles(filelist) {
|
|
const tasks = [];
|
|
// let size = 0; // TODO
|
|
const detectFiletype = (file) => {
|
|
// the 4096 is an heuristic observed and taken from:
|
|
// https://stackoverflow.com/questions/25016442
|
|
// however the proposed answer is just wrong as it doesn't consider folder with
|
|
// name such as: test.png and as Stackoverflow favor consanguinity with their
|
|
// point system, I couldn't rectify the proposed answer. The following code is
|
|
// actually working as expected
|
|
if (file.size % 4096 !== 0) {
|
|
return Promise.resolve("file");
|
|
}
|
|
return new Promise((resolve) => {
|
|
const reader = new window.FileReader();
|
|
const tid = setTimeout(() => reader.abort(), 1000);
|
|
reader.onload = () => resolve("file");
|
|
reader.onabort = () => resolve("file");
|
|
reader.onerror = () => { resolve("directory"); clearTimeout(tid); };
|
|
reader.readAsArrayBuffer(file);
|
|
});
|
|
};
|
|
for (const currentFile of filelist) {
|
|
const type = await detectFiletype(currentFile);
|
|
let path = currentPath() + currentFile.name;
|
|
let task = null;
|
|
switch (type) {
|
|
case "file":
|
|
// size += currentFile.size;
|
|
task = {
|
|
type: "file",
|
|
file: () => new Promise((resolve) => resolve(currentFile)),
|
|
path,
|
|
date: currentFile.lastModified,
|
|
exec: workerImplFile,
|
|
virtual: save(path, currentFile.size),
|
|
done: false,
|
|
ready: () => true,
|
|
entry: currentFile,
|
|
|
|
};
|
|
break;
|
|
case "directory":
|
|
path += "/";
|
|
task = {
|
|
type: "directory",
|
|
path,
|
|
date: currentFile.lastModified,
|
|
exec: workerImplDirectory,
|
|
virtual: mkdir(path),
|
|
done: false,
|
|
ready: () => true,
|
|
};
|
|
break;
|
|
default:
|
|
assert.fail(`NOT_SUPPORTED type="${type}"`);
|
|
}
|
|
task = assert.truthy(task);
|
|
task.virtual.before();
|
|
tasks.push(task);
|
|
}
|
|
return { tasks, size: 0 };
|
|
}
|
|
|
|
async function processItems(itemList) {
|
|
const bfs = async(queue) => {
|
|
const tasks = [];
|
|
let size = 0;
|
|
const basepath = currentPath();
|
|
while (queue.length > 0) {
|
|
const entry = queue.shift();
|
|
const path = basepath + entry.fullPath.substring(1);
|
|
let task = {};
|
|
if (entry === null) continue;
|
|
else if (entry.isFile) {
|
|
const entrySize = await new Promise((resolve) => {
|
|
if (typeof entry.getMetadata === "function") {
|
|
entry.getMetadata(({ size }) => resolve(size));
|
|
}
|
|
else resolve(null); // eg: firefox
|
|
});
|
|
task = {
|
|
type: "file",
|
|
file: () => new Promise((resolve, reject) => entry.file(
|
|
(file) => resolve(file),
|
|
(error) => reject(error),
|
|
)),
|
|
path,
|
|
exec: workerImplFile,
|
|
virtual: save(path, entrySize),
|
|
done: false,
|
|
ready: () => false,
|
|
};
|
|
size += entrySize;
|
|
} else if (entry.isDirectory) {
|
|
task = {
|
|
type: "directory",
|
|
path: path + "/",
|
|
exec: workerImplDirectory,
|
|
virtual: mkdir(path),
|
|
done: false,
|
|
ready: () => false,
|
|
};
|
|
queue = queue.concat(await new Promise((resolve) => {
|
|
entry.createReader().readEntries(resolve);
|
|
}));
|
|
} else {
|
|
assert.fail("NOT_IMPLEMENTED - unknown entry type in ctrl_upload.js");
|
|
}
|
|
task.ready = () => {
|
|
const isInDirectory = (filepath, folder) => folder.indexOf(filepath) === 0;
|
|
for (let i=0; i<tasks.length; i++) {
|
|
// filter out tasks that are NOT dependencies of the current task
|
|
if (tasks[i].path === task.path) break;
|
|
else if (tasks[i].type === "file") continue;
|
|
else if (isInDirectory(tasks[i].path, task.path) === false) continue;
|
|
|
|
// block execution unless dependent task has completed
|
|
if (tasks[i].done === false) return false;
|
|
}
|
|
return true;
|
|
};
|
|
task.virtual.before();
|
|
tasks.push(task);
|
|
}
|
|
return { tasks, size };
|
|
};
|
|
const entries = [];
|
|
for (const item of itemList) entries.push(item.webkitGetAsEntry());
|
|
return await bfs(entries);
|
|
}
|
|
|
|
function formatPercent(number) {
|
|
return `${number}%`;
|
|
}
|
|
|
|
function formatSpeed(bytes, si = true) {
|
|
const thresh = si ? 1000 : 1024;
|
|
if (Math.abs(bytes) < thresh) {
|
|
return bytes.toFixed(1) + "B/s";
|
|
}
|
|
const units = si
|
|
? ["kB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB"]
|
|
: ["KiB", "MiB", "GiB", "TiB", "PiB", "EiB", "ZiB", "YiB"];
|
|
let u = -1;
|
|
do {
|
|
bytes /= thresh;
|
|
++u;
|
|
} while (Math.abs(bytes) >= thresh && u < units.length - 1);
|
|
return bytes.toFixed(1) + units[u] + "/s";
|
|
}
|