apps: Add a very wip web app.
All checks were successful
Build Tilde Friends / Build-All (push) Successful in 30m8s
All checks were successful
Build Tilde Friends / Build-All (push) Successful in 30m8s
This commit is contained in:
parent
894c72a82f
commit
b9000c154f
5
apps/web.json
Normal file
5
apps/web.json
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
"type": "tildefriends-app",
|
||||||
|
"emoji": "🕸",
|
||||||
|
"previous": "&n7hu5b8/TsfiG6FDlCRG5nPCrIdCr96+xpIJ/aQT/uM=.sha256"
|
||||||
|
}
|
85
apps/web/app.js
Normal file
85
apps/web/app.js
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
let g_hash;
|
||||||
|
|
||||||
|
async function query(sql, params) {
|
||||||
|
let results = [];
|
||||||
|
await ssb.sqlAsync(sql, params, function(row) {
|
||||||
|
results.push(row);
|
||||||
|
});
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function resolve(id) {
|
||||||
|
try {
|
||||||
|
let blob = await ssb.blobGet(id);
|
||||||
|
if (blob) {
|
||||||
|
let json;
|
||||||
|
try {
|
||||||
|
json = JSON.parse(utf8Decode(blob));
|
||||||
|
} catch {
|
||||||
|
return {id: utf8Decode(blob)};
|
||||||
|
}
|
||||||
|
if (json?.links) {
|
||||||
|
for (let [key, value] of Object.entries(json.links)) {
|
||||||
|
json.links[key] = await resolve(value);
|
||||||
|
}
|
||||||
|
return json;
|
||||||
|
} else {
|
||||||
|
return 'huh?' + json;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return `missing<${id}>`;
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
return id + ': ' + e.message;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function get_names(identities) {
|
||||||
|
return Object.fromEntries((await query(`
|
||||||
|
SELECT author, name FROM (
|
||||||
|
SELECT
|
||||||
|
messages.author,
|
||||||
|
RANK() OVER (PARTITION BY messages.author ORDER BY messages.sequence DESC) AS author_rank,
|
||||||
|
messages.content ->> 'name' AS name
|
||||||
|
FROM messages
|
||||||
|
JOIN json_each(?) AS identities ON identities.value = messages.author
|
||||||
|
WHERE
|
||||||
|
json_extract(messages.content, '$.type') = 'about' AND
|
||||||
|
content ->> 'about' = messages.author AND name IS NOT NULL)
|
||||||
|
WHERE author_rank = 1
|
||||||
|
`, [JSON.stringify(identities)])).map(x => [x.author, x.name]));
|
||||||
|
}
|
||||||
|
|
||||||
|
async function render(hash) {
|
||||||
|
g_hash = hash;
|
||||||
|
if (!hash) {
|
||||||
|
let sites = (await query(`
|
||||||
|
SELECT site.author, site.id
|
||||||
|
FROM messages site
|
||||||
|
WHERE site.content ->> 'type' = 'web-init'
|
||||||
|
`, []));
|
||||||
|
let names = await get_names(sites.map(x => x.author));
|
||||||
|
if (hash === g_hash) {
|
||||||
|
await app.setDocument(`<ul style="background-color: #ddd">${sites.map(x => `<li><a target="_top" href="#${encodeURIComponent(x.id)}">${names[x.author] ?? x.author} - ${x.id}</a></li>`).join('\n')}</ul>`);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let site_id = hash.charAt(0) == '#' ? decodeURIComponent(hash.substring(1)) : decodeURIComponent(hash);
|
||||||
|
await app.setDocument(`<html style="margin: 0; padding: 0; width: 100vw; height: 100vh; margin: 0; padding: 0">
|
||||||
|
<body style="display: flex; flex-direction: column; width: 100vw; height: 100vh">
|
||||||
|
<iframe src="${encodeURIComponent(site_id)}/index.html" style="flex: 1 1; border: 0; background-color: #fff"></iframe>
|
||||||
|
</body>
|
||||||
|
</html>`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
core.register('message', async function message_handler(message) {
|
||||||
|
if (message.event == 'hashChange') {
|
||||||
|
await render(message.hash);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
async function main() {
|
||||||
|
render(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
main();
|
59
apps/web/handler.js
Normal file
59
apps/web/handler.js
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
async function query(sql, params) {
|
||||||
|
let results = [];
|
||||||
|
await ssb.sqlAsync(sql, params, function(row) {
|
||||||
|
results.push(row);
|
||||||
|
});
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
|
function guess_content_type(name) {
|
||||||
|
if (name.endsWith('.html')) {
|
||||||
|
return 'text/html; charset=UTF-8';
|
||||||
|
} else if (name.endsWith('.js') || name.endsWith('.mjs')) {
|
||||||
|
return 'text/javascript; charset=UTF-8';
|
||||||
|
} else if (name.endsWith('.css')) {
|
||||||
|
return 'text/stylesheet; charset=UTF-8';
|
||||||
|
} else {
|
||||||
|
return 'application/binary';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function main() {
|
||||||
|
let path = request.path.replaceAll(/(%[0-9a-fA-F]{2})/g, x => String.fromCharCode(parseInt(x.substring(1), 16)));
|
||||||
|
let match = path.match(/^(%.{44}\.sha256)(?:\/)?(.*)$/);
|
||||||
|
|
||||||
|
let content_type = guess_content_type(request.path);
|
||||||
|
let root = await query(`
|
||||||
|
SELECT root.content ->> 'root' AS root
|
||||||
|
FROM messages site
|
||||||
|
JOIN messages root
|
||||||
|
ON site.id = ? AND root.author = site.author AND root.content ->> 'site' = site.id
|
||||||
|
ORDER BY root.sequence DESC LIMIT 1
|
||||||
|
`, [match[1]]);
|
||||||
|
let root_id = root[0]['root'];
|
||||||
|
let last_id = root_id;
|
||||||
|
let blob = await ssb.blobGet(root_id);
|
||||||
|
try {
|
||||||
|
for (let part of match[2]?.split('/')) {
|
||||||
|
let dir = JSON.parse(utf8Decode(blob));
|
||||||
|
last_id = dir?.links[part];
|
||||||
|
blob = await ssb.blobGet(dir?.links[part]);
|
||||||
|
content_type = guess_content_type(part);
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
}
|
||||||
|
|
||||||
|
respond({
|
||||||
|
status_code: 200,
|
||||||
|
data: blob ? utf8Decode(blob) : `${last_id} not found`,
|
||||||
|
content_type: content_type,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
main().catch(function(e) {
|
||||||
|
respond({
|
||||||
|
status_code: 200,
|
||||||
|
data: `${e.message}\n${e.stack}`,
|
||||||
|
content_type: 'text/plain',
|
||||||
|
});
|
||||||
|
});
|
Loading…
x
Reference in New Issue
Block a user