TabFS/extension/background.js

210 lines
4.7 KiB
JavaScript
Raw Normal View History

2018-11-11 14:32:44 +01:00
const ws = new WebSocket("ws://localhost:8888");
2018-11-12 01:18:21 +01:00
const unix = {
EPERM: 1,
ENOENT: 2,
ESRCH: 3,
EINTR: 4,
EIO: 5,
ENXIO: 6,
// Unix file types
S_IFMT: 0170000, // type of file mask
S_IFIFO: 010000, // named pipe (fifo)
S_IFCHR: 020000, // character special
S_IFDIR: 040000, // directory
S_IFBLK: 060000, // block special
S_IFREG: 0100000, // regular
S_IFLNK: 0120000, // symbolic link
S_IFSOCK: 0140000, // socket
}
function UnixError(error) {
this.name = "UnixError";
this.error = error;
}
UnixError.prototype = Error.prototype;
function getTab(id) {
return new Promise((resolve, reject) => chrome.tabs.get(id, resolve));
}
2018-11-15 08:52:26 +01:00
function queryTabs() {
return new Promise((resolve, reject) => chrome.tabs.query({}, resolve));
}
function sendDebuggerCommand(tab, method, commandParams) {
return new Promise(resolve => chrome.debugger.sendCommand({tabId: id}, method, commandParams, resolve));
}
const fhManager = (function() {
const handles = {};
let nextFh = 0;
return {
allocate(obj) { // -> fh
const fh = nextFh++;
handles[fh] = obj;
return fh;
},
ref(fh) {
if (!handles[fh]) throw new UnixError(unix.EIO);
return handles[fh];
},
free(fh) {
delete handles[fh];
}
};
})();
// tabs/by-id/ID/title
// tabs/by-id/ID/url
// tabs/by-id/ID/console
// tabs/by-id/ID/mem (?)
// tabs/by-id/ID/cpu (?)
// tabs/by-id/ID/screenshot.png
// tabs/by-id/ID/printed.pdf
// tabs/by-id/ID/control
// tabs/by-id/ID/sources/
function pathComponent(path, i) {
const components = path.split('/');
return components[i >= 0 ? i : components.length + i];
}
2018-11-15 08:52:26 +01:00
const router = {
"tabs": {
"by-id": {
async readdir() {
const tabs = await queryTabs();
return tabs.map(tab => String(tab.id));
},
"*": {
"url": {
async getattr() {
return {
st_mode: unix.S_IFREG | 0444,
st_nlink: 1,
st_size: 100 // FIXME
};
},
async open(path) {
return fhManager.allocate(await getTab(parseInt(pathComponent(path, -2))));
},
async read(path, fh, size, offset) {
const tab = fhManager.ref(fh);
return tab.url.substr(offset, size);
},
async release(path, fh) {
fhManager.free(fh);
}
},
/* "title": fileFromProperty('title'),
* "sources": folderFromResource(
* (tab, path) => new Promise(resolve => chrome.debugger.attach(
* { tabId: tab.id }, "1-3", resolve)),
* {
* readdir() {
* }
* }*/
2018-11-15 08:52:26 +01:00
}
}
}
};
2018-11-12 01:31:02 +01:00
2018-11-15 08:52:26 +01:00
function findRoute(path) {
let route = router;
for (let segment of path.split("/")) {
if (segment === "") continue;
route = route[segment] || route["*"];
2018-11-12 01:31:02 +01:00
if (!route) throw new UnixError(unix.ENOENT);
2018-11-15 08:52:26 +01:00
}
return route;
2018-11-15 08:52:26 +01:00
}
async function getattr(path) {
let route = findRoute(path);
if (route.getattr) {
return route.getattr(path);
2018-11-12 01:31:02 +01:00
} else {
2018-11-15 08:52:26 +01:00
return {
st_mode: unix.S_IFDIR | 0755,
st_nlink: 3
};
2018-11-12 01:31:02 +01:00
}
}
async function readdir(path) {
let route = findRoute(path);
if (route.readdir) return route.readdir(path);
return Object.keys(route);
2018-11-12 01:31:02 +01:00
}
async function open(path) {
let route = findRoute(path);
if (route.open) return route.open(path);
}
async function read(path, fh, size, offset) {
let route = findRoute(path);
if (route.read) return route.read(path, fh, size, offset);
}
async function release(path, fh) {
let route = findRoute(path);
if (route.read) return route.release(path, fh);
}
2018-11-15 08:52:26 +01:00
ws.onmessage = async function(event) {
2018-11-11 14:32:44 +01:00
const req = JSON.parse(event.data);
let response = { op: req.op, error: unix.EIO };
console.time(req.op + ':' + req.path);
try {
if (req.op === 'getattr') {
response = {
op: 'getattr',
st_mode: 0,
st_nlink: 0,
st_size: 0,
...(await getattr(req.path))
};
} else if (req.op === 'open') {
response = {
op: 'open',
fh: await open(req.path)
};
2018-11-12 01:31:02 +01:00
} else if (req.op === 'readdir') {
response = {
op: 'readdir',
entries: [".", "..", ...(await readdir(req.path))]
};
} else if (req.op === 'read') {
const buf = await read(req.path, req.fh, req.size, req.offset)
response = {
op: 'read',
buf,
size: buf.length
};
} else if (req.op === 'release') {
await release(req.path, req.fh);
response = {
op: 'release'
};
}
} catch (e) {
2018-11-11 20:44:36 +01:00
response = {
op: req.op,
error: e instanceof UnixError ? e.error : unix.EIO
}
2018-11-12 01:18:21 +01:00
}
console.timeEnd(req.op + ':' + req.path);
2018-11-11 14:32:44 +01:00
response.id = req.id;
2018-11-11 14:32:44 +01:00
ws.send(JSON.stringify(response));
};