Local first
NDK allows for local-first software.
This mode of operation depends on a few key things:
- A cache adapter must be present
- Events
event:publish-failed
should be handled by the application
A few considerations:
- Failed events are automatically retried by NDK
- Handling of failed events is up to the application; handling here exclusively refers to notifying the user and updating the UI accordingly
Blocked publishing
The default behavior when publishing an event will make event.publish()
block until the event has been published or a failure has ocurred.
const event = new NDKEvent(ndk, { kind: 1, content: 'Blocking event' });
const publishedToRelays = await event.publish();
console.log(publishedToRelays); // relays where the event has published to
Optimistic publishing
If you want to publish an event without waiting for it to be published, you can use the event.publish()
method.
const event = new NDKEvent(ndk, { kind: 1, content: 'Optimistic event' });
event.publish();
When using a cache adapter that supports unpublished event tracking (like NDKCacheAdapterDexie
), the event will be first written to the cache and then published to relays. When a minimal amount of relays have successfully received the event, the event will be removed from the cache.
With this technique you can fire and forget event publshing.
Handling persistent failures
When an event fails to publish, you can handle the failure by listening to the event.failed
event.
You should handle this event to notify the user that the event has failed to publish and update the UI accordingly.
// application-wide
function handlePublishingFailures(event: NDKEvent, error: NDKPublishError) {
console.log(`Event ${event.id} failed to publish`, { publishedToRelays: error.publishedToRelays });
}
ndk.on("event:publish-failed", handlePublishingFailures);
const event = new NDKEvent(ndk, { kind: 1, content: 'Failing event' });
event.publish();
Querying cached failed events
Cache adapters with support for failed publish tracking can be queried via the getUnpublishedEvents
interface.
const failedEvents = ndk.cachedAdapter.getUnpublishedEvents()
console.log(failedEvents.length + " events have not published before; trying now");
failedEvents.forEach((event) => event.publish());
When an event successfully publishes, the event will emit published
.
Handling retries
When booting up your application, NDK will automatically reattempt to publish any events that have failed to publish in the past.