[#1681] ir/netmap: Require MAINTENANCE mode to be allowed by network
There is a need to prevent limitless abuse of MAINTENANCE status of the storage nodes. To do this, configuration of the NeoFS network is going to be extended with the flag which allows the state. Until this is done, it makes sense to prepare a site for this in the code. Define `state.NetworkSettings` interface as an abstraction of global network configuration within the `state` package. Make `NetMapCandidateValidator` to depend on `NetworkSettings` and provide corresponding field setter. Change `VerifyAndUpdate` method's behavior to return an error for candidates with MAINTENANCE state if this state is disallowed by the network configuration. Provide `NetworkSettings` from the wrapper over Netmap contract's client on Inner Ring application side. The provider is implemented to statically disallow MAINTENANCE mode in order to save previous behavior. Signed-off-by: Leonard Lyubich <leonard@nspcc.ru> Signed-off-by: Leonard Lyubich <ctulhurider@gmail.com>
This commit is contained in:
parent
42fb40e841
commit
f64ae55806
4 changed files with 93 additions and 4 deletions
|
@ -691,6 +691,7 @@ func New(ctx context.Context, log *zap.Logger, cfg *viper.Viper, errChan chan<-
|
||||||
}
|
}
|
||||||
|
|
||||||
var netMapCandidateStateValidator statevalidation.NetMapCandidateValidator
|
var netMapCandidateStateValidator statevalidation.NetMapCandidateValidator
|
||||||
|
netMapCandidateStateValidator.SetNetworkSettings((*networkSettings)(server.netmapClient))
|
||||||
|
|
||||||
// create netmap processor
|
// create netmap processor
|
||||||
server.netmapProcessor, err = netmap.New(&netmap.Params{
|
server.netmapProcessor, err = netmap.New(&netmap.Params{
|
||||||
|
|
20
pkg/innerring/netmap.go
Normal file
20
pkg/innerring/netmap.go
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
package innerring
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/nspcc-dev/neofs-node/pkg/innerring/processors/netmap/nodevalidation/state"
|
||||||
|
netmapclient "github.com/nspcc-dev/neofs-node/pkg/morph/client/netmap"
|
||||||
|
)
|
||||||
|
|
||||||
|
/*
|
||||||
|
File contains dependencies for processor of the Netmap contract's notifications.
|
||||||
|
*/
|
||||||
|
|
||||||
|
// wraps Netmap contract's client and provides state.NetworkSettings.
|
||||||
|
type networkSettings netmapclient.Client
|
||||||
|
|
||||||
|
// MaintenanceModeAllowed requests network configuration from the Sidechain
|
||||||
|
// and check allowance of storage node's maintenance mode according to it.
|
||||||
|
// Always returns state.ErrMaintenanceModeDisallowed.
|
||||||
|
func (s *networkSettings) MaintenanceModeAllowed() error {
|
||||||
|
return state.ErrMaintenanceModeDisallowed
|
||||||
|
}
|
|
@ -12,6 +12,20 @@ import (
|
||||||
"github.com/nspcc-dev/neofs-sdk-go/netmap"
|
"github.com/nspcc-dev/neofs-sdk-go/netmap"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// ErrMaintenanceModeDisallowed is returned when maintenance mode is disallowed.
|
||||||
|
var ErrMaintenanceModeDisallowed = errors.New("maintenance mode is disallowed")
|
||||||
|
|
||||||
|
// NetworkSettings encapsulates current settings of the NeoFS network and
|
||||||
|
// provides interface used for processing the network map candidates.
|
||||||
|
type NetworkSettings interface {
|
||||||
|
// MaintenanceModeAllowed checks if maintenance state of the storage nodes
|
||||||
|
// is allowed to be set, and returns:
|
||||||
|
// no error if allowed;
|
||||||
|
// ErrMaintenanceModeDisallowed if disallowed;
|
||||||
|
// other error if there are any problems with the check.
|
||||||
|
MaintenanceModeAllowed() error
|
||||||
|
}
|
||||||
|
|
||||||
// NetMapCandidateValidator represents tool which checks state of nodes which
|
// NetMapCandidateValidator represents tool which checks state of nodes which
|
||||||
// are going to register in the NeoFS network (enter the network map).
|
// are going to register in the NeoFS network (enter the network map).
|
||||||
//
|
//
|
||||||
|
@ -21,21 +35,34 @@ import (
|
||||||
// NetMapCandidateValidator implements
|
// NetMapCandidateValidator implements
|
||||||
// github.com/nspcc-dev/neofs-node/pkg/innerring/processors/netmap.NodeValidator.
|
// github.com/nspcc-dev/neofs-node/pkg/innerring/processors/netmap.NodeValidator.
|
||||||
type NetMapCandidateValidator struct {
|
type NetMapCandidateValidator struct {
|
||||||
|
netSettings NetworkSettings
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetNetworkSettings specifies provider of the NetworkSettings interface.
|
||||||
|
// MUST be called before any VerifyAndUpdate call. Parameter MUST NOT be nil.
|
||||||
|
func (x *NetMapCandidateValidator) SetNetworkSettings(netSettings NetworkSettings) {
|
||||||
|
x.netSettings = netSettings
|
||||||
}
|
}
|
||||||
|
|
||||||
// VerifyAndUpdate checks state of the network map candidate described by
|
// VerifyAndUpdate checks state of the network map candidate described by
|
||||||
// netmap.NodeInfo parameter. Returns no error if status is correct, otherwise
|
// netmap.NodeInfo parameter. Returns no error if status is correct, otherwise
|
||||||
// returns an error describing a violation of the rules:
|
// returns an error describing a violation of the rules:
|
||||||
//
|
//
|
||||||
// status MUST be ONLINE
|
// status MUST be either ONLINE or MAINTENANCE;
|
||||||
|
// if status is MAINTENANCE, then it SHOULD be allowed by the network.
|
||||||
//
|
//
|
||||||
// VerifyAndUpdate does not mutate the parameter in a binary format.
|
// VerifyAndUpdate does not mutate the parameter in a binary format.
|
||||||
|
// MUST NOT be called before SetNetworkSettings.
|
||||||
//
|
//
|
||||||
// See also netmap.NodeInfo.IsOnline/SetOnline.
|
// See also netmap.NodeInfo.IsOnline/SetOnline and other similar methods.
|
||||||
func (x *NetMapCandidateValidator) VerifyAndUpdate(node *netmap.NodeInfo) error {
|
func (x *NetMapCandidateValidator) VerifyAndUpdate(node *netmap.NodeInfo) error {
|
||||||
if node.IsOnline() {
|
if node.IsOnline() {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return errors.New("invalid status: MUST be ONLINE")
|
if node.IsMaintenance() {
|
||||||
|
return x.netSettings.MaintenanceModeAllowed()
|
||||||
|
}
|
||||||
|
|
||||||
|
return errors.New("invalid status: MUST be either ONLINE or MAINTENANCE")
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,13 +8,31 @@ import (
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// implements state.NetworkSettings for testing.
|
||||||
|
type testNetworkSettings struct {
|
||||||
|
disallowed bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x testNetworkSettings) MaintenanceModeAllowed() error {
|
||||||
|
if x.disallowed {
|
||||||
|
return state.ErrMaintenanceModeDisallowed
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func TestValidator_VerifyAndUpdate(t *testing.T) {
|
func TestValidator_VerifyAndUpdate(t *testing.T) {
|
||||||
var v state.NetMapCandidateValidator
|
var vDefault state.NetMapCandidateValidator
|
||||||
|
var s testNetworkSettings
|
||||||
|
|
||||||
|
vDefault.SetNetworkSettings(s)
|
||||||
|
|
||||||
for _, testCase := range []struct {
|
for _, testCase := range []struct {
|
||||||
name string
|
name string
|
||||||
preparer func(*netmap.NodeInfo) // modifies zero instance
|
preparer func(*netmap.NodeInfo) // modifies zero instance
|
||||||
valid bool // is node valid after preparation
|
valid bool // is node valid after preparation
|
||||||
|
|
||||||
|
validatorPreparer func(*state.NetMapCandidateValidator) // optionally modifies default validator
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
name: "UNDEFINED",
|
name: "UNDEFINED",
|
||||||
|
@ -31,6 +49,22 @@ func TestValidator_VerifyAndUpdate(t *testing.T) {
|
||||||
preparer: (*netmap.NodeInfo).SetOffline,
|
preparer: (*netmap.NodeInfo).SetOffline,
|
||||||
valid: false,
|
valid: false,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
name: "MAINTENANCE/allowed",
|
||||||
|
preparer: (*netmap.NodeInfo).SetMaintenance,
|
||||||
|
valid: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "MAINTENANCE/disallowed",
|
||||||
|
preparer: (*netmap.NodeInfo).SetMaintenance,
|
||||||
|
valid: false,
|
||||||
|
validatorPreparer: func(v *state.NetMapCandidateValidator) {
|
||||||
|
var s testNetworkSettings
|
||||||
|
s.disallowed = true
|
||||||
|
|
||||||
|
v.SetNetworkSettings(s)
|
||||||
|
},
|
||||||
|
},
|
||||||
} {
|
} {
|
||||||
var node netmap.NodeInfo
|
var node netmap.NodeInfo
|
||||||
|
|
||||||
|
@ -40,6 +74,13 @@ func TestValidator_VerifyAndUpdate(t *testing.T) {
|
||||||
// save binary representation for mutation check
|
// save binary representation for mutation check
|
||||||
binNode := node.Marshal()
|
binNode := node.Marshal()
|
||||||
|
|
||||||
|
var v state.NetMapCandidateValidator
|
||||||
|
if testCase.validatorPreparer == nil {
|
||||||
|
v = vDefault
|
||||||
|
} else {
|
||||||
|
testCase.validatorPreparer(&v)
|
||||||
|
}
|
||||||
|
|
||||||
err := v.VerifyAndUpdate(&node)
|
err := v.VerifyAndUpdate(&node)
|
||||||
|
|
||||||
if testCase.valid {
|
if testCase.valid {
|
||||||
|
|
Loading…
Reference in a new issue