clientv3: make sure snapshot has integrity check hash

I've seen some cases SHA blobs are missing (still investigating).
Adding a check to make sure snapshot save always downloads
hash digests for integrity checks.

Signed-off-by: Gyuho Lee <leegyuho@amazon.com>
This commit is contained in:
Gyuho Lee 2020-05-14 12:18:49 -07:00
parent 4ddcc36057
commit 39c43cfb3c
2 changed files with 38 additions and 18 deletions

View File

@ -20,7 +20,7 @@ import (
"io" "io"
pb "go.etcd.io/etcd/v3/etcdserver/etcdserverpb" pb "go.etcd.io/etcd/v3/etcdserver/etcdserverpb"
"go.uber.org/zap"
"google.golang.org/grpc" "google.golang.org/grpc"
) )
@ -68,6 +68,7 @@ type Maintenance interface {
} }
type maintenance struct { type maintenance struct {
lg *zap.Logger
dial func(endpoint string) (pb.MaintenanceClient, func(), error) dial func(endpoint string) (pb.MaintenanceClient, func(), error)
remote pb.MaintenanceClient remote pb.MaintenanceClient
callOpts []grpc.CallOption callOpts []grpc.CallOption
@ -75,6 +76,7 @@ type maintenance struct {
func NewMaintenance(c *Client) Maintenance { func NewMaintenance(c *Client) Maintenance {
api := &maintenance{ api := &maintenance{
lg: c.lg,
dial: func(endpoint string) (pb.MaintenanceClient, func(), error) { dial: func(endpoint string) (pb.MaintenanceClient, func(), error) {
conn, err := c.Dial(endpoint) conn, err := c.Dial(endpoint)
if err != nil { if err != nil {
@ -93,6 +95,7 @@ func NewMaintenance(c *Client) Maintenance {
func NewMaintenanceFromMaintenanceClient(remote pb.MaintenanceClient, c *Client) Maintenance { func NewMaintenanceFromMaintenanceClient(remote pb.MaintenanceClient, c *Client) Maintenance {
api := &maintenance{ api := &maintenance{
lg: c.lg,
dial: func(string) (pb.MaintenanceClient, func(), error) { dial: func(string) (pb.MaintenanceClient, func(), error) {
return remote, func() {}, nil return remote, func() {}, nil
}, },
@ -193,23 +196,32 @@ func (m *maintenance) Snapshot(ctx context.Context) (io.ReadCloser, error) {
return nil, toErr(ctx, err) return nil, toErr(ctx, err)
} }
m.lg.Info("opened snapshot stream; downloading")
pr, pw := io.Pipe() pr, pw := io.Pipe()
go func() { go func() {
for { for {
resp, err := ss.Recv() resp, err := ss.Recv()
if err != nil { if err != nil {
switch err {
case io.EOF:
m.lg.Info("completed snapshot read; closing")
default:
m.lg.Warn("failed to receive from snapshot stream; closing", zap.Error(err))
}
pw.CloseWithError(err) pw.CloseWithError(err)
return return
} }
if resp == nil && err == nil {
break // can "resp == nil && err == nil"
} // before we receive snapshot SHA digest?
// No, server sends EOF with an empty response
// after it sends SHA digest at the end
if _, werr := pw.Write(resp.Blob); werr != nil { if _, werr := pw.Write(resp.Blob); werr != nil {
pw.CloseWithError(werr) pw.CloseWithError(werr)
return return
} }
} }
pw.Close()
}() }()
return &snapshotReadCloser{ctx: ctx, ReadCloser: pr}, nil return &snapshotReadCloser{ctx: ctx, ReadCloser: pr}, nil
} }

View File

@ -28,6 +28,7 @@ import (
"strings" "strings"
"time" "time"
"github.com/dustin/go-humanize"
bolt "go.etcd.io/bbolt" bolt "go.etcd.io/bbolt"
"go.etcd.io/etcd/v3/clientv3" "go.etcd.io/etcd/v3/clientv3"
"go.etcd.io/etcd/v3/etcdserver" "go.etcd.io/etcd/v3/etcdserver"
@ -89,6 +90,14 @@ type v3Manager struct {
skipHashCheck bool skipHashCheck bool
} }
// hasChecksum returns "true" if the file size "n"
// has appended sha256 hash digest.
func hasChecksum(n int64) bool {
// 512 is chosen because it's a minimum disk sector size
// smaller than (and multiplies to) OS page size in most systems
return (n % 512) == sha256.Size
}
// Save fetches snapshot from remote etcd server and saves data to target path. // Save fetches snapshot from remote etcd server and saves data to target path.
func (s *v3Manager) Save(ctx context.Context, cfg clientv3.Config, dbPath string) error { func (s *v3Manager) Save(ctx context.Context, cfg clientv3.Config, dbPath string) error {
if len(cfg.Endpoints) != 1 { if len(cfg.Endpoints) != 1 {
@ -108,10 +117,7 @@ func (s *v3Manager) Save(ctx context.Context, cfg clientv3.Config, dbPath string
if err != nil { if err != nil {
return fmt.Errorf("could not open %s (%v)", partpath, err) return fmt.Errorf("could not open %s (%v)", partpath, err)
} }
s.lg.Info( s.lg.Info("created temporary db file", zap.String("path", partpath))
"created temporary db file",
zap.String("path", partpath),
)
now := time.Now() now := time.Now()
var rd io.ReadCloser var rd io.ReadCloser
@ -119,23 +125,25 @@ func (s *v3Manager) Save(ctx context.Context, cfg clientv3.Config, dbPath string
if err != nil { if err != nil {
return err return err
} }
s.lg.Info( s.lg.Info("fetching snapshot", zap.String("endpoint", cfg.Endpoints[0]))
"fetching snapshot", var size int64
zap.String("endpoint", cfg.Endpoints[0]), size, err = io.Copy(f, rd)
) if err != nil {
if _, err = io.Copy(f, rd); err != nil {
return err return err
} }
if !hasChecksum(size) {
return fmt.Errorf("sha256 checksum not found [bytes: %d]", size)
}
if err = fileutil.Fsync(f); err != nil { if err = fileutil.Fsync(f); err != nil {
return err return err
} }
if err = f.Close(); err != nil { if err = f.Close(); err != nil {
return err return err
} }
s.lg.Info( s.lg.Info("fetched snapshot",
"fetched snapshot",
zap.String("endpoint", cfg.Endpoints[0]), zap.String("endpoint", cfg.Endpoints[0]),
zap.Duration("took", time.Since(now)), zap.String("size", humanize.Bytes(uint64(size))),
zap.String("took", humanize.Time(now)),
) )
if err = os.Rename(partpath, dbPath); err != nil { if err = os.Rename(partpath, dbPath); err != nil {
@ -362,7 +370,7 @@ func (s *v3Manager) saveDB() error {
if serr != nil { if serr != nil {
return serr return serr
} }
hasHash := (off % 512) == sha256.Size hasHash := hasChecksum(off)
if hasHash { if hasHash {
if err := db.Truncate(off - sha256.Size); err != nil { if err := db.Truncate(off - sha256.Size); err != nil {
return err return err