Patch for concurrent iterator & others (onto v1.11.6) #386
@ -89,6 +89,11 @@ func (db *nofreezedb) Ancient(kind string, number uint64) ([]byte, error) {
|
|||||||
return nil, errNotSupported
|
return nil, errNotSupported
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ReadAncients returns an error as we don't have a backing chain freezer.
|
||||||
|
func (db *nofreezedb) ReadAncients(kind string, start, max, maxByteSize uint64) ([][]byte, error) {
|
||||||
|
return nil, errNotSupported
|
||||||
|
}
|
||||||
|
|
||||||
// Ancients returns an error as we don't have a backing chain freezer.
|
// Ancients returns an error as we don't have a backing chain freezer.
|
||||||
func (db *nofreezedb) Ancients() (uint64, error) {
|
func (db *nofreezedb) Ancients() (uint64, error) {
|
||||||
return 0, errNotSupported
|
return 0, errNotSupported
|
||||||
|
@ -180,6 +180,18 @@ func (f *freezer) Ancient(kind string, number uint64) ([]byte, error) {
|
|||||||
return nil, errUnknownTable
|
return nil, errUnknownTable
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ReadAncients retrieves multiple items in sequence, starting from the index 'start'.
|
||||||
|
// It will return
|
||||||
|
// - at most 'max' items,
|
||||||
|
// - at least 1 item (even if exceeding the maxByteSize), but will otherwise
|
||||||
|
// return as many items as fit into maxByteSize.
|
||||||
|
func (f *freezer) ReadAncients(kind string, start, count, maxBytes uint64) ([][]byte, error) {
|
||||||
|
if table := f.tables[kind]; table != nil {
|
||||||
|
return table.RetrieveItems(start, count, maxBytes)
|
||||||
|
}
|
||||||
|
return nil, errUnknownTable
|
||||||
|
}
|
||||||
|
|
||||||
// Ancients returns the length of the frozen items.
|
// Ancients returns the length of the frozen items.
|
||||||
func (f *freezer) Ancients() (uint64, error) {
|
func (f *freezer) Ancients() (uint64, error) {
|
||||||
return atomic.LoadUint64(&f.frozen), nil
|
return atomic.LoadUint64(&f.frozen), nil
|
||||||
|
@ -70,6 +70,19 @@ func (i *indexEntry) marshallBinary() []byte {
|
|||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// bounds returns the start- and end- offsets, and the file number of where to
|
||||||
|
// read there data item marked by the two index entries. The two entries are
|
||||||
|
// assumed to be sequential.
|
||||||
|
func (start *indexEntry) bounds(end *indexEntry) (startOffset, endOffset, fileId uint32) {
|
||||||
|
if start.filenum != end.filenum {
|
||||||
|
// If a piece of data 'crosses' a data-file,
|
||||||
|
// it's actually in one piece on the second data-file.
|
||||||
|
// We return a zero-indexEntry for the second file as start
|
||||||
|
return 0, end.offset, end.filenum
|
||||||
|
}
|
||||||
|
return start.offset, end.offset, end.filenum
|
||||||
|
}
|
||||||
|
|
||||||
// freezerTable represents a single chained data table within the freezer (e.g. blocks).
|
// freezerTable represents a single chained data table within the freezer (e.g. blocks).
|
||||||
// It consists of a data file (snappy encoded arbitrary data blobs) and an indexEntry
|
// It consists of a data file (snappy encoded arbitrary data blobs) and an indexEntry
|
||||||
// file (uncompressed 64 bit indices into the data file).
|
// file (uncompressed 64 bit indices into the data file).
|
||||||
@ -546,84 +559,183 @@ func (t *freezerTable) append(item uint64, encodedBlob []byte, wlock bool) (bool
|
|||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// getBounds returns the indexes for the item
|
// getIndices returns the index entries for the given from-item, covering 'count' items.
|
||||||
// returns start, end, filenumber and error
|
// N.B: The actual number of returned indices for N items will always be N+1 (unless an
|
||||||
func (t *freezerTable) getBounds(item uint64) (uint32, uint32, uint32, error) {
|
// error is returned).
|
||||||
buffer := make([]byte, indexEntrySize)
|
// OBS: This method assumes that the caller has already verified (and/or trimmed) the range
|
||||||
var startIdx, endIdx indexEntry
|
// so that the items are within bounds. If this method is used to read out of bounds,
|
||||||
// Read second index
|
// it will return error.
|
||||||
if _, err := t.index.ReadAt(buffer, int64((item+1)*indexEntrySize)); err != nil {
|
func (t *freezerTable) getIndices(from, count uint64) ([]*indexEntry, error) {
|
||||||
return 0, 0, 0, err
|
// Apply the table-offset
|
||||||
|
from = from - uint64(t.itemOffset)
|
||||||
|
// For reading N items, we need N+1 indices.
|
||||||
|
buffer := make([]byte, (count+1)*indexEntrySize)
|
||||||
|
if _, err := t.index.ReadAt(buffer, int64(from*indexEntrySize)); err != nil {
|
||||||
|
return nil, err
|
||||||
}
|
}
|
||||||
endIdx.unmarshalBinary(buffer)
|
var (
|
||||||
// Read first index (unless it's the very first item)
|
indices []*indexEntry
|
||||||
if item != 0 {
|
offset int
|
||||||
if _, err := t.index.ReadAt(buffer, int64(item*indexEntrySize)); err != nil {
|
)
|
||||||
return 0, 0, 0, err
|
for i := from; i <= from+count; i++ {
|
||||||
}
|
index := new(indexEntry)
|
||||||
startIdx.unmarshalBinary(buffer)
|
index.unmarshalBinary(buffer[offset:])
|
||||||
} else {
|
offset += indexEntrySize
|
||||||
|
indices = append(indices, index)
|
||||||
|
}
|
||||||
|
if from == 0 {
|
||||||
// Special case if we're reading the first item in the freezer. We assume that
|
// Special case if we're reading the first item in the freezer. We assume that
|
||||||
// the first item always start from zero(regarding the deletion, we
|
// the first item always start from zero(regarding the deletion, we
|
||||||
// only support deletion by files, so that the assumption is held).
|
// only support deletion by files, so that the assumption is held).
|
||||||
// This means we can use the first item metadata to carry information about
|
// This means we can use the first item metadata to carry information about
|
||||||
// the 'global' offset, for the deletion-case
|
// the 'global' offset, for the deletion-case
|
||||||
return 0, endIdx.offset, endIdx.filenum, nil
|
indices[0].offset = 0
|
||||||
|
indices[0].filenum = indices[1].filenum
|
||||||
}
|
}
|
||||||
if startIdx.filenum != endIdx.filenum {
|
return indices, nil
|
||||||
// If a piece of data 'crosses' a data-file,
|
|
||||||
// it's actually in one piece on the second data-file.
|
|
||||||
// We return a zero-indexEntry for the second file as start
|
|
||||||
return 0, endIdx.offset, endIdx.filenum, nil
|
|
||||||
}
|
|
||||||
return startIdx.offset, endIdx.offset, endIdx.filenum, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Retrieve looks up the data offset of an item with the given number and retrieves
|
// Retrieve looks up the data offset of an item with the given number and retrieves
|
||||||
// the raw binary blob from the data file.
|
// the raw binary blob from the data file.
|
||||||
func (t *freezerTable) Retrieve(item uint64) ([]byte, error) {
|
func (t *freezerTable) Retrieve(item uint64) ([]byte, error) {
|
||||||
blob, err := t.retrieve(item)
|
items, err := t.RetrieveItems(item, 1, 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if t.noCompression {
|
return items[0], nil
|
||||||
return blob, nil
|
|
||||||
}
|
|
||||||
return snappy.Decode(nil, blob)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// retrieve looks up the data offset of an item with the given number and retrieves
|
// RetrieveItems returns multiple items in sequence, starting from the index 'start'.
|
||||||
// the raw binary blob from the data file. OBS! This method does not decode
|
// It will return at most 'max' items, but will abort earlier to respect the
|
||||||
// compressed data.
|
// 'maxBytes' argument. However, if the 'maxBytes' is smaller than the size of one
|
||||||
func (t *freezerTable) retrieve(item uint64) ([]byte, error) {
|
// item, it _will_ return one element and possibly overflow the maxBytes.
|
||||||
|
func (t *freezerTable) RetrieveItems(start, count, maxBytes uint64) ([][]byte, error) {
|
||||||
|
// First we read the 'raw' data, which might be compressed.
|
||||||
|
diskData, sizes, err := t.retrieveItems(start, count, maxBytes)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var (
|
||||||
|
output = make([][]byte, 0, count)
|
||||||
|
offset int // offset for reading
|
||||||
|
outputSize int // size of uncompressed data
|
||||||
|
)
|
||||||
|
// Now slice up the data and decompress.
|
||||||
|
for i, diskSize := range sizes {
|
||||||
|
item := diskData[offset : offset+diskSize]
|
||||||
|
offset += diskSize
|
||||||
|
decompressedSize := diskSize
|
||||||
|
if !t.noCompression {
|
||||||
|
decompressedSize, _ = snappy.DecodedLen(item)
|
||||||
|
}
|
||||||
|
if i > 0 && uint64(outputSize+decompressedSize) > maxBytes {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if !t.noCompression {
|
||||||
|
data, err := snappy.Decode(nil, item)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
output = append(output, data)
|
||||||
|
} else {
|
||||||
|
output = append(output, item)
|
||||||
|
}
|
||||||
|
outputSize += decompressedSize
|
||||||
|
}
|
||||||
|
return output, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// retrieveItems reads up to 'count' items from the table. It reads at least
|
||||||
|
// one item, but otherwise avoids reading more than maxBytes bytes.
|
||||||
|
// It returns the (potentially compressed) data, and the sizes.
|
||||||
|
func (t *freezerTable) retrieveItems(start, count, maxBytes uint64) ([]byte, []int, error) {
|
||||||
t.lock.RLock()
|
t.lock.RLock()
|
||||||
defer t.lock.RUnlock()
|
defer t.lock.RUnlock()
|
||||||
// Ensure the table and the item is accessible
|
// Ensure the table and the item is accessible
|
||||||
if t.index == nil || t.head == nil {
|
if t.index == nil || t.head == nil {
|
||||||
return nil, errClosed
|
return nil, nil, errClosed
|
||||||
}
|
}
|
||||||
if atomic.LoadUint64(&t.items) <= item {
|
itemCount := atomic.LoadUint64(&t.items) // max number
|
||||||
return nil, errOutOfBounds
|
// Ensure the start is written, not deleted from the tail, and that the
|
||||||
|
// caller actually wants something
|
||||||
|
if itemCount <= start || uint64(t.itemOffset) > start || count == 0 {
|
||||||
|
return nil, nil, errOutOfBounds
|
||||||
}
|
}
|
||||||
// Ensure the item was not deleted from the tail either
|
if start+count > itemCount {
|
||||||
if uint64(t.itemOffset) > item {
|
count = itemCount - start
|
||||||
return nil, errOutOfBounds
|
|
||||||
}
|
}
|
||||||
startOffset, endOffset, filenum, err := t.getBounds(item - uint64(t.itemOffset))
|
var (
|
||||||
|
output = make([]byte, maxBytes) // Buffer to read data into
|
||||||
|
outputSize int // Used size of that buffer
|
||||||
|
)
|
||||||
|
// readData is a helper method to read a single data item from disk.
|
||||||
|
readData := func(fileId, start uint32, length int) error {
|
||||||
|
// In case a small limit is used, and the elements are large, may need to
|
||||||
|
// realloc the read-buffer when reading the first (and only) item.
|
||||||
|
if len(output) < length {
|
||||||
|
output = make([]byte, length)
|
||||||
|
}
|
||||||
|
dataFile, exist := t.files[fileId]
|
||||||
|
if !exist {
|
||||||
|
return fmt.Errorf("missing data file %d", fileId)
|
||||||
|
}
|
||||||
|
if _, err := dataFile.ReadAt(output[outputSize:outputSize+length], int64(start)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
outputSize += length
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
// Read all the indexes in one go
|
||||||
|
indices, err := t.getIndices(start, count)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, nil, err
|
||||||
}
|
}
|
||||||
dataFile, exist := t.files[filenum]
|
var (
|
||||||
if !exist {
|
sizes []int // The sizes for each element
|
||||||
return nil, fmt.Errorf("missing data file %d", filenum)
|
totalSize = 0 // The total size of all data read so far
|
||||||
|
readStart = indices[0].offset // Where, in the file, to start reading
|
||||||
|
unreadSize = 0 // The size of the as-yet-unread data
|
||||||
|
)
|
||||||
|
|
||||||
|
for i, firstIndex := range indices[:len(indices)-1] {
|
||||||
|
secondIndex := indices[i+1]
|
||||||
|
// Determine the size of the item.
|
||||||
|
offset1, offset2, _ := firstIndex.bounds(secondIndex)
|
||||||
|
size := int(offset2 - offset1)
|
||||||
|
// Crossing a file boundary?
|
||||||
|
if secondIndex.filenum != firstIndex.filenum {
|
||||||
|
// If we have unread data in the first file, we need to do that read now.
|
||||||
|
if unreadSize > 0 {
|
||||||
|
if err := readData(firstIndex.filenum, readStart, unreadSize); err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
unreadSize = 0
|
||||||
|
}
|
||||||
|
readStart = 0
|
||||||
|
}
|
||||||
|
if i > 0 && uint64(totalSize+size) > maxBytes {
|
||||||
|
// About to break out due to byte limit being exceeded. We don't
|
||||||
|
// read this last item, but we need to do the deferred reads now.
|
||||||
|
if unreadSize > 0 {
|
||||||
|
if err := readData(secondIndex.filenum, readStart, unreadSize); err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
// Defer the read for later
|
||||||
|
unreadSize += size
|
||||||
|
totalSize += size
|
||||||
|
sizes = append(sizes, size)
|
||||||
|
if i == len(indices)-2 || uint64(totalSize) > maxBytes {
|
||||||
|
// Last item, need to do the read now
|
||||||
|
if err := readData(secondIndex.filenum, readStart, unreadSize); err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// Retrieve the data itself, decompress and return
|
return output[:outputSize], sizes, nil
|
||||||
blob := make([]byte, endOffset-startOffset)
|
|
||||||
if _, err := dataFile.ReadAt(blob, int64(startOffset)); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
t.readMeter.Mark(int64(len(blob) + 2*indexEntrySize))
|
|
||||||
return blob, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// has returns an indicator whether the specified number data
|
// has returns an indicator whether the specified number data
|
||||||
|
@ -74,7 +74,7 @@ func TestFreezerBasics(t *testing.T) {
|
|||||||
exp := getChunk(15, y)
|
exp := getChunk(15, y)
|
||||||
got, err := f.Retrieve(uint64(y))
|
got, err := f.Retrieve(uint64(y))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatalf("reading item %d: %v", y, err)
|
||||||
}
|
}
|
||||||
if !bytes.Equal(got, exp) {
|
if !bytes.Equal(got, exp) {
|
||||||
t.Fatalf("test %d, got \n%x != \n%x", y, got, exp)
|
t.Fatalf("test %d, got \n%x != \n%x", y, got, exp)
|
||||||
@ -692,3 +692,118 @@ func TestAppendTruncateParallel(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TestSequentialRead does some basic tests on the RetrieveItems.
|
||||||
|
func TestSequentialRead(t *testing.T) {
|
||||||
|
rm, wm, sg := metrics.NewMeter(), metrics.NewMeter(), metrics.NewGauge()
|
||||||
|
fname := fmt.Sprintf("batchread-%d", rand.Uint64())
|
||||||
|
{ // Fill table
|
||||||
|
f, err := newCustomTable(os.TempDir(), fname, rm, wm, sg, 50, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
// Write 15 bytes 30 times
|
||||||
|
for x := 0; x < 30; x++ {
|
||||||
|
data := getChunk(15, x)
|
||||||
|
f.Append(uint64(x), data)
|
||||||
|
}
|
||||||
|
f.DumpIndex(0, 30)
|
||||||
|
f.Close()
|
||||||
|
}
|
||||||
|
{ // Open it, iterate, verify iteration
|
||||||
|
f, err := newCustomTable(os.TempDir(), fname, rm, wm, sg, 50, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
items, err := f.RetrieveItems(0, 10000, 100000)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if have, want := len(items), 30; have != want {
|
||||||
|
t.Fatalf("want %d items, have %d ", want, have)
|
||||||
|
}
|
||||||
|
for i, have := range items {
|
||||||
|
want := getChunk(15, i)
|
||||||
|
if !bytes.Equal(want, have) {
|
||||||
|
t.Fatalf("data corruption: have\n%x\n, want \n%x\n", have, want)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
f.Close()
|
||||||
|
}
|
||||||
|
{ // Open it, iterate, verify byte limit. The byte limit is less than item
|
||||||
|
// size, so each lookup should only return one item
|
||||||
|
f, err := newCustomTable(os.TempDir(), fname, rm, wm, sg, 40, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
items, err := f.RetrieveItems(0, 10000, 10)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if have, want := len(items), 1; have != want {
|
||||||
|
t.Fatalf("want %d items, have %d ", want, have)
|
||||||
|
}
|
||||||
|
for i, have := range items {
|
||||||
|
want := getChunk(15, i)
|
||||||
|
if !bytes.Equal(want, have) {
|
||||||
|
t.Fatalf("data corruption: have\n%x\n, want \n%x\n", have, want)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
f.Close()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestSequentialReadByteLimit does some more advanced tests on batch reads.
|
||||||
|
// These tests check that when the byte limit hits, we correctly abort in time,
|
||||||
|
// but also properly do all the deferred reads for the previous data, regardless
|
||||||
|
// of whether the data crosses a file boundary or not.
|
||||||
|
func TestSequentialReadByteLimit(t *testing.T) {
|
||||||
|
rm, wm, sg := metrics.NewMeter(), metrics.NewMeter(), metrics.NewGauge()
|
||||||
|
fname := fmt.Sprintf("batchread-2-%d", rand.Uint64())
|
||||||
|
{ // Fill table
|
||||||
|
f, err := newCustomTable(os.TempDir(), fname, rm, wm, sg, 100, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
// Write 10 bytes 30 times,
|
||||||
|
// Splitting it at every 100 bytes (10 items)
|
||||||
|
for x := 0; x < 30; x++ {
|
||||||
|
data := getChunk(10, x)
|
||||||
|
f.Append(uint64(x), data)
|
||||||
|
}
|
||||||
|
f.Close()
|
||||||
|
}
|
||||||
|
for i, tc := range []struct {
|
||||||
|
items uint64
|
||||||
|
limit uint64
|
||||||
|
want int
|
||||||
|
}{
|
||||||
|
{9, 89, 8},
|
||||||
|
{10, 99, 9},
|
||||||
|
{11, 109, 10},
|
||||||
|
{100, 89, 8},
|
||||||
|
{100, 99, 9},
|
||||||
|
{100, 109, 10},
|
||||||
|
} {
|
||||||
|
{
|
||||||
|
f, err := newCustomTable(os.TempDir(), fname, rm, wm, sg, 100, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
items, err := f.RetrieveItems(0, tc.items, tc.limit)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if have, want := len(items), tc.want; have != want {
|
||||||
|
t.Fatalf("test %d: want %d items, have %d ", i, want, have)
|
||||||
|
}
|
||||||
|
for ii, have := range items {
|
||||||
|
want := getChunk(10, ii)
|
||||||
|
if !bytes.Equal(want, have) {
|
||||||
|
t.Fatalf("test %d: data corruption item %d: have\n%x\n, want \n%x\n", i, ii, have, want)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
f.Close()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -62,6 +62,12 @@ func (t *table) Ancient(kind string, number uint64) ([]byte, error) {
|
|||||||
return t.db.Ancient(kind, number)
|
return t.db.Ancient(kind, number)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ReadAncients is a noop passthrough that just forwards the request to the underlying
|
||||||
|
// database.
|
||||||
|
func (t *table) ReadAncients(kind string, start, count, maxBytes uint64) ([][]byte, error) {
|
||||||
|
return t.db.ReadAncients(kind, start, count, maxBytes)
|
||||||
|
}
|
||||||
|
|
||||||
// Ancients is a noop passthrough that just forwards the request to the underlying
|
// Ancients is a noop passthrough that just forwards the request to the underlying
|
||||||
// database.
|
// database.
|
||||||
func (t *table) Ancients() (uint64, error) {
|
func (t *table) Ancients() (uint64, error) {
|
||||||
|
@ -76,6 +76,13 @@ type AncientReader interface {
|
|||||||
// Ancient retrieves an ancient binary blob from the append-only immutable files.
|
// Ancient retrieves an ancient binary blob from the append-only immutable files.
|
||||||
Ancient(kind string, number uint64) ([]byte, error)
|
Ancient(kind string, number uint64) ([]byte, error)
|
||||||
|
|
||||||
|
// ReadAncients retrieves multiple items in sequence, starting from the index 'start'.
|
||||||
|
// It will return
|
||||||
|
// - at most 'count' items,
|
||||||
|
// - at least 1 item (even if exceeding the maxBytes), but will otherwise
|
||||||
|
// return as many items as fit into maxBytes.
|
||||||
|
ReadAncients(kind string, start, count, maxBytes uint64) ([][]byte, error)
|
||||||
|
|
||||||
// Ancients returns the ancient item numbers in the ancient store.
|
// Ancients returns the ancient item numbers in the ancient store.
|
||||||
Ancients() (uint64, error)
|
Ancients() (uint64, error)
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user