forked from cory/tildefriends
Cory McWilliams
357d944a8d
git-svn-id: https://www.unprompted.com/svn/projects/tildefriends/trunk@3964 ed5197a5-7fde-0310-b194-c3ffbd925b24
625 lines
17 KiB
JavaScript
625 lines
17 KiB
JavaScript
import * as tfrpc from '/tfrpc.js';
|
|
|
|
const k_posts_max = 40;
|
|
const k_votes_max = 20;
|
|
|
|
var g_ready = false;
|
|
var g_selected = null;
|
|
let g_whoami = null;
|
|
|
|
var g_blocking_cache = {};
|
|
var g_following_cache = {};
|
|
var g_following_deep_cache = {};
|
|
var g_sequence = {};
|
|
|
|
async function following(db, id) {
|
|
if (g_following_cache[id]) {
|
|
return g_following_cache[id];
|
|
}
|
|
var o = await db.get(id + ":following");
|
|
const k_version = 5;
|
|
var f = o ? JSON.parse(o) : o;
|
|
if (!f || f.version != k_version) {
|
|
f = {users: [], sequence: 0, version: k_version};
|
|
}
|
|
f.users = new Set(f.users);
|
|
await ssb.sqlStream(
|
|
"SELECT "+
|
|
" sequence, "+
|
|
" json_extract(content, '$.contact') AS contact, "+
|
|
" json_extract(content, '$.following') AS following "+
|
|
"FROM messages "+
|
|
"WHERE "+
|
|
" author = ?1 AND "+
|
|
" sequence > ?2 AND "+
|
|
" json_extract(content, '$.type') = 'contact' "+
|
|
"UNION SELECT MAX(sequence) AS sequence, NULL, NULL FROM messages WHERE author = ?1 "+
|
|
"ORDER BY sequence",
|
|
[id, f.sequence],
|
|
function(row) {
|
|
if (row.following) {
|
|
f.users.add(row.contact);
|
|
} else {
|
|
f.users.delete(row.contact);
|
|
}
|
|
if (row.sequence) {
|
|
f.sequence = row.sequence;
|
|
}
|
|
});
|
|
g_sequence[id] = f.sequence;
|
|
var as_set = f.users;
|
|
f.users = Array.from(f.users).sort();
|
|
var j = JSON.stringify(f);
|
|
if (o != j) {
|
|
await db.set(id + ":following", j);
|
|
}
|
|
f.users = as_set;
|
|
g_following_cache[id] = f.users;
|
|
return f.users;
|
|
}
|
|
|
|
async function followingDeep(db, seed_ids, depth, blocked) {
|
|
if (depth <= 0) {
|
|
return seed_ids;
|
|
}
|
|
var key = JSON.stringify([seed_ids, depth, blocked]);
|
|
if (g_following_deep_cache[key]) {
|
|
return g_following_deep_cache[key];
|
|
}
|
|
var f = await Promise.all(seed_ids.map(x => following(db, x).then(x => [...x])));
|
|
var ids = [].concat(...f);
|
|
if (blocked) {
|
|
ids = ids.filter(x => !blocked.has(x));
|
|
}
|
|
var x = await followingDeep(db, [...new Set(ids)].sort(), depth - 1, blocked);
|
|
x = [...new Set([].concat(...x, ...seed_ids))].sort();
|
|
g_following_deep_cache[key] = x;
|
|
return x;
|
|
}
|
|
|
|
async function blocking(db, id) {
|
|
if (g_blocking_cache[id]) {
|
|
return g_blocking_cache[id];
|
|
}
|
|
var o = await db.get(id + ":blocking");
|
|
const k_version = 5;
|
|
var f = o ? JSON.parse(o) : o;
|
|
if (!f || f.version != k_version) {
|
|
f = {users: [], sequence: 0, version: k_version};
|
|
}
|
|
f.users = new Set(f.users);
|
|
if (!g_sequence[id] || g_sequence[id] > f.sequence) {
|
|
await ssb.sqlStream(
|
|
"SELECT "+
|
|
" sequence, "+
|
|
" json_extract(content, '$.contact') AS contact, "+
|
|
" json_extract(content, '$.blocking') AS blocking "+
|
|
"FROM messages "+
|
|
"WHERE "+
|
|
" author = ?1 AND "+
|
|
" sequence > ?2 AND "+
|
|
" json_extract(content, '$.type') = 'contact' "+
|
|
"UNION SELECT MAX(sequence) AS sequence, NULL, NULL FROM messages WHERE author = ?1 "+
|
|
"ORDER BY sequence",
|
|
[id, f.sequence],
|
|
function(row) {
|
|
if (row.blocking) {
|
|
f.users.add(row.contact);
|
|
} else {
|
|
f.users.delete(row.contact);
|
|
}
|
|
if (row.sequence) {
|
|
f.sequence = row.sequence;
|
|
}
|
|
});
|
|
g_sequence[id] = f.sequence;
|
|
}
|
|
var as_set = f.users;
|
|
f.users = Array.from(f.users).sort();
|
|
var j = JSON.stringify(f);
|
|
if (o != j) {
|
|
await db.set(id + ":blocking", j);
|
|
}
|
|
f.users = as_set;
|
|
g_blocking_cache[id] = f.users;
|
|
return f.users;
|
|
}
|
|
|
|
async function getAbout(db, id) {
|
|
var o = await db.get(id + ":about");
|
|
const k_version = 5;
|
|
var f = o ? JSON.parse(o) : o;
|
|
if (!f || f.version != k_version) {
|
|
f = {about: {}, sequence: 0, version: k_version};
|
|
}
|
|
if (g_sequence[id] === undefined || g_sequence[id] > f.sequence) {
|
|
await ssb.sqlStream(
|
|
"SELECT "+
|
|
" sequence, "+
|
|
" content "+
|
|
"FROM messages "+
|
|
"WHERE "+
|
|
" author = ?1 AND "+
|
|
" sequence > ?2 AND "+
|
|
" json_extract(content, '$.type') = 'about' AND "+
|
|
" json_extract(content, '$.about') = ?1 "+
|
|
"UNION SELECT MAX(sequence) as sequence, NULL FROM messages WHERE author = ?1 "+
|
|
"ORDER BY sequence",
|
|
[id, f.sequence],
|
|
function(row) {
|
|
if (row.content) {
|
|
var about = {};
|
|
try {
|
|
about = JSON.parse(row.content);
|
|
} catch {
|
|
}
|
|
delete about.about;
|
|
delete about.type;
|
|
f.about = Object.assign(f.about, about);
|
|
}
|
|
if (row.sequence) {
|
|
f.sequence = Math.max(f.sequence, row.sequence);
|
|
}
|
|
});
|
|
g_sequence[id] = f.sequence;
|
|
var j = JSON.stringify(f);
|
|
if (o != j) {
|
|
await db.set(id + ":about", j);
|
|
}
|
|
}
|
|
return f.about;
|
|
}
|
|
|
|
function fnv32a(value)
|
|
{
|
|
var result = 0x811c9dc5;
|
|
for (var i = 0; i < value.length; i++) {
|
|
result ^= value.charCodeAt(i);
|
|
result += (result << 1) + (result << 4) + (result << 7) + (result << 8) + (result << 24);
|
|
}
|
|
return result >>> 0;
|
|
}
|
|
|
|
async function getRecentPostsSingleId(db, id, limit) {
|
|
var recent = [];
|
|
await ssb.sqlStream(
|
|
"SELECT "+
|
|
" rowid, "+
|
|
" id, "+
|
|
" timestamp "+
|
|
"FROM messages "+
|
|
"WHERE "+
|
|
" author = ? AND "+
|
|
" json_extract(content, '$.type') = 'post' "+
|
|
"ORDER BY sequence DESC LIMIT ?",
|
|
[id, limit],
|
|
function(row) {
|
|
if (row.id) {
|
|
recent.push({id: row.id, timestamp: row.timestamp});
|
|
}
|
|
});
|
|
recent.sort((x, y) => y.timestamp - x.timestamp);
|
|
return recent.map(x => x.id);
|
|
}
|
|
|
|
async function getRecentPostIds(db, id, ids, limit) {
|
|
if (ids.length == 1) {
|
|
return await getRecentPostsSingleId(db, ids[0], limit);
|
|
}
|
|
const k_version = 11;
|
|
const k_batch_max = 32;
|
|
var o = await db.get(id + ':recent_posts');
|
|
var recent = [];
|
|
var f = o ? JSON.parse(o) : o;
|
|
var ids_hash = fnv32a(JSON.stringify(ids));
|
|
if (!f || f.version != k_version || f.ids_hash != ids_hash) {
|
|
f = {recent: [], rowid: 0, version: k_version, ids_hash: ids_hash};
|
|
}
|
|
var row_id_max = 0;
|
|
await ssb.sqlStream(
|
|
"SELECT MAX(rowid) as rowid FROM messages",
|
|
[],
|
|
function(row) {
|
|
row_id_max = row.rowid;
|
|
});
|
|
for (var i = 0; i < ids.length; i += k_batch_max) {
|
|
var ids_batch = ids.slice(i, Math.min(i + k_batch_max, ids.length));
|
|
await ssb.sqlStream(
|
|
"SELECT "+
|
|
" rowid, "+
|
|
" id, "+
|
|
" timestamp "+
|
|
"FROM messages "+
|
|
"WHERE "+
|
|
" rowid > ? AND "+
|
|
" rowid <= ? AND "+
|
|
" author IN (" + ids_batch.map(x => '?').join(", ") + ") "+
|
|
"ORDER BY timestamp DESC LIMIT ?",
|
|
[].concat([f.rowid, row_id_max], ids_batch, [limit]),
|
|
function(row) {
|
|
if (row.id) {
|
|
recent.push({id: row.id, timestamp: row.timestamp});
|
|
}
|
|
});
|
|
}
|
|
f.rowid = row_id_max;
|
|
f.recent = [].concat(recent, f.recent);
|
|
var have = {};
|
|
f.recent = f.recent.filter(function(x) {
|
|
if (!have[x.id]) {
|
|
have[x.id] = true;
|
|
return true;
|
|
}
|
|
});
|
|
f.recent.sort((x, y) => y.timestamp - x.timestamp);
|
|
f.recent = f.recent.slice(0, limit);
|
|
var j = JSON.stringify(f);
|
|
if (o != j) {
|
|
await db.set(id + ":recent_posts", j);
|
|
}
|
|
return f.recent.map(x => x.id);
|
|
}
|
|
|
|
async function getRecentPostIds2(db, id, ids, start_time) {
|
|
if (ids.length == 1) {
|
|
return getRecentPostsSingleId(db, ids[0], 20);
|
|
}
|
|
const k_batch_max = 32;
|
|
var row_id_max = 0;
|
|
await ssb.sqlStream(
|
|
"SELECT MAX(rowid) as rowid FROM messages",
|
|
[],
|
|
function(row) {
|
|
row_id_max = row.rowid;
|
|
});
|
|
var posts_by_author = {};
|
|
for (var i = 0; i < ids.length; i += k_batch_max) {
|
|
var ids_batch = ids.slice(i, Math.min(i + k_batch_max, ids.length));
|
|
await ssb.sqlStream(
|
|
"SELECT "+
|
|
" author, "+
|
|
" id "+
|
|
"FROM messages "+
|
|
"WHERE "+
|
|
" author IN (" + ids_batch.map(x => '?').join(", ") + ") AND "+
|
|
" timestamp > ? AND "+
|
|
" rowid <= ? AND "+
|
|
" json_extract(content, '$.type') = 'post' "+
|
|
"ORDER BY timestamp DESC",
|
|
[].concat(ids_batch, [start_time, row_id_max]),
|
|
function(row) {
|
|
if (row.id) {
|
|
if (!posts_by_author[row.author]) {
|
|
posts_by_author[row.author] = [];
|
|
}
|
|
posts_by_author[row.author].push(row.id);
|
|
}
|
|
});
|
|
}
|
|
return Object.values(posts_by_author).map(x => x[0]);
|
|
}
|
|
|
|
async function getRelatedPostIds(db, message, ids, limit) {
|
|
const k_batch_max = 16;
|
|
var recent = [];
|
|
var row_id_max = 0;
|
|
await ssb.sqlStream(
|
|
"SELECT MAX(rowid) as rowid FROM messages",
|
|
[],
|
|
function(row) {
|
|
row_id_max = row.rowid;
|
|
});
|
|
var id = message.id;
|
|
try {
|
|
id = JSON.parse(message.content).root || id;
|
|
} catch {
|
|
}
|
|
for (var i = 0; i < ids.length; i += k_batch_max) {
|
|
var ids_batch = ids.slice(i, Math.min(i + k_batch_max, ids.length));
|
|
await ssb.sqlStream(
|
|
"SELECT "+
|
|
" rowid, "+
|
|
" id, "+
|
|
" timestamp "+
|
|
"FROM messages "+
|
|
"WHERE "+
|
|
" timestamp >= ? AND "+
|
|
" rowid <= ? AND "+
|
|
" author IN (" + ids_batch.map(x => '?').join(", ") + ") AND "+
|
|
" json_extract(content, '$.type') = 'post' AND "+
|
|
" (id = ? OR json_extract(content, '$.root') = ?) "+
|
|
"ORDER BY timestamp DESC LIMIT ?",
|
|
[].concat([message.timestamp || 0, row_id_max], ids_batch, [message.id, id, limit]),
|
|
function(row) {
|
|
if (row.id) {
|
|
recent.push({id: row.id, timestamp: row.timestamp});
|
|
}
|
|
});
|
|
}
|
|
recent.sort((x, y) => y.timestamp - x.timestamp);
|
|
recent = recent.slice(0, limit);
|
|
return recent.map(x => x.id);
|
|
}
|
|
|
|
async function getVotes(db, id) {
|
|
var o = await db.get(id + ":votes");
|
|
const k_version = 7;
|
|
var f = o ? JSON.parse(o) : o;
|
|
if (!f || f.version != k_version) {
|
|
f = {votes: [], sequence: 0, version: k_version};
|
|
}
|
|
if (g_sequence[id] === undefined || g_sequence[id] > f.sequence) {
|
|
var votes = [];
|
|
await ssb.sqlStream(
|
|
"SELECT "+
|
|
" author, "+
|
|
" id, "+
|
|
" sequence, "+
|
|
" timestamp, "+
|
|
" content "+
|
|
"FROM messages "+
|
|
"WHERE "+
|
|
" author = ? AND "+
|
|
" sequence > ? AND "+
|
|
" json_extract(content, '$.type') = 'vote' "+
|
|
"UNION SELECT NULL, NULL, MAX(sequence) AS sequence, NULL, NULL FROM messages WHERE author = ? "+
|
|
"ORDER BY sequence DESC LIMIT ?",
|
|
[id, f.sequence, id, k_votes_max],
|
|
function(row) {
|
|
if (row.id) {
|
|
votes.push(row);
|
|
}
|
|
if (row.sequence) {
|
|
f.sequence = Math.max(f.sequence, row.sequence);
|
|
}
|
|
});
|
|
g_sequence[id] = f.sequence;
|
|
f.votes = [].concat(votes, f.votes).slice(0, k_votes_max);
|
|
var j = JSON.stringify(f);
|
|
if (o != j) {
|
|
await db.set(id + ":votes", j);
|
|
}
|
|
}
|
|
return f.votes;
|
|
}
|
|
|
|
async function getPosts(db, ids) {
|
|
var posts = [];
|
|
if (ids.length) {
|
|
await ssb.sqlStream(
|
|
"SELECT rowid, * FROM messages WHERE id IN (" + ids.map(x => "?").join(", ") + ") ORDER BY timestamp DESC",
|
|
ids,
|
|
row => posts.push(row));
|
|
}
|
|
return posts;
|
|
}
|
|
|
|
tfrpc.register(async function ready() {
|
|
let identities = await ssb.getIdentities();
|
|
let whoami = await app.localStorageGet('whoami');
|
|
await tfrpc.rpc.set_identities(identities);
|
|
g_ready = true;
|
|
refresh_internal(whoami, g_selected, true);
|
|
});
|
|
|
|
tfrpc.register(async function store_blob(blob) {
|
|
if (Array.isArray(blob)) {
|
|
blob = Uint8Array.from(blob);
|
|
}
|
|
return await ssb.blobStore(blob);
|
|
});
|
|
|
|
ssb.addEventListener('broadcasts', async function() {
|
|
await tfrpc.rpc.set('broadcasts', await ssb.getBroadcasts());
|
|
});
|
|
|
|
core.register('onConnectionsChanged', async function() {
|
|
var connections = await ssb.connections();
|
|
await tfrpc.rpc.set('connections', connections);
|
|
});
|
|
|
|
async function updateSequences(db) {
|
|
var k_batch_max = 100;
|
|
var changes = {};
|
|
var keys = Object.keys(g_sequence);
|
|
for (var i = 0; i < keys.length; i += k_batch_max) {
|
|
var ids_batch = keys.slice(i, Math.min(i + k_batch_max, keys.length));
|
|
await ssb.sqlStream(
|
|
"SELECT "+
|
|
" author, "+
|
|
" MAX(sequence) AS sequence "+
|
|
"FROM messages "+
|
|
"WHERE "+
|
|
" author IN (" + ids_batch.map(x => '?').join(", ") + ")",
|
|
ids_batch,
|
|
function(row) {
|
|
if (g_sequence[row.author] != row.sequence) {
|
|
g_sequence[row.author] = row.sequence;
|
|
changes[row.author] = row.sequence;
|
|
}
|
|
});
|
|
}
|
|
return changes;
|
|
}
|
|
|
|
async function refresh_internal(whoami, selected, force) {
|
|
if (whoami !== g_whoami || selected !== g_selected || force) {
|
|
if (g_whoami !== whoami && whoami) {
|
|
await app.localStorageSet('whoami', whoami);
|
|
}
|
|
g_whoami = whoami;
|
|
g_selected = selected;
|
|
} else {
|
|
return;
|
|
}
|
|
if (typeof(whoami) !== 'string') {
|
|
return;
|
|
}
|
|
if (!g_ready) {
|
|
return;
|
|
}
|
|
var timing = [];
|
|
timing.push({name: 'start', time: new Date()});
|
|
g_following_cache = {};
|
|
g_following_deep_cache = {};
|
|
await tfrpc.rpc.clear();
|
|
await tfrpc.rpc.set_identities(await ssb.getIdentities());
|
|
await tfrpc.rpc.set('all_identities', await ssb.getAllIdentities());
|
|
await tfrpc.rpc.set('selected', selected);
|
|
var db = await database("ssb");
|
|
g_sequence = {};
|
|
try {
|
|
g_sequence = JSON.parse(await db.get('sequence'));
|
|
} catch (e) {
|
|
}
|
|
await updateSequences(db);
|
|
timing.push({name: 'init', time: new Date()});
|
|
var blocked = await blocking(db, whoami);
|
|
timing.push({name: 'blocked', time: new Date()});
|
|
var all_followed = await followingDeep(db, [whoami], 2, blocked);
|
|
timing.push({name: 'all_followed', time: new Date()});
|
|
let actual_selected = (selected ? [selected] : all_followed) ?? [];
|
|
await Promise.all([
|
|
tfrpc.rpc.set('whoami', whoami),
|
|
tfrpc.rpc.set('broadcasts', await ssb.getBroadcasts()),
|
|
tfrpc.rpc.set('connections', await ssb.connections()),
|
|
tfrpc.rpc.set('apps', await core.apps()),
|
|
]);
|
|
timing.push({name: 'core', time: new Date()});
|
|
var ids;
|
|
if (selected && selected.startsWith('%')) {
|
|
var m = await getPosts(db, [selected]);
|
|
m = m.length ? m[0] : {id: selected};
|
|
ids = await getRelatedPostIds(db, m, all_followed, k_posts_max);
|
|
} else {
|
|
ids = await getRecentPostIds2(db, whoami, actual_selected, (new Date()).valueOf() - (24 * 60 * 60 * 1000));
|
|
}
|
|
timing.push({name: 'get_post_ids', time: new Date()});
|
|
var posts = await getPosts(db, ids);
|
|
timing.push({name: 'get_posts', time: new Date()});
|
|
var roots = posts.map(function(x) {
|
|
try {
|
|
return JSON.parse(x.content).root;
|
|
} catch {
|
|
return null;
|
|
}
|
|
});
|
|
var have = new Set(posts.map(x => x.id));
|
|
roots = [...new Set(roots)].filter(x => x && !have.has(x));
|
|
var all_posts = [].concat(posts, await getPosts(db, roots));
|
|
timing.push({name: 'get_root_posts', time: new Date()});
|
|
await tfrpc.rpc.push_posts(all_posts);
|
|
timing.push({name: 'send_posts', time: new Date()});
|
|
let all_users = {};
|
|
await Promise.all(all_followed.map(id => getAbout(db, id).then(function(results) {
|
|
if (Object.keys(results).length) {
|
|
all_users[id] = results;
|
|
}
|
|
})));
|
|
await tfrpc.rpc.push_users(all_users);
|
|
timing.push({name: 'about', time: new Date()});
|
|
var all_votes = [];
|
|
for (let id of all_followed) {
|
|
var results = await getVotes(db, id);
|
|
if (results.length) {
|
|
all_votes.push(results);
|
|
}
|
|
}
|
|
all_votes = all_votes.flat();
|
|
await tfrpc.rpc.push_votes(all_votes);
|
|
timing.push({name: 'votes', time: new Date()});
|
|
if (selected && selected.length == 1 && selected[0].startsWith('@')) {
|
|
let size = 0;
|
|
await ssb.sqlStream(
|
|
'SELECT SUM(LENGTH(content)) AS length FROM messages WHERE author = ?1',
|
|
selected,
|
|
function(row) {
|
|
size = row.length;
|
|
});
|
|
let users = {};
|
|
users[selected[0]] = {size: size};
|
|
await tfrpc.rpc.push_users(users);
|
|
}
|
|
await tfrpc.rpc.push_following(Object.fromEntries(all_followed.map(id => [id, [...(g_following_cache[id] || [])]])));
|
|
timing.push({name: 'following', time: new Date()});
|
|
await tfrpc.rpc.push_blocking(whoami, [...(g_blocking_cache[whoami] || [])]);
|
|
timing.push({name: 'send_blocking', time: new Date()});
|
|
await db.set('sequence', JSON.stringify(g_sequence));
|
|
|
|
var times = {};
|
|
var previous = null;
|
|
for (let t of timing) {
|
|
times[t.name] = (t.time - (previous || t).time) / 1000.0 + ' s';
|
|
previous = t;
|
|
}times.total = (new Date() - timing[0].time) / 1000.0 + ' s';
|
|
await tfrpc.rpc.ready(times);
|
|
}
|
|
|
|
ssb.addEventListener('message', async function(id) {
|
|
var db = await database("ssb");
|
|
var posts = await getPosts(db, [id]);
|
|
for (let post of posts) {
|
|
if (post.author == g_whoami ||
|
|
JSON.parse(post.content).type != 'post') {
|
|
await tfrpc.rpc.push_posts([post]);
|
|
} else {
|
|
await tfrpc.rpc.add_unread(1);
|
|
}
|
|
}
|
|
});
|
|
|
|
tfrpc.register(async function refresh(whoami, selected, force) {
|
|
return refresh_internal(whoami, selected, force);
|
|
});
|
|
|
|
tfrpc.register(async function createIdentity() {
|
|
return ssb.createIdentity();
|
|
});
|
|
|
|
tfrpc.register(async function appendMessage(message) {
|
|
await addAppSources(message);
|
|
return ssb.appendMessageWithIdentity(g_whoami, message);
|
|
});
|
|
|
|
async function addAppSources(message) {
|
|
if (message.mentions) {
|
|
for (let mention of message.mentions) {
|
|
if (mention.type == 'application/tildefriends') {
|
|
var blob = await ssb.blobGet(mention.link);
|
|
var json = JSON.parse(utf8Decode(blob));
|
|
for (let file of Object.keys(json.files)) {
|
|
message.mentions.push({
|
|
name: file,
|
|
link: json.files[file],
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
core.register('message', async function(m) {
|
|
if (m.message) {
|
|
if (m.message.connect) {
|
|
await ssb.connect(m.message.connect);
|
|
}
|
|
} else if (m.event == 'hashChange') {
|
|
let hash = m.hash.length > 1 ? m.hash.substring(1) : null;
|
|
let changed = g_selected !== hash;
|
|
await refresh_internal(g_whoami, hash, changed);
|
|
} else if (m.event == 'focus' || m.event == 'blur') {
|
|
/* Shh. */
|
|
} else {
|
|
print(JSON.stringify(m));
|
|
}
|
|
});
|
|
|
|
async function main() {
|
|
if (core.user?.credentials?.permissions?.authenticated) {
|
|
await app.setDocument(utf8Decode(await getFile("index.html")));
|
|
} else {
|
|
await app.setDocument('<div style="color: #f00">You must be signed in to use this app at this time. Login at the top right.</div>');
|
|
}
|
|
}
|
|
|
|
main(); |