package kv import ( "context" "errors" "reflect" "github.com/golang/snappy" fastssz "github.com/prysmaticlabs/fastssz" ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1" "go.opencensus.io/trace" "google.golang.org/protobuf/proto" ) func decode(ctx context.Context, data []byte, dst proto.Message) error { ctx, span := trace.StartSpan(ctx, "BeaconDB.decode") defer span.End() if ctx.Err() != nil { return ctx.Err() } data, err := snappy.Decode(nil, data) if err != nil { return err } if isSSZStorageFormat(dst) { return dst.(fastssz.Unmarshaler).UnmarshalSSZ(data) } return proto.Unmarshal(data, dst) } func encode(ctx context.Context, msg proto.Message) ([]byte, error) { ctx, span := trace.StartSpan(ctx, "BeaconDB.encode") defer span.End() if ctx.Err() != nil { return nil, ctx.Err() } if msg == nil || reflect.ValueOf(msg).IsNil() { return nil, errors.New("cannot encode nil message") } var enc []byte var err error if isSSZStorageFormat(msg) { enc, err = msg.(fastssz.Marshaler).MarshalSSZ() if err != nil { return nil, err } } else { enc, err = proto.Marshal(msg) if err != nil { return nil, err } } return snappy.Encode(nil, enc), nil } // isSSZStorageFormat returns true if the object type should be saved in SSZ encoded format. func isSSZStorageFormat(obj interface{}) bool { switch obj.(type) { case *ethpb.BeaconState: return true case *ethpb.SignedBeaconBlock: return true case *ethpb.SignedAggregateAttestationAndProof: return true case *ethpb.BeaconBlock: return true case *ethpb.Attestation: return true case *ethpb.Deposit: return true case *ethpb.AttesterSlashing: return true case *ethpb.ProposerSlashing: return true case *ethpb.VoluntaryExit: return true case *ethpb.ValidatorRegistrationV1: return true default: return false } }