TabFS/extension/background.js

275 lines
6.7 KiB
JavaScript
Raw Normal View History

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/text.txt
// 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];
}
const debugged = {};
2018-11-15 08:52:26 +01:00
const router = {
"tabs": {
/* "last-focused": {
* // FIXME: symlink to tab by id.
* async readlink() {
* return "../windows/last-focused/selected-tab"
* }
* },
*/
2018-11-15 08:52:26 +01:00
"by-id": {
async readdir() {
const tabs = await queryTabs();
return tabs.map(tab => String(tab.id));
},
"*": {
"url": {
async read(path, fh, size, offset) {
const tab = await getTab(parseInt(pathComponent(path, -2)));
return (tab.url + "\n").substr(offset, size);
}
},
"title": {
async read(path, fh, size, offset) {
const tab = await getTab(parseInt(pathComponent(path, -2)));
return (tab.title + "\n").substr(offset, size);
}
},
"tree": {
async opendir(path) {
const tabId = parseInt(pathComponent(path, -2));
if (!debugged[tabId]) {
await new Promise(resolve => chrome.debugger.attach({tabId}, "1.2", resolve));
debugged[tabId] = 0;
}
debugged[tabId] += 1;
return 0;
},
async readdir(path) {
const tabId = parseInt(pathComponent(path, -2));
if (!debugged[tabId]) throw new UnixError(unix.EIO);
const result = await new Promise(resolve => chrome.debugger.sendCommand({tabId}, "Page.getResourceTree", {}, resolve));
const frameTree = result.frameTree;
return frameTree.resources.map(r => String(r.contentSize));
},
async releasedir(path) {
return 0;
}
}
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);
} else if (route.read) {
// default file attrs
return {
st_mode: unix.S_IFREG | 0444,
st_nlink: 1,
st_size: 100 // FIXME
};
2018-11-12 01:31:02 +01:00
} else {
// default dir attrs
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 open(path) {
let route = findRoute(path);
if (route.open) return route.open(path);
else return 0; // empty fh
}
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.release) return route.release(path, fh);
}
async function opendir(path) {
let route = findRoute(path);
if (route.opendir) return route.opendir(path);
else return 0; // empty fh
}
async function readdir(path) {
let route = findRoute(path);
if (route.readdir) return route.readdir(path);
return Object.keys(route);
}
async function releasedir(path) {
let route = findRoute(path);
if (route.releasedir) return route.releasedir(path);
}
let ws;
async function onmessage(event) {
2018-11-11 14:32:44 +01:00
const req = JSON.parse(event.data);
console.log('req', req);
2018-11-11 14:32:44 +01:00
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 === 'read') {
const buf = await read(req.path, req.fh, req.size, req.offset)
response = {
op: 'read',
buf
};
} else if (req.op === 'release') {
await release(req.path, req.fh);
response = {
op: 'release'
};
} else if (req.op === 'opendir') {
response = {
op: 'opendir',
fh: await opendir(req.path)
};
} else if (req.op === 'readdir') {
response = {
op: 'readdir',
entries: [".", "..", ...(await readdir(req.path))]
};
} else if (req.op === 'releasedir') {
await releasedir(req.path, req.fh);
response = { op: 'releasedir' };
}
} 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;
console.log('resp', response);
2018-11-11 14:32:44 +01:00
ws.send(JSON.stringify(response));
};
function tryConnect() {
ws = new WebSocket("ws://localhost:8888");
updateToolbarIcon();
ws.onopen = ws.onclose = updateToolbarIcon;
ws.onmessage = onmessage;
}
function updateToolbarIcon() {
if (ws && ws.readyState == 1) { // OPEN
chrome.browserAction.setBadgeBackgroundColor({color: 'blue'});
chrome.browserAction.setBadgeText({text: 'f'});
} else {
chrome.browserAction.setBadgeBackgroundColor({color: 'red'});
chrome.browserAction.setBadgeText({text: '!'});
}
}
tryConnect();
chrome.browserAction.onClicked.addListener(function() {
tryConnect();
});