You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
217 lines
6.0 KiB
217 lines
6.0 KiB
//go:build !(js && wasm) |
|
|
|
package database |
|
|
|
import ( |
|
"bytes" |
|
|
|
"github.com/dgraph-io/badger/v4" |
|
"lol.mleku.dev/chk" |
|
"lol.mleku.dev/log" |
|
"next.orly.dev/pkg/database/indexes" |
|
"next.orly.dev/pkg/database/indexes/types" |
|
"git.mleku.dev/mleku/nostr/encoders/event" |
|
) |
|
|
|
// FetchEventsBySerials fetches multiple events by their serials in a single database transaction. |
|
// Returns a map of serial uint64 value to event, only including successfully fetched events. |
|
// |
|
// This function tries multiple storage formats in order: |
|
// 1. cmp (compact format with serial references) - newest, most space-efficient |
|
// 2. sev (small event inline) - legacy Reiser4 optimization |
|
// 3. evt (traditional separate storage) - legacy fallback |
|
func (d *D) FetchEventsBySerials(serials []*types.Uint40) (events map[uint64]*event.E, err error) { |
|
// Pre-allocate map with estimated capacity to reduce reallocations |
|
events = make(map[uint64]*event.E, len(serials)) |
|
|
|
if len(serials) == 0 { |
|
return events, nil |
|
} |
|
|
|
// Create resolver for compact event decoding |
|
resolver := NewDatabaseSerialResolver(d, d.serialCache) |
|
|
|
if err = d.View( |
|
func(txn *badger.Txn) (err error) { |
|
for _, ser := range serials { |
|
var ev *event.E |
|
serialVal := ser.Get() |
|
|
|
// Try cmp (compact format) first - most efficient |
|
ev, err = d.fetchCompactEvent(txn, ser, resolver) |
|
if err == nil && ev != nil { |
|
events[serialVal] = ev |
|
continue |
|
} |
|
err = nil // Reset error, try legacy formats |
|
|
|
// Try sev (small event inline) prefix - legacy Reiser4 optimization |
|
ev, err = d.fetchSmallEvent(txn, ser) |
|
if err == nil && ev != nil { |
|
events[serialVal] = ev |
|
continue |
|
} |
|
err = nil // Reset error, try evt |
|
|
|
// Not found in sev table, try evt (traditional) prefix |
|
ev, err = d.fetchLegacyEvent(txn, ser) |
|
if err == nil && ev != nil { |
|
events[serialVal] = ev |
|
continue |
|
} |
|
err = nil // Reset error, event not found |
|
} |
|
return nil |
|
}, |
|
); err != nil { |
|
return |
|
} |
|
|
|
return events, nil |
|
} |
|
|
|
// fetchCompactEvent tries to fetch an event from the compact format (cmp prefix). |
|
func (d *D) fetchCompactEvent(txn *badger.Txn, ser *types.Uint40, resolver SerialResolver) (ev *event.E, err error) { |
|
// Build cmp key |
|
keyBuf := new(bytes.Buffer) |
|
if err = indexes.CompactEventEnc(ser).MarshalWrite(keyBuf); chk.E(err) { |
|
return nil, err |
|
} |
|
|
|
item, err := txn.Get(keyBuf.Bytes()) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
var compactData []byte |
|
if compactData, err = item.ValueCopy(nil); chk.E(err) { |
|
return nil, err |
|
} |
|
|
|
// Need to get the event ID from SerialEventId table |
|
eventId, err := d.GetEventIdBySerial(ser) |
|
if err != nil { |
|
log.D.F("fetchCompactEvent: failed to get event ID for serial %d: %v", ser.Get(), err) |
|
return nil, err |
|
} |
|
|
|
// Unmarshal compact event |
|
ev, err = UnmarshalCompactEvent(compactData, eventId, resolver) |
|
if err != nil { |
|
log.D.F("fetchCompactEvent: failed to unmarshal compact event for serial %d: %v", ser.Get(), err) |
|
return nil, err |
|
} |
|
|
|
return ev, nil |
|
} |
|
|
|
// fetchSmallEvent tries to fetch an event from the small event inline format (sev prefix). |
|
func (d *D) fetchSmallEvent(txn *badger.Txn, ser *types.Uint40) (ev *event.E, err error) { |
|
smallBuf := new(bytes.Buffer) |
|
if err = indexes.SmallEventEnc(ser).MarshalWrite(smallBuf); chk.E(err) { |
|
return nil, err |
|
} |
|
|
|
// Iterate with prefix to find the small event key |
|
opts := badger.DefaultIteratorOptions |
|
opts.Prefix = smallBuf.Bytes() |
|
opts.PrefetchValues = true |
|
opts.PrefetchSize = 1 |
|
it := txn.NewIterator(opts) |
|
defer it.Close() |
|
|
|
it.Rewind() |
|
if !it.Valid() { |
|
return nil, nil // Not found |
|
} |
|
|
|
// Found in sev table - extract inline data |
|
key := it.Item().Key() |
|
// Key format: sev|serial|size_uint16|event_data |
|
if len(key) <= 8+2 { // prefix(3) + serial(5) + size(2) = 10 bytes minimum |
|
return nil, nil |
|
} |
|
|
|
sizeIdx := 8 // After sev(3) + serial(5) |
|
// Read uint16 big-endian size |
|
size := int(key[sizeIdx])<<8 | int(key[sizeIdx+1]) |
|
dataStart := sizeIdx + 2 |
|
|
|
if len(key) < dataStart+size { |
|
return nil, nil |
|
} |
|
|
|
eventData := key[dataStart : dataStart+size] |
|
|
|
// Check if this is compact format (starts with version byte 1) |
|
if len(eventData) > 0 && eventData[0] == CompactFormatVersion { |
|
// This is compact format stored in sev - need to decode with resolver |
|
resolver := NewDatabaseSerialResolver(d, d.serialCache) |
|
eventId, idErr := d.GetEventIdBySerial(ser) |
|
if idErr != nil { |
|
// Fall back to legacy unmarshal |
|
ev = new(event.E) |
|
if err = ev.UnmarshalBinary(bytes.NewBuffer(eventData)); err != nil { |
|
return nil, err |
|
} |
|
return ev, nil |
|
} |
|
return UnmarshalCompactEvent(eventData, eventId, resolver) |
|
} |
|
|
|
// Legacy binary format |
|
ev = new(event.E) |
|
if err = ev.UnmarshalBinary(bytes.NewBuffer(eventData)); err != nil { |
|
return nil, err |
|
} |
|
|
|
return ev, nil |
|
} |
|
|
|
// fetchLegacyEvent tries to fetch an event from the legacy format (evt prefix). |
|
func (d *D) fetchLegacyEvent(txn *badger.Txn, ser *types.Uint40) (ev *event.E, err error) { |
|
buf := new(bytes.Buffer) |
|
if err = indexes.EventEnc(ser).MarshalWrite(buf); chk.E(err) { |
|
return nil, err |
|
} |
|
|
|
item, err := txn.Get(buf.Bytes()) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
var v []byte |
|
if v, err = item.ValueCopy(nil); chk.E(err) { |
|
return nil, err |
|
} |
|
|
|
// Check if we have valid data before attempting to unmarshal |
|
if len(v) < 32+32+1+2+1+1+64 { // ID + Pubkey + min varint fields + Sig |
|
return nil, nil |
|
} |
|
|
|
// Check if this is compact format (starts with version byte 1) |
|
if len(v) > 0 && v[0] == CompactFormatVersion { |
|
// This is compact format stored in evt - need to decode with resolver |
|
resolver := NewDatabaseSerialResolver(d, d.serialCache) |
|
eventId, idErr := d.GetEventIdBySerial(ser) |
|
if idErr != nil { |
|
// Fall back to legacy unmarshal |
|
ev = new(event.E) |
|
if err = ev.UnmarshalBinary(bytes.NewBuffer(v)); err != nil { |
|
return nil, err |
|
} |
|
return ev, nil |
|
} |
|
return UnmarshalCompactEvent(v, eventId, resolver) |
|
} |
|
|
|
// Legacy binary format |
|
ev = new(event.E) |
|
if err = ev.UnmarshalBinary(bytes.NewBuffer(v)); err != nil { |
|
return nil, err |
|
} |
|
|
|
return ev, nil |
|
} |
|
|
|
|