mirror of https://github.com/ethereum/go-ethereum
core/rawdb, cmd, ethdb, eth: implement freezer tail deletion (#23954)
* core/rawdb, cmd, ethdb, eth: implement freezer tail deletion * core/rawdb: address comments from martin and sina * core/rawdb: fixes cornercase in tail deletion * core/rawdb: separate metadata into a standalone file * core/rawdb: remove unused code * core/rawdb: add random test * core/rawdb: polish code * core/rawdb: fsync meta file before manipulating the index * core/rawdb: fix typo * core/rawdb: address commentspull/24529/head
parent
8c8a9e5ca1
commit
538a868384
@ -0,0 +1,109 @@ |
||||
// Copyright 2022 The go-ethereum Authors
|
||||
// This file is part of the go-ethereum library.
|
||||
//
|
||||
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Lesser General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// The go-ethereum library is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU Lesser General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Lesser General Public License
|
||||
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>
|
||||
|
||||
package rawdb |
||||
|
||||
import ( |
||||
"io" |
||||
"os" |
||||
|
||||
"github.com/ethereum/go-ethereum/log" |
||||
"github.com/ethereum/go-ethereum/rlp" |
||||
) |
||||
|
||||
const freezerVersion = 1 // The initial version tag of freezer table metadata
|
||||
|
||||
// freezerTableMeta wraps all the metadata of the freezer table.
|
||||
type freezerTableMeta struct { |
||||
// Version is the versioning descriptor of the freezer table.
|
||||
Version uint16 |
||||
|
||||
// VirtualTail indicates how many items have been marked as deleted.
|
||||
// Its value is equal to the number of items removed from the table
|
||||
// plus the number of items hidden in the table, so it should never
|
||||
// be lower than the "actual tail".
|
||||
VirtualTail uint64 |
||||
} |
||||
|
||||
// newMetadata initializes the metadata object with the given virtual tail.
|
||||
func newMetadata(tail uint64) *freezerTableMeta { |
||||
return &freezerTableMeta{ |
||||
Version: freezerVersion, |
||||
VirtualTail: tail, |
||||
} |
||||
} |
||||
|
||||
// readMetadata reads the metadata of the freezer table from the
|
||||
// given metadata file.
|
||||
func readMetadata(file *os.File) (*freezerTableMeta, error) { |
||||
_, err := file.Seek(0, io.SeekStart) |
||||
if err != nil { |
||||
return nil, err |
||||
} |
||||
var meta freezerTableMeta |
||||
if err := rlp.Decode(file, &meta); err != nil { |
||||
return nil, err |
||||
} |
||||
return &meta, nil |
||||
} |
||||
|
||||
// writeMetadata writes the metadata of the freezer table into the
|
||||
// given metadata file.
|
||||
func writeMetadata(file *os.File, meta *freezerTableMeta) error { |
||||
_, err := file.Seek(0, io.SeekStart) |
||||
if err != nil { |
||||
return err |
||||
} |
||||
return rlp.Encode(file, meta) |
||||
} |
||||
|
||||
// loadMetadata loads the metadata from the given metadata file.
|
||||
// Initializes the metadata file with the given "actual tail" if
|
||||
// it's empty.
|
||||
func loadMetadata(file *os.File, tail uint64) (*freezerTableMeta, error) { |
||||
stat, err := file.Stat() |
||||
if err != nil { |
||||
return nil, err |
||||
} |
||||
// Write the metadata with the given actual tail into metadata file
|
||||
// if it's non-existent. There are two possible scenarios here:
|
||||
// - the freezer table is empty
|
||||
// - the freezer table is legacy
|
||||
// In both cases, write the meta into the file with the actual tail
|
||||
// as the virtual tail.
|
||||
if stat.Size() == 0 { |
||||
m := newMetadata(tail) |
||||
if err := writeMetadata(file, m); err != nil { |
||||
return nil, err |
||||
} |
||||
return m, nil |
||||
} |
||||
m, err := readMetadata(file) |
||||
if err != nil { |
||||
return nil, err |
||||
} |
||||
// Update the virtual tail with the given actual tail if it's even
|
||||
// lower than it. Theoretically it shouldn't happen at all, print
|
||||
// a warning here.
|
||||
if m.VirtualTail < tail { |
||||
log.Warn("Updated virtual tail", "have", m.VirtualTail, "now", tail) |
||||
m.VirtualTail = tail |
||||
if err := writeMetadata(file, m); err != nil { |
||||
return nil, err |
||||
} |
||||
} |
||||
return m, nil |
||||
} |
@ -0,0 +1,61 @@ |
||||
// Copyright 2022 The go-ethereum Authors
|
||||
// This file is part of the go-ethereum library.
|
||||
//
|
||||
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Lesser General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// The go-ethereum library is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU Lesser General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Lesser General Public License
|
||||
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>
|
||||
|
||||
package rawdb |
||||
|
||||
import ( |
||||
"io/ioutil" |
||||
"os" |
||||
"testing" |
||||
) |
||||
|
||||
func TestReadWriteFreezerTableMeta(t *testing.T) { |
||||
f, err := ioutil.TempFile(os.TempDir(), "*") |
||||
if err != nil { |
||||
t.Fatalf("Failed to create file %v", err) |
||||
} |
||||
err = writeMetadata(f, newMetadata(100)) |
||||
if err != nil { |
||||
t.Fatalf("Failed to write metadata %v", err) |
||||
} |
||||
meta, err := readMetadata(f) |
||||
if err != nil { |
||||
t.Fatalf("Failed to read metadata %v", err) |
||||
} |
||||
if meta.Version != freezerVersion { |
||||
t.Fatalf("Unexpected version field") |
||||
} |
||||
if meta.VirtualTail != uint64(100) { |
||||
t.Fatalf("Unexpected virtual tail field") |
||||
} |
||||
} |
||||
|
||||
func TestInitializeFreezerTableMeta(t *testing.T) { |
||||
f, err := ioutil.TempFile(os.TempDir(), "*") |
||||
if err != nil { |
||||
t.Fatalf("Failed to create file %v", err) |
||||
} |
||||
meta, err := loadMetadata(f, uint64(100)) |
||||
if err != nil { |
||||
t.Fatalf("Failed to read metadata %v", err) |
||||
} |
||||
if meta.Version != freezerVersion { |
||||
t.Fatalf("Unexpected version field") |
||||
} |
||||
if meta.VirtualTail != uint64(100) { |
||||
t.Fatalf("Unexpected virtual tail field") |
||||
} |
||||
} |
@ -0,0 +1,120 @@ |
||||
// Copyright 2022 The go-ethereum Authors
|
||||
// This file is part of the go-ethereum library.
|
||||
//
|
||||
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Lesser General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// The go-ethereum library is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU Lesser General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Lesser General Public License
|
||||
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
package rawdb |
||||
|
||||
import ( |
||||
"io" |
||||
"io/ioutil" |
||||
"os" |
||||
"path/filepath" |
||||
) |
||||
|
||||
// copyFrom copies data from 'srcPath' at offset 'offset' into 'destPath'.
|
||||
// The 'destPath' is created if it doesn't exist, otherwise it is overwritten.
|
||||
// Before the copy is executed, there is a callback can be registered to
|
||||
// manipulate the dest file.
|
||||
// It is perfectly valid to have destPath == srcPath.
|
||||
func copyFrom(srcPath, destPath string, offset uint64, before func(f *os.File) error) error { |
||||
// Create a temp file in the same dir where we want it to wind up
|
||||
f, err := ioutil.TempFile(filepath.Dir(destPath), "*") |
||||
if err != nil { |
||||
return err |
||||
} |
||||
fname := f.Name() |
||||
|
||||
// Clean up the leftover file
|
||||
defer func() { |
||||
if f != nil { |
||||
f.Close() |
||||
} |
||||
os.Remove(fname) |
||||
}() |
||||
// Apply the given function if it's not nil before we copy
|
||||
// the content from the src.
|
||||
if before != nil { |
||||
if err := before(f); err != nil { |
||||
return err |
||||
} |
||||
} |
||||
// Open the source file
|
||||
src, err := os.Open(srcPath) |
||||
if err != nil { |
||||
return err |
||||
} |
||||
if _, err = src.Seek(int64(offset), 0); err != nil { |
||||
src.Close() |
||||
return err |
||||
} |
||||
// io.Copy uses 32K buffer internally.
|
||||
_, err = io.Copy(f, src) |
||||
if err != nil { |
||||
src.Close() |
||||
return err |
||||
} |
||||
// Rename the temporary file to the specified dest name.
|
||||
// src may be same as dest, so needs to be closed before
|
||||
// we do the final move.
|
||||
src.Close() |
||||
|
||||
if err := f.Close(); err != nil { |
||||
return err |
||||
} |
||||
f = nil |
||||
|
||||
if err := os.Rename(fname, destPath); err != nil { |
||||
return err |
||||
} |
||||
return nil |
||||
} |
||||
|
||||
// openFreezerFileForAppend opens a freezer table file and seeks to the end
|
||||
func openFreezerFileForAppend(filename string) (*os.File, error) { |
||||
// Open the file without the O_APPEND flag
|
||||
// because it has differing behaviour during Truncate operations
|
||||
// on different OS's
|
||||
file, err := os.OpenFile(filename, os.O_RDWR|os.O_CREATE, 0644) |
||||
if err != nil { |
||||
return nil, err |
||||
} |
||||
// Seek to end for append
|
||||
if _, err = file.Seek(0, io.SeekEnd); err != nil { |
||||
return nil, err |
||||
} |
||||
return file, nil |
||||
} |
||||
|
||||
// openFreezerFileForReadOnly opens a freezer table file for read only access
|
||||
func openFreezerFileForReadOnly(filename string) (*os.File, error) { |
||||
return os.OpenFile(filename, os.O_RDONLY, 0644) |
||||
} |
||||
|
||||
// openFreezerFileTruncated opens a freezer table making sure it is truncated
|
||||
func openFreezerFileTruncated(filename string) (*os.File, error) { |
||||
return os.OpenFile(filename, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0644) |
||||
} |
||||
|
||||
// truncateFreezerFile resizes a freezer table file and seeks to the end
|
||||
func truncateFreezerFile(file *os.File, size int64) error { |
||||
if err := file.Truncate(size); err != nil { |
||||
return err |
||||
} |
||||
// Seek to end for append
|
||||
if _, err := file.Seek(0, io.SeekEnd); err != nil { |
||||
return err |
||||
} |
||||
return nil |
||||
} |
@ -0,0 +1,76 @@ |
||||
// Copyright 2022 The go-ethereum Authors
|
||||
// This file is part of the go-ethereum library.
|
||||
//
|
||||
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Lesser General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// The go-ethereum library is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU Lesser General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Lesser General Public License
|
||||
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
package rawdb |
||||
|
||||
import ( |
||||
"bytes" |
||||
"io/ioutil" |
||||
"os" |
||||
"testing" |
||||
) |
||||
|
||||
func TestCopyFrom(t *testing.T) { |
||||
var ( |
||||
content = []byte{0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8} |
||||
prefix = []byte{0x9, 0xa, 0xb, 0xc, 0xd, 0xf} |
||||
) |
||||
var cases = []struct { |
||||
src, dest string |
||||
offset uint64 |
||||
writePrefix bool |
||||
}{ |
||||
{"foo", "bar", 0, false}, |
||||
{"foo", "bar", 1, false}, |
||||
{"foo", "bar", 8, false}, |
||||
{"foo", "foo", 0, false}, |
||||
{"foo", "foo", 1, false}, |
||||
{"foo", "foo", 8, false}, |
||||
{"foo", "bar", 0, true}, |
||||
{"foo", "bar", 1, true}, |
||||
{"foo", "bar", 8, true}, |
||||
} |
||||
for _, c := range cases { |
||||
ioutil.WriteFile(c.src, content, 0644) |
||||
|
||||
if err := copyFrom(c.src, c.dest, c.offset, func(f *os.File) error { |
||||
if !c.writePrefix { |
||||
return nil |
||||
} |
||||
f.Write(prefix) |
||||
return nil |
||||
}); err != nil { |
||||
os.Remove(c.src) |
||||
t.Fatalf("Failed to copy %v", err) |
||||
} |
||||
|
||||
blob, err := ioutil.ReadFile(c.dest) |
||||
if err != nil { |
||||
os.Remove(c.src) |
||||
os.Remove(c.dest) |
||||
t.Fatalf("Failed to read %v", err) |
||||
} |
||||
want := content[c.offset:] |
||||
if c.writePrefix { |
||||
want = append(prefix, want...) |
||||
} |
||||
if !bytes.Equal(blob, want) { |
||||
t.Fatal("Unexpected value") |
||||
} |
||||
os.Remove(c.src) |
||||
os.Remove(c.dest) |
||||
} |
||||
} |
Loading…
Reference in new issue