2020-06-27 23:53:33 +02:00
|
|
|
package logic
|
|
|
|
|
|
|
|
import (
|
2022-05-01 14:46:38 +02:00
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
|
2021-02-28 23:33:28 +01:00
|
|
|
"github.com/go-playground/validator"
|
2020-06-27 23:53:33 +02:00
|
|
|
"github.com/pkg/errors"
|
2020-08-31 16:04:00 +02:00
|
|
|
|
2020-06-27 23:53:33 +02:00
|
|
|
"github.com/zrepl/zrepl/config"
|
|
|
|
"github.com/zrepl/zrepl/replication/logic/pdu"
|
|
|
|
)
|
|
|
|
|
2022-05-01 14:46:38 +02:00
|
|
|
//go:generate enumer -type=InitialReplicationAutoResolution -trimprefix=InitialReplicationAutoResolution
|
|
|
|
type InitialReplicationAutoResolution uint32
|
|
|
|
|
|
|
|
const (
|
|
|
|
InitialReplicationAutoResolutionMostRecent InitialReplicationAutoResolution = 1 << iota
|
|
|
|
InitialReplicationAutoResolutionAll
|
|
|
|
InitialReplicationAutoResolutionFail
|
|
|
|
)
|
|
|
|
|
|
|
|
var initialReplicationAutoResolutionConfigMap = map[InitialReplicationAutoResolution]string{
|
|
|
|
InitialReplicationAutoResolutionMostRecent: "most_recent",
|
|
|
|
InitialReplicationAutoResolutionAll: "all",
|
|
|
|
InitialReplicationAutoResolutionFail: "fail",
|
|
|
|
}
|
|
|
|
|
|
|
|
func InitialReplicationAutoResolutionFromConfig(in string) (InitialReplicationAutoResolution, error) {
|
|
|
|
for v, s := range initialReplicationAutoResolutionConfigMap {
|
|
|
|
if s == in {
|
|
|
|
return v, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
l := make([]string, 0, len(initialReplicationAutoResolutionConfigMap))
|
|
|
|
for _, v := range InitialReplicationAutoResolutionValues() {
|
|
|
|
l = append(l, initialReplicationAutoResolutionConfigMap[v])
|
|
|
|
}
|
|
|
|
return 0, fmt.Errorf("invalid value %q, must be one of %s", in, strings.Join(l, ", "))
|
|
|
|
}
|
|
|
|
|
|
|
|
type ConflictResolution struct {
|
|
|
|
InitialReplication InitialReplicationAutoResolution
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *ConflictResolution) Validate() error {
|
|
|
|
if !c.InitialReplication.IsAInitialReplicationAutoResolution() {
|
|
|
|
return errors.Errorf("must be one of %s", InitialReplicationAutoResolutionValues())
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ConflictResolutionFromConfig(in *config.ConflictResolution) (*ConflictResolution, error) {
|
|
|
|
|
|
|
|
initialReplication, err := InitialReplicationAutoResolutionFromConfig(in.InitialReplication)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Errorf("field `initial_replication` is invalid: %q is not one of %v", in.InitialReplication, InitialReplicationAutoResolutionValues())
|
|
|
|
}
|
|
|
|
|
|
|
|
return &ConflictResolution{
|
|
|
|
InitialReplication: initialReplication,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2020-06-27 23:53:33 +02:00
|
|
|
type PlannerPolicy struct {
|
2022-05-01 14:46:38 +02:00
|
|
|
EncryptedSend tri // all sends must be encrypted (send -w, and encryption!=off)
|
|
|
|
ConflictResolution *ConflictResolution `validate:"ne=nil"`
|
|
|
|
ReplicationConfig *pdu.ReplicationConfig `validate:"ne=nil"`
|
|
|
|
SizeEstimationConcurrency int `validate:"gte=1"`
|
2021-02-28 23:33:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
var validate = validator.New()
|
|
|
|
|
|
|
|
func (p PlannerPolicy) Validate() error {
|
2022-05-01 14:46:38 +02:00
|
|
|
if err := validate.Struct(p); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := p.ConflictResolution.Validate(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
2020-06-27 23:53:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func ReplicationConfigFromConfig(in *config.Replication) (*pdu.ReplicationConfig, error) {
|
|
|
|
initial, err := pduReplicationGuaranteeKindFromConfig(in.Protection.Initial)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "field 'initial'")
|
|
|
|
}
|
|
|
|
incremental, err := pduReplicationGuaranteeKindFromConfig(in.Protection.Incremental)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "field 'incremental'")
|
|
|
|
}
|
|
|
|
return &pdu.ReplicationConfig{
|
|
|
|
Protection: &pdu.ReplicationConfigProtection{
|
|
|
|
Initial: initial,
|
|
|
|
Incremental: incremental,
|
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func pduReplicationGuaranteeKindFromConfig(in string) (k pdu.ReplicationGuaranteeKind, _ error) {
|
|
|
|
switch in {
|
|
|
|
case "guarantee_nothing":
|
|
|
|
return pdu.ReplicationGuaranteeKind_GuaranteeNothing, nil
|
|
|
|
case "guarantee_incremental":
|
|
|
|
return pdu.ReplicationGuaranteeKind_GuaranteeIncrementalReplication, nil
|
|
|
|
case "guarantee_resumability":
|
|
|
|
return pdu.ReplicationGuaranteeKind_GuaranteeResumability, nil
|
|
|
|
default:
|
|
|
|
return k, errors.Errorf("%q is not in guarantee_{nothing,incremental,resumability}", in)
|
|
|
|
}
|
|
|
|
}
|