Files
nostrlib/eventstore/mmm/replace.go

102 lines
2.7 KiB
Go

package mmm
import (
"fmt"
"iter"
"runtime"
"fiatjaf.com/nostr"
"fiatjaf.com/nostr/eventstore/internal"
)
func (il *IndexingLayer) ReplaceEvent(evt nostr.Event) error {
il.mmmm.writeMutex.Lock()
defer il.mmmm.writeMutex.Unlock()
runtime.LockOSThread()
defer runtime.UnlockOSThread()
filter := nostr.Filter{Kinds: []nostr.Kind{evt.Kind}, Authors: []nostr.PubKey{evt.PubKey}}
if evt.Kind.IsAddressable() {
// when addressable, add the "d" tag to the filter
filter.Tags = nostr.TagMap{"d": []string{evt.Tags.GetD()}}
}
// prepare transactions
mmmtxn, err := il.mmmm.lmdbEnv.BeginTxn(nil, 0)
if err != nil {
return err
}
defer func() {
// defer abort but only if we haven't committed (we'll set it to nil after committing)
if mmmtxn != nil {
mmmtxn.Abort()
}
}()
mmmtxn.RawRead = true
iltxn, err := il.lmdbEnv.BeginTxn(nil, 0)
if err != nil {
return err
}
defer func() {
// defer abort but only if we haven't committed (we'll set it to nil after committing)
if iltxn != nil {
iltxn.Abort()
}
}()
iltxn.RawRead = true
// now we fetch the past events, whatever they are, delete them and then save the new
var results iter.Seq[nostr.Event] = func(yield func(nostr.Event) bool) {
err = il.query(iltxn, filter, 10 /* in theory limit could be just 1 and this should work */, yield)
}
if err != nil {
return fmt.Errorf("failed to query past events with %s: %w", filter, err)
}
var acquiredFreeRangeFromDelete *position
shouldStore := true
for previous := range results {
if internal.IsOlder(previous, evt) {
if pos, shouldPurge, err := il.delete(mmmtxn, iltxn, previous.ID); err != nil {
return fmt.Errorf("failed to delete event %s for replacing: %w", previous.ID, err)
} else if shouldPurge {
// purge
if err := mmmtxn.Del(il.mmmm.indexId, previous.ID[0:8], nil); err != nil {
return err
}
acquiredFreeRangeFromDelete = &pos
}
} else {
// there is a newer event already stored, so we won't store this
shouldStore = false
}
}
if shouldStore {
_, err := il.mmmm.storeOn(mmmtxn, iltxn, il, evt)
if err != nil {
return err
}
}
// commit in this order to minimize problematic inconsistencies
if err := mmmtxn.Commit(); err != nil {
return fmt.Errorf("can't commit mmmtxn: %w", err)
}
mmmtxn = nil
if err := iltxn.Commit(); err != nil {
return fmt.Errorf("can't commit iltxn: %w", err)
}
iltxn = nil
// finally merge in the new free range (in this order it makes more sense, the worst that can
// happen is that we lose this free range but we'll have it again on the next startup)
if acquiredFreeRangeFromDelete != nil {
il.mmmm.mergeNewFreeRange(*acquiredFreeRangeFromDelete)
}
return nil
}