forked from coracle/flotilla
229 lines
6.3 KiB
Svelte
229 lines
6.3 KiB
Svelte
<script lang="ts">
|
|
import "@src/app.css"
|
|
import {onMount} from "svelte"
|
|
import type {SvelteComponent} from "svelte"
|
|
import {get} from "svelte/store"
|
|
import {page} from "$app/stores"
|
|
import {goto} from "$app/navigation"
|
|
import {browser} from "$app/environment"
|
|
import {sleep, take, sortBy, ago, now, HOUR} from "@welshman/lib"
|
|
import type {TrustedEvent} from "@welshman/util"
|
|
import {
|
|
PROFILE,
|
|
REACTION,
|
|
ZAP_RESPONSE,
|
|
FOLLOWS,
|
|
RELAYS,
|
|
INBOX_RELAYS,
|
|
WRAP,
|
|
getPubkeyTagValues,
|
|
getListTags,
|
|
} from "@welshman/util"
|
|
import {throttled} from "@welshman/store"
|
|
import {
|
|
relays,
|
|
handles,
|
|
loadRelay,
|
|
db,
|
|
initStorage,
|
|
repository,
|
|
session,
|
|
pubkey,
|
|
publishStatusData,
|
|
plaintext,
|
|
freshness,
|
|
storageAdapters,
|
|
tracker,
|
|
} from "@welshman/app"
|
|
import * as lib from "@welshman/lib"
|
|
import * as util from "@welshman/util"
|
|
import * as app from "@welshman/app"
|
|
import ModalBox from "@lib/components/ModalBox.svelte"
|
|
import Drawer from "@lib/components/Drawer.svelte"
|
|
import Toast from "@app/components/Toast.svelte"
|
|
import Landing from "@app/components/Landing.svelte"
|
|
import PrimaryNav from "@app/components/PrimaryNav.svelte"
|
|
import {modals, clearModal} from "@app/modal"
|
|
import {theme} from "@app/theme"
|
|
import {INDEXER_RELAYS} from "@app/state"
|
|
import {loadUserData} from "@app/commands"
|
|
import * as state from "@app/state"
|
|
|
|
const subs: any[] = []
|
|
|
|
const onDialogClose = (e: Event) => {
|
|
if (!$session) {
|
|
e.preventDefault()
|
|
|
|
// Prevent default doesn't always work, just re-open when needed
|
|
setTimeout(() => dialog.showModal())
|
|
} else if (modal) {
|
|
clearModal()
|
|
}
|
|
}
|
|
|
|
let ready: Promise<unknown>
|
|
let dialog: HTMLDialogElement
|
|
let drawer: SvelteComponent
|
|
let modal: any
|
|
|
|
onMount(() => {
|
|
Object.assign(window, {get, ...lib, ...util, ...app, ...state})
|
|
|
|
const getScoreEvent = () => {
|
|
const ALWAYS_KEEP = Infinity
|
|
const NEVER_KEEP = 0
|
|
|
|
const reactionKinds = [REACTION, ZAP_RESPONSE]
|
|
const metaKinds = [PROFILE, FOLLOWS, RELAYS, INBOX_RELAYS]
|
|
const $sessionKeys = new Set(Object.keys(app.sessions.get()))
|
|
const $userFollows = new Set(getPubkeyTagValues(getListTags(get(app.userFollows))))
|
|
const $maxWot = get(app.maxWot)
|
|
|
|
return (e: TrustedEvent) => {
|
|
const isFollowing = $userFollows.has(e.pubkey)
|
|
|
|
// No need to keep a record of everyone who follows the current user
|
|
if (e.kind === FOLLOWS && !isFollowing) return NEVER_KEEP
|
|
|
|
// Always keep stuff by or tagging a signed in user
|
|
if ($sessionKeys.has(e.pubkey)) return ALWAYS_KEEP
|
|
if (e.tags.some(t => $sessionKeys.has(t[1]))) return ALWAYS_KEEP
|
|
|
|
// Get rid of irrelevant messages, reactions, and likes
|
|
if (e.wrap || e.kind === 4 || e.kind === WRAP) return NEVER_KEEP
|
|
if (reactionKinds.includes(e.kind)) return NEVER_KEEP
|
|
|
|
// If the user follows this person, use max wot score
|
|
let score = isFollowing ? $maxWot : app.getUserWotScore(e.pubkey)
|
|
|
|
// Inflate the score for profiles/relays/follows to avoid redundant fetches
|
|
// Demote non-metadata type events, and introduce recency bias
|
|
score *= metaKinds.includes(e.kind) ? 2 : e.created_at / now()
|
|
|
|
return score
|
|
}
|
|
}
|
|
|
|
const migrateFreshness = (data: {key: string; value: number}[]) => {
|
|
const cutoff = ago(HOUR)
|
|
|
|
return data.filter(({value}) => value < cutoff)
|
|
}
|
|
|
|
const migratePlaintext = (data: {key: string; value: number}[]) => data.slice(0, 10_000)
|
|
|
|
const migrateEvents = (events: TrustedEvent[]) => {
|
|
if (events.length < 50_000) {
|
|
return events
|
|
}
|
|
|
|
const scoreEvent = getScoreEvent()
|
|
|
|
return take(
|
|
30_000,
|
|
sortBy(e => -scoreEvent(e), events),
|
|
)
|
|
}
|
|
|
|
if (!db) {
|
|
ready = initStorage("flotilla", 4, {
|
|
events: storageAdapters.fromRepository(repository, {throttle: 300, migrate: migrateEvents}),
|
|
relays: {keyPath: "url", store: throttled(1000, relays)},
|
|
handles: {keyPath: "nip05", store: throttled(1000, handles)},
|
|
publishStatus: storageAdapters.fromObjectStore(publishStatusData),
|
|
freshness: storageAdapters.fromObjectStore(freshness, {
|
|
throttle: 1000,
|
|
migrate: migrateFreshness,
|
|
}),
|
|
plaintext: storageAdapters.fromObjectStore(plaintext, {
|
|
throttle: 1000,
|
|
migrate: migratePlaintext,
|
|
}),
|
|
tracker: storageAdapters.fromTracker(tracker, {throttle: 1000}),
|
|
}).then(() => sleep(300))
|
|
|
|
repository.on("update", ({added}) => {
|
|
for (const event of added) {
|
|
state.ensureUnwrapped(event)
|
|
}
|
|
})
|
|
|
|
for (const url of INDEXER_RELAYS) {
|
|
loadRelay(url)
|
|
}
|
|
|
|
if ($pubkey) {
|
|
loadUserData($pubkey)
|
|
}
|
|
|
|
ready.then(async () => {
|
|
await sleep(1)
|
|
|
|
page.subscribe($page => {
|
|
modal = modals.get($page.url.hash.slice(1))
|
|
|
|
if (!$session && !modal) {
|
|
modal = {component: Landing}
|
|
}
|
|
|
|
if (modal) {
|
|
if (modal.options?.drawer) {
|
|
drawer.open()
|
|
} else {
|
|
dialog.showModal()
|
|
}
|
|
} else if ($session) {
|
|
drawer?.close()
|
|
dialog?.close()
|
|
}
|
|
})
|
|
})
|
|
}
|
|
|
|
return () => {
|
|
for (const unsub of subs) {
|
|
unsub()
|
|
}
|
|
}
|
|
})
|
|
</script>
|
|
|
|
{#await ready}
|
|
<div data-theme={$theme} />
|
|
{:then}
|
|
<div data-theme={$theme}>
|
|
<div class="flex h-screen overflow-hidden">
|
|
<PrimaryNav>
|
|
{#if $pubkey}
|
|
<slot />
|
|
{/if}
|
|
</PrimaryNav>
|
|
</div>
|
|
<dialog
|
|
bind:this={dialog}
|
|
on:cancel={onDialogClose}
|
|
class="modal modal-bottom !z-modal sm:modal-middle">
|
|
{#if modal && !modal.options?.drawer}
|
|
{#key modal}
|
|
<ModalBox {...modal} />
|
|
{/key}
|
|
<Toast />
|
|
{/if}
|
|
{#if $session}
|
|
<form method="dialog" class="modal-backdrop">
|
|
<button />
|
|
</form>
|
|
{/if}
|
|
</dialog>
|
|
<Drawer bind:this={drawer}>
|
|
{#if modal && modal.options?.drawer}
|
|
{#key modal}
|
|
<svelte:component this={modal.component} {...modal.props} />
|
|
{/key}
|
|
{/if}
|
|
</Drawer>
|
|
<Toast />
|
|
</div>
|
|
{/await}
|