mirror of
https://github.com/mickael-kerjean/filestash.git
synced 2025-10-30 17:46:41 +08:00
506 lines
23 KiB
JavaScript
506 lines
23 KiB
JavaScript
import { createElement, createFragment, createRender } from "../../lib/skeleton/index.js";
|
|
import { animate, slideYOut } from "../../lib/animate.js";
|
|
import rxjs, { effect, onClick } from "../../lib/rx.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 { mkdir, save } from "./model_virtual_layer.js";
|
|
import t from "../../locales/index.js";
|
|
|
|
const workers$ = new rxjs.BehaviorSubject({ tasks: [], size: null });
|
|
|
|
export default function(render) {
|
|
const $page = createFragment(`
|
|
<div is="component_filezone"></div>
|
|
<div is="component_upload_fab"></div>
|
|
`);
|
|
|
|
if (!document.querySelector(`[is="component_upload_queue"]`)) {
|
|
const $queue = createElement(`<div is="component_upload_queue"></div>`);
|
|
document.body.appendChild($queue);
|
|
componentUploadQueue(createRender($queue), { workers$ });
|
|
}
|
|
|
|
componentFilezone(createRender($page.children[0]), { workers$ });
|
|
componentUploadFAB(createRender($page.children[1]), { 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 = document.body.querySelector(selector);
|
|
|
|
$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", entry);
|
|
}
|
|
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">24</span>
|
|
<span class="grandTotal">24</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"><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>
|
|
`);
|
|
|
|
// feature1: close the queue
|
|
onClick(qs($page, `img[alt="close"]`)).pipe(
|
|
rxjs.tap(async (cancel) => {
|
|
const cleanup = await animate($page, { time: 200, keyframes: slideYOut(50) });
|
|
console.log(workers$.value);
|
|
qs($page, ".stats_content").innerHTML = "";
|
|
$page.classList.add("hidden");
|
|
cleanup();
|
|
}),
|
|
).subscribe();
|
|
|
|
// feature2: setup the task queue in the dom
|
|
workers$.subscribe(({ tasks }) => {
|
|
if (tasks.length === 0) return;
|
|
const $fragment = document.createDocumentFragment();
|
|
for (let i = 0; i<tasks.length; i++) {
|
|
const $task = $file.cloneNode(true);
|
|
$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() - last <= 500) return;
|
|
last = new Date();
|
|
const speed = workersSpeed.reduce((acc, el) => acc + el, 0);
|
|
const $speed = $page.firstElementChild.nextElementSibling.firstElementChild;
|
|
$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 cancel = () => exec.cancel();
|
|
switch (status) {
|
|
case "todo":
|
|
break;
|
|
case "doing":
|
|
updateDOMTaskProgress($task, formatPercent(0));
|
|
$task.firstElementChild.nextElementSibling.nextElementSibling.appendChild($iconStop);
|
|
$iconStop.onclick = () => {
|
|
cancel();
|
|
$task.firstElementChild.nextElementSibling.nextElementSibling.classList.add("hidden");
|
|
};
|
|
$close.addEventListener("click", cancel);
|
|
break;
|
|
case "done":
|
|
updateDOMGlobalSpeed(nworker, 0);
|
|
updateDOMTaskProgress($task, t("Done"));
|
|
updateDOMTaskSpeed($task, 0);
|
|
$task.removeAttribute("data-path");
|
|
$task.classList.remove("todo_color");
|
|
$task.firstElementChild.nextElementSibling.nextElementSibling.classList.add("hidden");
|
|
$close.removeEventListener("click", cancel);
|
|
break;
|
|
case "error":
|
|
const $retry = $iconRetry.cloneNode(true);
|
|
updateDOMGlobalTitle($page, t("Error"));
|
|
updateDOMGlobalSpeed(nworker, 0);
|
|
updateDOMTaskProgress($task, t("Error"));
|
|
updateDOMTaskSpeed($task, 0);
|
|
$task.removeAttribute("data-path");
|
|
$task.classList.remove("todo_color");
|
|
$task.firstElementChild.nextElementSibling.nextElementSibling.firstElementChild.remove();
|
|
$task.firstElementChild.nextElementSibling.nextElementSibling.appendChild($retry);
|
|
$retry.onclick = () => { console.log("CLICK RETRY"); }
|
|
$close.removeEventListener("click", cancel);
|
|
$task.classList.add("error_color");
|
|
break;
|
|
default:
|
|
assert.fail(`UNEXPECTED_STATUS status="${status}" path="${$task.getAttribute("path")}"`);
|
|
}
|
|
};
|
|
|
|
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((done) => setTimeout(done, 1000));
|
|
continue;
|
|
}
|
|
const $task = qs($page, `[data-path="${task.path}"]`);
|
|
const exec = task.exec({
|
|
error: (err) => updateDOMWithStatus($task, { status: "error", nworker }),
|
|
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 });
|
|
}
|
|
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(this, nworker)).then(() => reservations[nworker] = false);
|
|
}
|
|
});
|
|
}
|
|
|
|
class IExecutor {
|
|
contructor() {}
|
|
cancel() { throw new Error("NOT_IMPLEMENTED"); }
|
|
run() { throw new Error("NOT_IMPLEMENTED"); }
|
|
}
|
|
|
|
function workerImplFile({ error, progress, speed }) {
|
|
return new class Worker extends IExecutor {
|
|
constructor() {
|
|
super();
|
|
this.xhr = null;
|
|
this.prevProgress = [];
|
|
}
|
|
|
|
cancel() {
|
|
this.xhr.abort();
|
|
}
|
|
|
|
async run({ entry, path, virtual }) {
|
|
return new Promise((done, err) => {
|
|
this.xhr = new XMLHttpRequest();
|
|
this.xhr.open("POST", "api/files/cat?path=" + encodeURIComponent(path));
|
|
this.xhr.withCredentials = true;
|
|
this.xhr.setRequestHeader("X-Requested-With", "XmlHttpRequest");
|
|
this.xhr.upload.onabort = () => {
|
|
err(new AjaxError("aborted", null, "ABORTED"));
|
|
error(new AjaxError("aborted", null, "ABORTED"));
|
|
};
|
|
this.xhr.upload.onprogress = (e) => {
|
|
if (!e.lengthComputable) return;
|
|
const percent = Math.floor(100 * e.loaded / e.total);
|
|
progress(percent);
|
|
if (this.prevProgress.length === 0) {
|
|
this.prevProgress.push(e);
|
|
return;
|
|
}
|
|
this.prevProgress.push(e);
|
|
|
|
const calculateTime = (p1, pm1) => (p1.timeStamp - pm1.timeStamp)/1000;
|
|
const calculateBytes = (p1, pm1) => p1.loaded - pm1.loaded;
|
|
const lastIdx = this.prevProgress.length - 1;
|
|
let avgSpeed = 0;
|
|
for (let i=1; i<this.prevProgress.length; i++) {
|
|
const p1 = this.prevProgress[i];
|
|
const pm1 = this.prevProgress[i-1];
|
|
avgSpeed += calculateBytes(p1, pm1) / calculateTime(p1, pm1)
|
|
}
|
|
avgSpeed = avgSpeed / (this.prevProgress.length - 1);
|
|
speed(avgSpeed);
|
|
if (e.timeStamp - this.prevProgress[0].timeStamp > 5000) {
|
|
this.prevProgress.shift();
|
|
}
|
|
};
|
|
this.xhr.onload = () => {
|
|
progress(100);
|
|
if (this.xhr.status !== 200) {
|
|
virtual.afterError();
|
|
err(new Error(this.xhr.statusText));
|
|
return;
|
|
}
|
|
virtual.afterSuccess();
|
|
done();
|
|
};
|
|
this.xhr.onerror = function(e) {
|
|
err(new AjaxError("failed", e, "FAILED"));
|
|
vitual.afterError();
|
|
};
|
|
entry.file(
|
|
(file) => this.xhr.send(file),
|
|
(err) => this.xhr.onerror(err),
|
|
);
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
function workerImplDirectory({ error, progress }) {
|
|
return new class Worker extends IExecutor {
|
|
constructor() {
|
|
super();
|
|
this.xhr = null;
|
|
}
|
|
|
|
cancel() {
|
|
this.xhr.abort();
|
|
}
|
|
|
|
run({ virtual, path }) {
|
|
return new Promise((done, err) => {
|
|
this.xhr = new XMLHttpRequest();
|
|
this.xhr.open("POST", "api/files/mkdir?path=" + encodeURIComponent(path));
|
|
this.xhr.withCredentials = true;
|
|
this.xhr.setRequestHeader("X-Requested-With", "XmlHttpRequest");
|
|
this.xhr.onerror = function(e) {
|
|
err(new AjaxError("failed", e, "FAILED"));
|
|
};
|
|
|
|
let percent = 0;
|
|
const id = setInterval(() => {
|
|
percent += 10;
|
|
if (percent >= 100) {
|
|
clearInterval(id);
|
|
return;
|
|
}
|
|
progress(percent);
|
|
}, 100);
|
|
this.xhr.upload.onabort = () => {
|
|
err(new AjaxError("aborted", null, "ABORTED"));
|
|
error(new AjaxError("aborted", null, "ABORTED"));
|
|
clearInterval(id);
|
|
virtual.afterError();
|
|
};
|
|
this.xhr.onload = () => {
|
|
clearInterval(id);
|
|
progress(100);
|
|
if (this.xhr.status !== 200) {
|
|
virtual.afterError();
|
|
err(new Error(this.xhr.statusText));
|
|
return;
|
|
}
|
|
virtual.afterSuccess();
|
|
done();
|
|
};
|
|
this.xhr.send(null);
|
|
});
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
async function processFiles(filelist) { // TODO
|
|
const files = [];
|
|
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((done, err) => {
|
|
const reader = new window.FileReader();
|
|
const tid = setTimeout(() => reader.abort(), 1000);
|
|
reader.onload = () => done("file");
|
|
reader.onabort = () => done("file");
|
|
reader.onerror = () => { done("directory"); clearTimeout(tid); }
|
|
reader.readAsArrayBuffer(file);
|
|
});
|
|
}
|
|
for (const currentFile of filelist) {
|
|
const type = await detectFiletype(currentFile);
|
|
const file = { type, date: currentFile.lastModified, name: currentFile.name, path: currentFile.name };
|
|
if (type === "directory") {
|
|
file.path += "/";
|
|
} else if (type === "file") {
|
|
file.entry = currentFile;
|
|
} else {
|
|
assert.fail(`NOT_SUPPORTED type="${type}"`, type);
|
|
}
|
|
file.exec = workerImplFile.bind(file);
|
|
files.push(file);
|
|
}
|
|
return files;
|
|
}
|
|
|
|
async function processItems(itemList) {
|
|
const bfs = async (queue) => {
|
|
const tasks = [];
|
|
let size = 0;
|
|
let path = "";
|
|
const basepath = currentPath();
|
|
while (queue.length > 0) {
|
|
const entry = queue.shift();
|
|
const path = basepath + entry.fullPath.substring(1);
|
|
let task = null;
|
|
if (entry === null) continue;
|
|
else if (entry.isFile) {
|
|
const entrySize = await new Promise((done) => entry.getMetadata(({ size }) => done(size)));
|
|
task = {
|
|
type: "file", entry,
|
|
path,
|
|
exec: workerImplFile,
|
|
virtual: save(path, entrySize),
|
|
done: false,
|
|
};
|
|
size += entrySize;
|
|
} else if (entry.isDirectory) {
|
|
task = {
|
|
type: "directory",
|
|
path: path + "/",
|
|
exec: workerImplDirectory,
|
|
virtual: mkdir(path),
|
|
done: false,
|
|
};
|
|
size += 5000; // that's to calculate the remaining time for an upload, aka made up size is ok
|
|
queue = queue.concat(await new Promise((done) => {
|
|
entry.createReader().readEntries(done);
|
|
}));
|
|
} else {
|
|
assert.fail("NOT_IMPLEMENTED - unknown entry type in ctrl_upload.js", entry);
|
|
}
|
|
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";
|
|
}
|