import {LitElement, html, css, guard, until} from './lit-all.min.js'; import * as tfrpc from '/static/tfrpc.js'; import {styles} from './tf-styles.js'; class TfElement extends LitElement { static get properties() { return { whoami: {type: String}, hash: {type: String}, tab: {type: String}, broadcasts: {type: Array}, connections: {type: Array}, loading: {type: Boolean}, loaded: {type: Boolean}, following: {type: Array}, users: {type: Object}, ids: {type: Array}, channels: {type: Array}, channels_unread: {type: Object}, channels_latest: {type: Object}, guest: {type: Boolean}, url: {type: String}, }; } static styles = styles; constructor() { super(); let self = this; this.hash = '#'; this.tab = 'news'; this.broadcasts = []; this.connections = []; this.following = []; this.users = {}; this.loaded = false; this.channels = []; this.channels_unread = {}; this.channels_latest = {}; this.loading_latest = 0; this.loading_latest_scheduled = 0; tfrpc.rpc.getBroadcasts().then((b) => { self.broadcasts = b || []; }); tfrpc.rpc.getConnections().then((c) => { self.connections = c || []; }); tfrpc.rpc.getHash().then((hash) => self.set_hash(hash)); tfrpc.register(function hashChanged(hash) { self.set_hash(hash); }); tfrpc.register(async function notifyNewMessage(id) { await self.fetch_new_message(id); }); tfrpc.register(function set(name, value) { if (name === 'broadcasts') { self.broadcasts = value; } else if (name === 'connections') { self.connections = value; } else if (name === 'identity') { self.whoami = value; } }); this.initial_load(); } async initial_load() { let whoami = await tfrpc.rpc.getActiveIdentity(); let ids = (await tfrpc.rpc.getIdentities()) || []; this.url = await tfrpc.rpc.url(); this.whoami = whoami ?? (ids.length ? ids[0] : undefined); this.guest = !this.whoami?.length; this.ids = ids; await this.load_channels(); } async load_channels() { let channels = await tfrpc.rpc.query( ` SELECT content ->> 'channel' AS channel, content ->> 'subscribed' AS subscribed FROM messages WHERE author = ? AND content ->> 'type' = 'channel' ORDER BY sequence `, [this.whoami] ); let channel_map = {}; for (let row of channels) { if (row.subscribed) { channel_map[row.channel] = true; } else { delete channel_map[row.channel]; } } this.channels = Object.keys(channel_map).sort(); } connectedCallback() { super.connectedCallback(); this._keydown = this.keydown.bind(this); window.addEventListener('keydown', this._keydown); } disconnectedCallback() { super.disconnectedCallback(); window.removeEventListener('keydown', this._keydown); } keydown(event) { if (event.altKey && event.key == 'ArrowUp') { this.next_channel(-1); event.preventDefault(); } else if (event.altKey && event.key == 'ArrowDown') { this.next_channel(1); event.preventDefault(); } } next_channel(delta) { let channel_names = ['', '@', '🔐', ...this.channels.map((x) => '#' + x)]; let index = channel_names.indexOf(this.hash.substring(1)); index = index != -1 ? index + delta : 0; tfrpc.rpc.setHash( '#' + encodeURIComponent( channel_names[(index + channel_names.length) % channel_names.length] ) ); } set_hash(hash) { this.hash = decodeURIComponent(hash || '#'); if (this.hash.startsWith('#q=')) { this.tab = 'search'; } else if (this.hash === '#connections') { this.tab = 'connections'; } else if (this.hash.startsWith('#sql=')) { this.tab = 'query'; } else { this.tab = 'news'; } } async fetch_about(following, users) { let ids = Object.keys(following).sort(); const k_cache_version = 1; let cache = await tfrpc.rpc.databaseGet('about'); let original_cache = cache; cache = cache ? JSON.parse(cache) : {}; if (cache.version !== k_cache_version) { cache = { version: k_cache_version, about: {}, last_row_id: 0, }; } let max_row_id = ( await tfrpc.rpc.query( ` SELECT MAX(rowid) AS max_row_id FROM messages `, [] ) )[0].max_row_id; for (let id of Object.keys(cache.about)) { if (ids.indexOf(id) == -1) { delete cache.about[id]; } } let abouts = await tfrpc.rpc.query( ` SELECT messages.author, json(messages.content) AS content, messages.sequence FROM messages, json_each(?1) AS following WHERE messages.author = following.value AND messages.rowid > ?3 AND messages.rowid <= ?4 AND json_extract(messages.content, '$.type') = 'about' UNION SELECT messages.author, json(messages.content) AS content, messages.sequence FROM messages, json_each(?2) AS following WHERE messages.author = following.value AND messages.rowid <= ?4 AND json_extract(messages.content, '$.type') = 'about' ORDER BY messages.author, messages.sequence `, [ JSON.stringify(ids.filter((id) => cache.about[id])), JSON.stringify(ids.filter((id) => !cache.about[id])), cache.last_row_id, max_row_id, ] ); for (let about of abouts) { let content = JSON.parse(about.content); if (content.about === about.author) { delete content.type; delete content.about; cache.about[about.author] = Object.assign( cache.about[about.author] || {}, content ); } } cache.last_row_id = max_row_id; let new_cache = JSON.stringify(cache); if (new_cache !== original_cache) { let start_time = new Date(); tfrpc.rpc.databaseSet('about', new_cache).then(function () { console.log('saving about took', (new Date() - start_time) / 1000); }); } users = users || {}; for (let id of Object.keys(cache.about)) { users[id] = Object.assign( {follow_depth: following[id]?.d}, users[id] || {}, cache.about[id] ); } return Object.assign({}, users); } async fetch_new_message(id) { let messages = await tfrpc.rpc.query( ` SELECT messages.id, previous, author, sequence, timestamp, hash, json(content) AS content, signature FROM messages JOIN json_each(?) AS following ON messages.author = following.value WHERE messages.id = ? `, [JSON.stringify(this.following), id] ); for (let message of messages) { if ( message.author == this.whoami && JSON.parse(message.content)?.type == 'channel' ) { this.load_channels(); } } this.schedule_load_latest(); } async _handle_whoami_changed(event) { let old_id = this.whoami; let new_id = event.srcElement.selected; console.log('received', new_id); if (this.whoami !== new_id) { console.log(event); this.whoami = new_id; console.log(`whoami ${old_id} => ${new_id}`); await tfrpc.rpc.localStorageSet('whoami', new_id); } } async create_identity() { if (confirm('Are you sure you want to create a new identity?')) { await tfrpc.rpc.createIdentity(); this.ids = (await tfrpc.rpc.getIdentities()) || []; if (this.ids && !this.whoami) { this.whoami = this.ids[0]; } } } async get_latest_private(following) { const k_version = 1; // { "version": 1, "range": [1234, 5678], messages: [ "%1.sha256", "%2.sha256", ... ], latest: rowid } let cache = JSON.parse( (await tfrpc.rpc.databaseGet(`private:${this.whoami}`)) ?? '{}' ); if (cache.version !== k_version) { cache = { version: k_version, messages: [], range: [], }; } let latest = ( await tfrpc.rpc.query('SELECT MAX(rowid) AS latest FROM messages') )[0].latest; let ranges = []; const k_chunk_size = 512; if (cache.range.length) { for (let i = cache.range[1]; i < latest; i += k_chunk_size) { ranges.push([i, Math.min(i + k_chunk_size, latest), true]); } for (let i = cache.range[0]; i >= 0; i -= k_chunk_size) { ranges.push([ Math.max(i - k_chunk_size, 0), Math.min(cache.range[0], i + k_chunk_size), false, ]); } } else { for (let i = 0; i < latest; i += k_chunk_size) { ranges.push([i, Math.min(i + k_chunk_size, latest), true]); } } for (let range of ranges) { let messages = await tfrpc.rpc.query( ` SELECT messages.rowid, messages.id, json(content) AS content FROM messages WHERE messages.rowid > ?1 AND messages.rowid <= ?2 AND json(messages.content) LIKE '"%' ORDER BY sequence DESC `, [range[0], range[1]] ); messages = (await this.decrypt(messages)).filter((x) => x.decrypted); if (messages.length) { cache.latest = Math.max( cache.latest ?? 0, ...messages.map((x) => x.rowid) ); if (range[2]) { cache.messages = [...cache.messages, ...messages.map((x) => x.id)]; } else { cache.messages = [...messages.map((x) => x.id), ...cache.messages]; } } cache.range[0] = Math.min(cache.range[0] ?? range[0], range[0]); cache.range[1] = Math.max(cache.range[1] ?? range[1], range[1]); await tfrpc.rpc.databaseSet( `private:${this.whoami}`, JSON.stringify(cache) ); } return cache.latest; } async load_channels_latest(following) { let start_time = new Date(); let latest_private = this.get_latest_private(following); let channels = await tfrpc.rpc.query( ` SELECT channels.value AS channel, MAX(messages.rowid) AS rowid FROM messages JOIN json_each(?1) AS channels ON messages.content ->> 'channel' = channels.value JOIN json_each(?2) AS following ON messages.author = following.value WHERE messages.content ->> 'type' = 'post' AND messages.content ->> 'root' IS NULL AND messages.author != ?4 GROUP by channel UNION SELECT '' AS channel, MAX(messages.rowid) AS rowid FROM messages JOIN json_each(?2) AS following ON messages.author = following.value WHERE messages.content ->> 'type' = 'post' AND messages.content ->> 'root' IS NULL AND messages.author != ?4 UNION SELECT '@' AS channel, MAX(messages.rowid) AS rowid FROM messages_fts(?3) JOIN messages ON messages.rowid = messages_fts.rowid JOIN json_each(?2) AS following ON messages.author = following.value WHERE messages.author != ?4 `, [ JSON.stringify(this.channels), JSON.stringify(following), '"' + this.whoami.replace('"', '""') + '"', this.whoami, ] ); this.channels_latest = Object.fromEntries( channels.map((x) => [x.channel, x.rowid]) ); console.log('channels took', (new Date() - start_time) / 1000.0); let self = this; start_time = new Date(); latest_private.then(function (latest) { self.channels_latest = Object.assign({}, self.channels_latest, { '🔐': latest, }); console.log('private took', (new Date() - start_time) / 1000.0); }); } _schedule_load_latest_timer() { --this.loading_latest_scheduled; this.schedule_load_latest(); } schedule_load_latest() { if (!this.loading_latest) { this.shadowRoot.getElementById('tf-tab-news')?.load_latest(); this.load(); } else if (!this.loading_latest_scheduled) { this.loading_latest_scheduled++; setTimeout(this._schedule_load_latest_timer.bind(this), 5000); } } async fetch_user_info(users) { let info = await tfrpc.rpc.query( ` SELECT messages.author, MAX(messages.sequence) AS max_seq, MAX(timestamp) AS max_ts FROM messages JOIN json_each(?) AS following ON messages.author = following.value GROUP BY messages.author `, [JSON.stringify(Object.keys(users))] ); for (let row of info) { users[row.author].seq = row.max_seq; users[row.author].ts = row.max_ts; } return users; } async load() { this.loading_latest = true; try { let start_time = new Date(); let whoami = this.whoami; let following = await tfrpc.rpc.following([whoami], 2); let users = {}; let by_count = []; for (let [id, v] of Object.entries(following)) { users[id] = { following: v.of, blocking: v.ob, followed: v.if, blocked: v.ib, }; by_count.push({count: v.of, id: id}); } this.load_channels_latest(Object.keys(following)); this.channels_unread = JSON.parse( (await tfrpc.rpc.databaseGet('unread')) ?? '{}' ); this.following = Object.keys(following); users = await this.fetch_about(following, users); console.log( 'about took', (new Date() - start_time) / 1000.0, 'seconds for', Object.keys(users).length, 'users' ); start_time = new Date(); users = await this.fetch_user_info(users); console.log( 'user info took', (new Date() - start_time) / 1000.0, 'seconds' ); this.users = users; console.log( `load finished ${whoami} => ${this.whoami} in ${(new Date() - start_time) / 1000}` ); this.whoami = whoami; this.loaded = whoami; } finally { this.loading_latest = false; } } channel_set_unread(event) { this.channels_unread[event.detail.channel ?? ''] = event.detail.unread; this.channels_unread = Object.assign({}, this.channels_unread); tfrpc.rpc.databaseSet('unread', JSON.stringify(this.channels_unread)); } async decrypt(messages) { let whoami = this.whoami; return Promise.all( messages.map(async function (message) { let content; try { content = JSON.parse(message?.content); } catch {} if (typeof content === 'string') { let decrypted; try { decrypted = await tfrpc.rpc.try_decrypt(whoami, content); } catch {} if (decrypted) { try { message.decrypted = JSON.parse(decrypted); } catch { message.decrypted = decrypted; } } } return message; }) ); } render_tab() { let following = this.following; let users = this.users; if (this.tab === 'news') { return html` `; } else if (this.tab === 'connections') { return html` `; } else if (this.tab === 'search') { return html` `; } else if (this.tab === 'query') { return html` `; } } async set_tab(tab) { this.tab = tab; if (tab === 'news') { await tfrpc.rpc.setHash('#'); } else if (tab === 'connections') { await tfrpc.rpc.setHash('#connections'); } else if (tab === 'query') { await tfrpc.rpc.setHash('#sql='); } } refresh() { tfrpc.rpc.sync(); } render() { let self = this; if (!this.loading && this.whoami && this.loaded !== this.whoami) { this.loading = true; this.load().finally(function () { self.loading = false; }); } const k_tabs = { '📰': 'news', '📡': 'connections', '🔍': 'search', '👩‍💻': 'query', }; let tabs = html`
${Object.entries(k_tabs).map( ([k, v]) => html` ` )}
`; let contents = this.guest ? html`

⚠️🦀 Must be logged in to Tilde Friends to scuttle here. 🦀⚠️

` : !this.loaded || this.loading ? html`
🦀 Loading...
` : this.render_tab(); return html`
${tabs}
${contents}
`; } } customElements.define('tf-app', TfElement);