This commit is contained in:
parent
8f7a9a1327
commit
7b151e1b17
@ -143,7 +143,7 @@ const Timeline = (props: TimelineProps) => {
|
|||||||
)}
|
)}
|
||||||
{mainFeed.map(eventElement)}
|
{mainFeed.map(eventElement)}
|
||||||
{(props.loadMore === undefined || props.loadMore === true) && (
|
{(props.loadMore === undefined || props.loadMore === true) && (
|
||||||
<LoadMore onLoadMore={feed.loadMore} shouldLoadMore={!feed.loading}>
|
<LoadMore onLoadMore={() => feed.loadMore()} shouldLoadMore={!feed.loading}>
|
||||||
<Skeleton width="100%" height="120px" margin="0 0 16px 0" />
|
<Skeleton width="100%" height="120px" margin="0 0 16px 0" />
|
||||||
<Skeleton width="100%" height="120px" margin="0 0 16px 0" />
|
<Skeleton width="100%" height="120px" margin="0 0 16px 0" />
|
||||||
<Skeleton width="100%" height="120px" margin="0 0 16px 0" />
|
<Skeleton width="100%" height="120px" margin="0 0 16px 0" />
|
||||||
|
@ -4,6 +4,6 @@ import { System, SystemSnapshot } from "System";
|
|||||||
export default function useSystemState() {
|
export default function useSystemState() {
|
||||||
return useSyncExternalStore<SystemSnapshot>(
|
return useSyncExternalStore<SystemSnapshot>(
|
||||||
cb => System.hook(cb),
|
cb => System.hook(cb),
|
||||||
() => System.getSnapshot()
|
() => System.snapshot()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
9
packages/app/src/Pages/Debug.tsx
Normal file
9
packages/app/src/Pages/Debug.tsx
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import SubDebug from "Element/SubDebug";
|
||||||
|
|
||||||
|
export default function DebugPage() {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<SubDebug />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
@ -1,6 +1,6 @@
|
|||||||
import { Connection } from "@snort/nostr";
|
import { Connection } from "@snort/nostr";
|
||||||
import { describe, expect } from "@jest/globals";
|
import { describe, expect } from "@jest/globals";
|
||||||
import { Query } from "./Query";
|
import { Query, QueryBase } from "./Query";
|
||||||
import { getRandomValues } from "crypto";
|
import { getRandomValues } from "crypto";
|
||||||
import { FlatNoteStore } from "./NoteCollection";
|
import { FlatNoteStore } from "./NoteCollection";
|
||||||
|
|
||||||
@ -44,24 +44,21 @@ describe("query", () => {
|
|||||||
q.eose(q.id, c3);
|
q.eose(q.id, c3);
|
||||||
expect(q.progress).toBe(1);
|
expect(q.progress).toBe(1);
|
||||||
|
|
||||||
const qs = new Query(
|
const qs = {
|
||||||
"test-1",
|
id: "test-1",
|
||||||
[
|
filters: [
|
||||||
{
|
{
|
||||||
kinds: [1],
|
kinds: [1],
|
||||||
authors: ["test-sub"],
|
authors: ["test-sub"],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
new FlatNoteStore()
|
} as QueryBase;
|
||||||
);
|
q.sendSubQueryToRelay(c1, qs);
|
||||||
q.subQueries.push(qs);
|
|
||||||
qs.sendToRelay(c1);
|
|
||||||
|
|
||||||
expect(q.progress).toBe(0.5);
|
expect(q.progress).toBe(3 / 4);
|
||||||
q.eose(qs.id, c1);
|
q.eose(qs.id, c1);
|
||||||
expect(q.progress).toBe(1);
|
expect(q.progress).toBe(1);
|
||||||
qs.sendToRelay(c2);
|
q.sendSubQueryToRelay(c2, qs);
|
||||||
// 1 + 0.5 (1/2 sent sub query)
|
expect(q.progress).toBe(4 / 5);
|
||||||
expect(q.progress).toBe(1.5 / 2);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -16,18 +16,21 @@ class QueryTrace {
|
|||||||
close?: number;
|
close?: number;
|
||||||
#wasForceClosed = false;
|
#wasForceClosed = false;
|
||||||
readonly #fnClose: (id: string) => void;
|
readonly #fnClose: (id: string) => void;
|
||||||
|
readonly #fnProgress: () => void;
|
||||||
|
|
||||||
constructor(sub: string, relay: string, connId: string, fnClose: (id: string) => void) {
|
constructor(sub: string, relay: string, connId: string, fnClose: (id: string) => void, fnProgress: () => void) {
|
||||||
this.id = uuid();
|
this.id = uuid();
|
||||||
this.subId = sub;
|
this.subId = sub;
|
||||||
this.relay = relay;
|
this.relay = relay;
|
||||||
this.connId = connId;
|
this.connId = connId;
|
||||||
this.start = unixNowMs();
|
this.start = unixNowMs();
|
||||||
this.#fnClose = fnClose;
|
this.#fnClose = fnClose;
|
||||||
|
this.#fnProgress = fnProgress;
|
||||||
}
|
}
|
||||||
|
|
||||||
sentToRelay() {
|
sentToRelay() {
|
||||||
this.sent = unixNowMs();
|
this.sent = unixNowMs();
|
||||||
|
this.#fnProgress();
|
||||||
}
|
}
|
||||||
|
|
||||||
gotEose() {
|
gotEose() {
|
||||||
@ -35,23 +38,28 @@ class QueryTrace {
|
|||||||
if (this.responseTime > 5_000) {
|
if (this.responseTime > 5_000) {
|
||||||
console.debug(`Slow query ${this.subId} on ${this.relay} took ${this.responseTime.toLocaleString()}ms`);
|
console.debug(`Slow query ${this.subId} on ${this.relay} took ${this.responseTime.toLocaleString()}ms`);
|
||||||
}
|
}
|
||||||
|
this.#fnProgress();
|
||||||
|
console.debug(`[EOSE][${this.subId}] ${this.relay}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
forceEose() {
|
forceEose() {
|
||||||
this.eose = unixNowMs();
|
this.eose = unixNowMs();
|
||||||
this.#wasForceClosed = true;
|
this.#wasForceClosed = true;
|
||||||
|
this.#fnProgress();
|
||||||
|
console.debug(`[F-EOSE][${this.subId}] ${this.relay}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
sendClose() {
|
sendClose() {
|
||||||
this.close = unixNowMs();
|
this.close = unixNowMs();
|
||||||
this.#fnClose(this.subId);
|
this.#fnClose(this.subId);
|
||||||
|
this.#fnProgress();
|
||||||
}
|
}
|
||||||
|
|
||||||
log() {
|
log() {
|
||||||
console.debug(
|
console.debug(
|
||||||
`QT:${this.id}, ${this.relay}, ${this.subId}, finished=${
|
`QT:${this.id}, ${this.subId}, finished=${
|
||||||
this.finished
|
this.finished
|
||||||
}, queued=${this.queued.toLocaleString()}ms, runtime=${this.runtime?.toLocaleString()}ms`
|
}, queued=${this.queued.toLocaleString()}ms, runtime=${this.runtime?.toLocaleString()}ms, ${this.relay}`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -59,7 +67,7 @@ class QueryTrace {
|
|||||||
* Time spent in queue
|
* Time spent in queue
|
||||||
*/
|
*/
|
||||||
get queued() {
|
get queued() {
|
||||||
return (this.sent === undefined ? unixNowMs() : this.sent) - this.start;
|
return (this.sent === undefined ? unixNowMs() : this.#wasForceClosed ? unwrap(this.eose) : this.sent) - this.start;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -84,10 +92,7 @@ class QueryTrace {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
export interface QueryBase {
|
||||||
* Active or queued query on the system
|
|
||||||
*/
|
|
||||||
export class Query {
|
|
||||||
/**
|
/**
|
||||||
* Uniquie ID of this query
|
* Uniquie ID of this query
|
||||||
*/
|
*/
|
||||||
@ -99,9 +104,18 @@ export class Query {
|
|||||||
filters: Array<RawReqFilter>;
|
filters: Array<RawReqFilter>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sub-Queries which are connected to this subscription
|
* List of relays to send this query to
|
||||||
*/
|
*/
|
||||||
subQueries: Array<Query> = [];
|
relays?: Array<string>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Active or queued query on the system
|
||||||
|
*/
|
||||||
|
export class Query implements QueryBase {
|
||||||
|
id: string;
|
||||||
|
filters: Array<RawReqFilter>;
|
||||||
|
relays?: Array<string>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Which relays this query has already been executed on
|
* Which relays this query has already been executed on
|
||||||
@ -113,11 +127,6 @@ export class Query {
|
|||||||
*/
|
*/
|
||||||
leaveOpen = false;
|
leaveOpen = false;
|
||||||
|
|
||||||
/**
|
|
||||||
* List of relays to send this query to
|
|
||||||
*/
|
|
||||||
relays: Array<string> = [];
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Time when this query can be removed
|
* Time when this query can be removed
|
||||||
*/
|
*/
|
||||||
@ -133,6 +142,8 @@ export class Query {
|
|||||||
*/
|
*/
|
||||||
#feed: NoteStore;
|
#feed: NoteStore;
|
||||||
|
|
||||||
|
subQueryCounter = 0;
|
||||||
|
|
||||||
constructor(id: string, filters: Array<RawReqFilter>, feed: NoteStore) {
|
constructor(id: string, filters: Array<RawReqFilter>, feed: NoteStore) {
|
||||||
this.id = id;
|
this.id = id;
|
||||||
this.filters = filters;
|
this.filters = filters;
|
||||||
@ -165,35 +176,36 @@ export class Query {
|
|||||||
}
|
}
|
||||||
|
|
||||||
sendToRelay(c: Connection) {
|
sendToRelay(c: Connection) {
|
||||||
if (this.relays.length > 0 && !this.relays.includes(c.Address)) {
|
if (!this.#canSendQuery(c, this)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (this.relays.length === 0 && c.Ephemeral) {
|
this.#sendQueryInternal(c, this);
|
||||||
console.debug("Cant send non-specific REQ to ephemeral connection");
|
}
|
||||||
|
|
||||||
|
sendSubQueryToRelay(c: Connection, subq: QueryBase) {
|
||||||
|
if (!this.#canSendQuery(c, subq)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (this.filters.some(a => a.search) && !c.SupportsNip(Nips.Search)) {
|
this.#sendQueryInternal(c, subq);
|
||||||
console.debug("Cant send REQ to non-search relay", c.Address);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const qt = new QueryTrace(this.id, c.Address, c.Id, x => c.CloseReq(x));
|
|
||||||
this.#tracing.push(qt);
|
|
||||||
c.QueueReq(["REQ", this.id, ...this.filters], () => qt.sentToRelay());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
connectionLost(c: Connection, active: Array<string>, pending: Array<string>) {
|
connectionLost(c: Connection, active: Array<string>, pending: Array<string>) {
|
||||||
const allQueriesLost = [...active, ...pending].filter(a => this.id === a || this.subQueries.some(b => b.id === a));
|
const allQueriesLost = [...active, ...pending].filter(a => this.id === a || this.#tracing.some(b => b.subId === a));
|
||||||
if (allQueriesLost.length > 0) {
|
if (allQueriesLost.length > 0) {
|
||||||
console.debug("Lost", allQueriesLost, c.Address, c.Id);
|
console.debug("Lost", allQueriesLost, c.Address, c.Id);
|
||||||
}
|
}
|
||||||
|
for (const qLost of allQueriesLost) {
|
||||||
|
const qt = this.#tracing.find(a => a.subId === qLost && a.connId == c.Id);
|
||||||
|
qt?.forceEose();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
sendClose() {
|
sendClose() {
|
||||||
for (const qt of this.#tracing) {
|
for (const qt of this.#tracing) {
|
||||||
qt.sendClose();
|
qt.sendClose();
|
||||||
}
|
}
|
||||||
for (const sq of this.subQueries) {
|
for (const qt of this.#tracing) {
|
||||||
sq.sendClose();
|
qt.sendClose();
|
||||||
}
|
}
|
||||||
this.cleanup();
|
this.cleanup();
|
||||||
}
|
}
|
||||||
@ -202,15 +214,9 @@ export class Query {
|
|||||||
const qt = this.#tracing.find(a => a.subId === sub && a.connId === conn.Id);
|
const qt = this.#tracing.find(a => a.subId === sub && a.connId === conn.Id);
|
||||||
qt?.gotEose();
|
qt?.gotEose();
|
||||||
if (sub === this.id) {
|
if (sub === this.id) {
|
||||||
console.debug(`[EOSE][${sub}] ${conn.Address}`);
|
|
||||||
if (!this.leaveOpen) {
|
if (!this.leaveOpen) {
|
||||||
qt?.sendClose();
|
qt?.sendClose();
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
const subQ = this.subQueries.find(a => a.id === sub);
|
|
||||||
if (subQ) {
|
|
||||||
subQ.eose(sub, conn);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -218,19 +224,19 @@ export class Query {
|
|||||||
* Get the progress to EOSE, can be used to determine when we should load more content
|
* Get the progress to EOSE, can be used to determine when we should load more content
|
||||||
*/
|
*/
|
||||||
get progress() {
|
get progress() {
|
||||||
let thisProgress = this.#tracing.reduce((acc, v) => (acc += v.finished ? 1 : 0), 0) / this.#tracing.length;
|
const thisProgress = this.#tracing.reduce((acc, v) => (acc += v.finished ? 1 : 0), 0) / this.#tracing.length;
|
||||||
if (isNaN(thisProgress)) {
|
if (isNaN(thisProgress)) {
|
||||||
thisProgress = 0;
|
return 0;
|
||||||
}
|
|
||||||
if (this.subQueries.length === 0) {
|
|
||||||
return thisProgress;
|
|
||||||
}
|
}
|
||||||
|
return thisProgress;
|
||||||
|
}
|
||||||
|
|
||||||
let totalProgress = thisProgress;
|
#onProgress() {
|
||||||
for (const sq of this.subQueries) {
|
const isFinished = this.progress === 1;
|
||||||
totalProgress += sq.progress;
|
if (this.feed.loading !== isFinished) {
|
||||||
|
console.debug(`[QT] ${this.id}, loading=${this.feed.loading}, progress=${this.progress}`);
|
||||||
|
this.feed.loading = isFinished;
|
||||||
}
|
}
|
||||||
return totalProgress / (this.subQueries.length + 1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#stopCheckTraces() {
|
#stopCheckTraces() {
|
||||||
@ -243,11 +249,37 @@ export class Query {
|
|||||||
this.#stopCheckTraces();
|
this.#stopCheckTraces();
|
||||||
this.#checkTrace = setInterval(() => {
|
this.#checkTrace = setInterval(() => {
|
||||||
for (const v of this.#tracing) {
|
for (const v of this.#tracing) {
|
||||||
//v.log();
|
|
||||||
if (v.runtime > 5_000 && !v.finished) {
|
if (v.runtime > 5_000 && !v.finished) {
|
||||||
v.forceEose();
|
v.forceEose();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, 2_000);
|
}, 500);
|
||||||
|
}
|
||||||
|
|
||||||
|
#canSendQuery(c: Connection, q: QueryBase) {
|
||||||
|
if (q.relays && !q.relays.includes(c.Address)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if ((q.relays?.length ?? 0) === 0 && c.Ephemeral) {
|
||||||
|
console.debug("Cant send non-specific REQ to ephemeral connection");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (q.filters.some(a => a.search) && !c.SupportsNip(Nips.Search)) {
|
||||||
|
console.debug("Cant send REQ to non-search relay", c.Address);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
#sendQueryInternal(c: Connection, q: QueryBase) {
|
||||||
|
const qt = new QueryTrace(
|
||||||
|
q.id,
|
||||||
|
c.Address,
|
||||||
|
c.Id,
|
||||||
|
x => c.CloseReq(x),
|
||||||
|
() => this.#onProgress()
|
||||||
|
);
|
||||||
|
this.#tracing.push(qt);
|
||||||
|
c.QueueReq(["REQ", q.id, ...q.filters], () => qt.sentToRelay());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -11,8 +11,9 @@ import {
|
|||||||
ReplaceableNoteStore,
|
ReplaceableNoteStore,
|
||||||
} from "./NoteCollection";
|
} from "./NoteCollection";
|
||||||
import { diffFilters } from "./RequestSplitter";
|
import { diffFilters } from "./RequestSplitter";
|
||||||
import { Query } from "./Query";
|
import { Query, QueryBase } from "./Query";
|
||||||
import { splitAllByWriteRelays } from "./GossipModel";
|
import { splitAllByWriteRelays } from "./GossipModel";
|
||||||
|
import ExternalStore from "ExternalStore";
|
||||||
|
|
||||||
export {
|
export {
|
||||||
NoteStore,
|
NoteStore,
|
||||||
@ -40,7 +41,7 @@ export type HookSystemSnapshot = () => void;
|
|||||||
/**
|
/**
|
||||||
* Manages nostr content retrieval system
|
* Manages nostr content retrieval system
|
||||||
*/
|
*/
|
||||||
export class NostrSystem {
|
export class NostrSystem extends ExternalStore<SystemSnapshot> {
|
||||||
/**
|
/**
|
||||||
* All currently connected websockets
|
* All currently connected websockets
|
||||||
*/
|
*/
|
||||||
@ -56,33 +57,12 @@ export class NostrSystem {
|
|||||||
*/
|
*/
|
||||||
HandleAuth?: AuthHandler;
|
HandleAuth?: AuthHandler;
|
||||||
|
|
||||||
/**
|
|
||||||
* State change hooks
|
|
||||||
*/
|
|
||||||
#stateHooks: Array<HookSystemSnapshot> = [];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Current snapshot of the system
|
|
||||||
*/
|
|
||||||
#snapshot: Readonly<SystemSnapshot> = { queries: [] };
|
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
|
super();
|
||||||
this.Sockets = new Map();
|
this.Sockets = new Map();
|
||||||
this.#cleanup();
|
this.#cleanup();
|
||||||
}
|
}
|
||||||
|
|
||||||
hook(cb: HookSystemSnapshot): HookSystemSnapshotRelease {
|
|
||||||
this.#stateHooks.push(cb);
|
|
||||||
return () => {
|
|
||||||
const idx = this.#stateHooks.findIndex(a => a === cb);
|
|
||||||
this.#stateHooks.splice(idx, 1);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
getSnapshot(): Readonly<SystemSnapshot> {
|
|
||||||
return this.#snapshot;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Connect to a NOSTR relay if not already connected
|
* Connect to a NOSTR relay if not already connected
|
||||||
*/
|
*/
|
||||||
@ -210,19 +190,20 @@ export class NostrSystem {
|
|||||||
|
|
||||||
const diff = diffFilters(q.filters, filters);
|
const diff = diffFilters(q.filters, filters);
|
||||||
if (!diff.changed && !req.options?.skipDiff) {
|
if (!diff.changed && !req.options?.skipDiff) {
|
||||||
this.#changed();
|
this.notifyChange();
|
||||||
return unwrap(q.feed) as Readonly<T>;
|
return unwrap(q.feed) as Readonly<T>;
|
||||||
} else {
|
} else {
|
||||||
const splitFilters = splitAllByWriteRelays(filters);
|
const splitFilters = splitAllByWriteRelays(filters);
|
||||||
for (const sf of splitFilters) {
|
for (const sf of splitFilters) {
|
||||||
const subQ = new Query(`${q.id}-${q.subQueries.length + 1}`, sf.filters, q.feed);
|
const subQ = {
|
||||||
subQ.relays = sf.relay ? [sf.relay] : [];
|
id: `${q.id}-${q.subQueryCounter++}`,
|
||||||
q.subQueries.push(subQ);
|
filters: sf.filters,
|
||||||
this.SendQuery(subQ);
|
relays: sf.relay ? [sf.relay] : [],
|
||||||
|
} as QueryBase;
|
||||||
|
this.SendSubQuery(q, subQ);
|
||||||
}
|
}
|
||||||
q.filters = filters;
|
q.filters = filters;
|
||||||
q.feed.loading = true;
|
this.notifyChange();
|
||||||
this.#changed();
|
|
||||||
return q.feed as Readonly<T>;
|
return q.feed as Readonly<T>;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -246,15 +227,17 @@ export class NostrSystem {
|
|||||||
const splitFilters = splitAllByWriteRelays(filters);
|
const splitFilters = splitAllByWriteRelays(filters);
|
||||||
if (splitFilters.length > 1) {
|
if (splitFilters.length > 1) {
|
||||||
for (const sf of splitFilters) {
|
for (const sf of splitFilters) {
|
||||||
const subQ = new Query(`${q.id}-${q.subQueries.length + 1}`, sf.filters, q.feed);
|
const subQ = {
|
||||||
subQ.relays = sf.relay ? [sf.relay] : [];
|
id: `${q.id}-${q.subQueryCounter++}`,
|
||||||
q.subQueries.push(subQ);
|
filters: sf.filters,
|
||||||
this.SendQuery(subQ);
|
relays: sf.relay ? [sf.relay] : [],
|
||||||
|
} as QueryBase;
|
||||||
|
this.SendSubQuery(q, subQ);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
this.SendQuery(q);
|
this.SendQuery(q);
|
||||||
}
|
}
|
||||||
this.#changed();
|
this.notifyChange();
|
||||||
return store;
|
return store;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -266,7 +249,7 @@ export class NostrSystem {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async SendQuery(q: Query) {
|
async SendQuery(q: Query) {
|
||||||
if (q.relays.length > 0) {
|
if (q.relays && q.relays.length > 0) {
|
||||||
for (const r of q.relays) {
|
for (const r of q.relays) {
|
||||||
const s = this.Sockets.get(r);
|
const s = this.Sockets.get(r);
|
||||||
if (s) {
|
if (s) {
|
||||||
@ -289,6 +272,30 @@ export class NostrSystem {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async SendSubQuery(q: Query, subQ: QueryBase) {
|
||||||
|
if (subQ.relays && subQ.relays.length > 0) {
|
||||||
|
for (const r of subQ.relays) {
|
||||||
|
const s = this.Sockets.get(r);
|
||||||
|
if (s) {
|
||||||
|
q.sendSubQueryToRelay(s, subQ);
|
||||||
|
} else {
|
||||||
|
const nc = await this.ConnectEphemeralRelay(r);
|
||||||
|
if (nc) {
|
||||||
|
q.sendSubQueryToRelay(nc, subQ);
|
||||||
|
} else {
|
||||||
|
console.warn("Failed to connect to new relay for:", r, subQ);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (const [, s] of this.Sockets) {
|
||||||
|
if (!s.Ephemeral) {
|
||||||
|
q.sendSubQueryToRelay(s, subQ);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Send events to writable relays
|
* Send events to writable relays
|
||||||
*/
|
*/
|
||||||
@ -316,20 +323,17 @@ export class NostrSystem {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
#changed() {
|
takeSnapshot(): SystemSnapshot {
|
||||||
this.#snapshot = Object.freeze({
|
return {
|
||||||
queries: [...this.Queries.values()].map(a => {
|
queries: [...this.Queries.values()].map(a => {
|
||||||
return {
|
return {
|
||||||
id: a.id,
|
id: a.id,
|
||||||
filters: a.filters,
|
filters: a.filters,
|
||||||
closing: a.closing,
|
closing: a.closing,
|
||||||
subFilters: a.subQueries.map(a => a.filters).flat(),
|
subFilters: [],
|
||||||
};
|
};
|
||||||
}),
|
}),
|
||||||
});
|
};
|
||||||
for (const h of this.#stateHooks) {
|
|
||||||
h();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#cleanup() {
|
#cleanup() {
|
||||||
@ -343,7 +347,7 @@ export class NostrSystem {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (changed) {
|
if (changed) {
|
||||||
this.#changed();
|
this.notifyChange();
|
||||||
}
|
}
|
||||||
setTimeout(() => this.#cleanup(), 1_000);
|
setTimeout(() => this.#cleanup(), 1_000);
|
||||||
}
|
}
|
||||||
|
@ -31,6 +31,7 @@ import NostrLinkHandler from "Pages/NostrLinkHandler";
|
|||||||
import Thread from "Element/Thread";
|
import Thread from "Element/Thread";
|
||||||
import { SubscribeRoutes } from "Pages/subscribe";
|
import { SubscribeRoutes } from "Pages/subscribe";
|
||||||
import ZapPoolPage from "Pages/ZapPool";
|
import ZapPoolPage from "Pages/ZapPool";
|
||||||
|
import DebugPage from "Pages/Debug";
|
||||||
|
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
window.__webpack_nonce__ = "ZmlhdGphZiBzYWlkIHNub3J0LnNvY2lhbCBpcyBwcmV0dHkgZ29vZCwgd2UgbWFkZSBpdCE=";
|
window.__webpack_nonce__ = "ZmlhdGphZiBzYWlkIHNub3J0LnNvY2lhbCBpcyBwcmV0dHkgZ29vZCwgd2UgbWFkZSBpdCE=";
|
||||||
@ -99,6 +100,10 @@ export const router = createBrowserRouter([
|
|||||||
...NewUserRoutes,
|
...NewUserRoutes,
|
||||||
...WalletRoutes,
|
...WalletRoutes,
|
||||||
...SubscribeRoutes,
|
...SubscribeRoutes,
|
||||||
|
{
|
||||||
|
path: "/debug",
|
||||||
|
element: <DebugPage />,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: "/*",
|
path: "/*",
|
||||||
element: <NostrLinkHandler />,
|
element: <NostrLinkHandler />,
|
||||||
|
Loading…
Reference in New Issue
Block a user