mirror of https://github.com/status-im/consul.git
test/deployer: add the method of deregistering services (#19525)
This commit is contained in:
parent
48d7d4a0fe
commit
8d6545ec43
|
@ -204,7 +204,7 @@ func Test_Snapshot_Restore_Agentless(t *testing.T) {
|
||||||
// Add a new static-server
|
// Add a new static-server
|
||||||
cfg = sp.Config()
|
cfg = sp.Config()
|
||||||
cluster = cfg.Cluster("dc1")
|
cluster = cfg.Cluster("dc1")
|
||||||
cluster.Nodes[3].Disabled = false // client 3 -- static-server
|
cluster.Nodes[3].Disabled = false // client 3 -- new static-server
|
||||||
require.NoError(t, sp.Relaunch(cfg))
|
require.NoError(t, sp.Relaunch(cfg))
|
||||||
|
|
||||||
// Ensure the static-client connected to static-server
|
// Ensure the static-client connected to static-server
|
||||||
|
|
|
@ -182,7 +182,7 @@ func (s *Sprawl) assignIPAddresses() error {
|
||||||
return fmt.Errorf("unknown network %q", addr.Network)
|
return fmt.Errorf("unknown network %q", addr.Network)
|
||||||
}
|
}
|
||||||
addr.IPAddress = net.IPByIndex(node.Index)
|
addr.IPAddress = net.IPByIndex(node.Index)
|
||||||
s.logger.Info("assign addr", "node", node.Name, "addr", addr.IPAddress)
|
s.logger.Info("assign addr", "node", node.Name, "addr", addr.IPAddress, "enabled", !node.Disabled)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -260,8 +260,11 @@ func (s *Sprawl) initConsulServers() error {
|
||||||
return fmt.Errorf("populateInitialConfigEntries[%s]: %w", cluster.Name, err)
|
return fmt.Errorf("populateInitialConfigEntries[%s]: %w", cluster.Name, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := s.populateInitialResources(cluster); err != nil {
|
if cluster.EnableV2 {
|
||||||
return fmt.Errorf("populateInitialResources[%s]: %w", cluster.Name, err)
|
// Resources are available only in V2
|
||||||
|
if err := s.populateInitialResources(cluster); err != nil {
|
||||||
|
return fmt.Errorf("populateInitialResources[%s]: %w", cluster.Name, err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := s.createAnonymousToken(cluster); err != nil {
|
if err := s.createAnonymousToken(cluster); err != nil {
|
||||||
|
@ -300,8 +303,8 @@ func (s *Sprawl) createFirstTime() error {
|
||||||
return fmt.Errorf("createAllServiceTokens: %w", err)
|
return fmt.Errorf("createAllServiceTokens: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := s.registerAllServicesForDataplaneInstances(); err != nil {
|
if err := s.syncAllServicesForDataplaneInstances(); err != nil {
|
||||||
return fmt.Errorf("registerAllServicesForDataplaneInstances: %w", err)
|
return fmt.Errorf("syncAllServicesForDataplaneInstances: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// We can do this ahead, because we've incrementally run terraform as
|
// We can do this ahead, because we've incrementally run terraform as
|
||||||
|
@ -368,8 +371,8 @@ func (s *Sprawl) preRegenTasks() error {
|
||||||
return fmt.Errorf("createAllServiceTokens: %w", err)
|
return fmt.Errorf("createAllServiceTokens: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := s.registerAllServicesForDataplaneInstances(); err != nil {
|
if err := s.syncAllServicesForDataplaneInstances(); err != nil {
|
||||||
return fmt.Errorf("registerAllServicesForDataplaneInstances: %w", err)
|
return fmt.Errorf("syncAllServicesForDataplaneInstances: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
|
|
|
@ -21,6 +21,7 @@ import (
|
||||||
"github.com/hashicorp/consul/testing/deployer/util"
|
"github.com/hashicorp/consul/testing/deployer/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// registerAllServicesToAgents registers services in agent-ful mode
|
||||||
func (s *Sprawl) registerAllServicesToAgents() error {
|
func (s *Sprawl) registerAllServicesToAgents() error {
|
||||||
for _, cluster := range s.topology.Clusters {
|
for _, cluster := range s.topology.Clusters {
|
||||||
if err := s.registerServicesToAgents(cluster); err != nil {
|
if err := s.registerServicesToAgents(cluster); err != nil {
|
||||||
|
@ -30,10 +31,10 @@ func (s *Sprawl) registerAllServicesToAgents() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Sprawl) registerAllServicesForDataplaneInstances() error {
|
func (s *Sprawl) syncAllServicesForDataplaneInstances() error {
|
||||||
for _, cluster := range s.topology.Clusters {
|
for _, cluster := range s.topology.Clusters {
|
||||||
if err := s.registerServicesForDataplaneInstances(cluster); err != nil {
|
if err := s.syncServicesForDataplaneInstances(cluster); err != nil {
|
||||||
return fmt.Errorf("registerServicesForDataplaneInstances[%s]: %w", cluster.Name, err)
|
return fmt.Errorf("syncServicesForDataplaneInstances[%s]: %w", cluster.Name, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
@ -86,7 +87,7 @@ func (s *Sprawl) registerAgentService(
|
||||||
}
|
}
|
||||||
|
|
||||||
if svc.IsMeshGateway {
|
if svc.IsMeshGateway {
|
||||||
return nil // handled at startup time for agent-full, but won't be for agent-less
|
return nil // handled at startup time for agent-ful, but won't be for agent-less
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -176,11 +177,94 @@ RETRY:
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Sprawl) registerServicesForDataplaneInstances(cluster *topology.Cluster) error {
|
// syncServicesForDataplaneInstances register/deregister services in the given cluster
|
||||||
|
func (s *Sprawl) syncServicesForDataplaneInstances(cluster *topology.Cluster) error {
|
||||||
identityInfo := make(map[topology.ServiceID]*Resource[*pbauth.WorkloadIdentity])
|
identityInfo := make(map[topology.ServiceID]*Resource[*pbauth.WorkloadIdentity])
|
||||||
|
|
||||||
|
// registerServiceAtNode is called when node is not disabled
|
||||||
|
registerServiceAtNode := func(node *topology.Node, svc *topology.Service) error {
|
||||||
|
if node.IsV2() {
|
||||||
|
pending := serviceInstanceToResources(node, svc)
|
||||||
|
|
||||||
|
workloadID := topology.NewServiceID(svc.WorkloadIdentity, svc.ID.Namespace, svc.ID.Partition)
|
||||||
|
if _, ok := identityInfo[workloadID]; !ok {
|
||||||
|
identityInfo[workloadID] = pending.WorkloadIdentity
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write workload
|
||||||
|
res, err := pending.Workload.Build()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error serializing resource %s: %w", util.IDToString(pending.Workload.Resource.Id), err)
|
||||||
|
}
|
||||||
|
workload, err := s.writeResource(cluster, res)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// Write check linked to workload
|
||||||
|
for _, check := range pending.HealthStatuses {
|
||||||
|
check.Resource.Owner = workload.Id
|
||||||
|
res, err := check.Build()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error serializing resource %s: %w", util.IDToString(check.Resource.Id), err)
|
||||||
|
}
|
||||||
|
if _, err := s.writeResource(cluster, res); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// maybe write destinations
|
||||||
|
if pending.Destinations != nil {
|
||||||
|
res, err := pending.Destinations.Build()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error serializing resource %s: %w", util.IDToString(pending.Destinations.Resource.Id), err)
|
||||||
|
}
|
||||||
|
if _, err := s.writeResource(cluster, res); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if pending.ProxyConfiguration != nil {
|
||||||
|
res, err := pending.ProxyConfiguration.Build()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error serializing resource %s: %w", util.IDToString(pending.ProxyConfiguration.Resource.Id), err)
|
||||||
|
}
|
||||||
|
if _, err := s.writeResource(cluster, res); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if err := s.registerCatalogServiceV1(cluster, node, svc); err != nil {
|
||||||
|
return fmt.Errorf("error registering service: %w", err)
|
||||||
|
}
|
||||||
|
if !svc.DisableServiceMesh {
|
||||||
|
if err := s.registerCatalogSidecarServiceV1(cluster, node, svc); err != nil {
|
||||||
|
return fmt.Errorf("error registering sidecar service: %w", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// deregisterServiceAtNode is called when node is disabled
|
||||||
|
deregisterServiceAtNode := func(node *topology.Node, svc *topology.Service) error {
|
||||||
|
if node.IsV2() {
|
||||||
|
// TODO: implement deregister services for v2
|
||||||
|
panic("deregister services is not implemented for V2")
|
||||||
|
} else {
|
||||||
|
if err := s.deregisterCatalogServiceV1(cluster, node, svc); err != nil {
|
||||||
|
return fmt.Errorf("error deregistering service: %w", err)
|
||||||
|
}
|
||||||
|
if !svc.DisableServiceMesh {
|
||||||
|
if err := s.deregisterCatalogSidecarServiceV1(cluster, node, svc); err != nil {
|
||||||
|
return fmt.Errorf("error deregistering sidecar service: %w", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var syncService func(node *topology.Node, svc *topology.Service) error
|
||||||
|
|
||||||
for _, node := range cluster.Nodes {
|
for _, node := range cluster.Nodes {
|
||||||
if !node.RunsWorkloads() || len(node.Services) == 0 || node.Disabled {
|
if !node.RunsWorkloads() || len(node.Services) == 0 {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -188,82 +272,42 @@ func (s *Sprawl) registerServicesForDataplaneInstances(cluster *topology.Cluster
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := s.registerCatalogNode(cluster, node); err != nil {
|
// Register virtual node service first if node is not disabled
|
||||||
return fmt.Errorf("error registering virtual node: %w", err)
|
if !node.Disabled {
|
||||||
|
if err := s.registerCatalogNode(cluster, node); err != nil {
|
||||||
|
return fmt.Errorf("error registering virtual node: %w", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Register/deregister services on the node
|
||||||
for _, svc := range node.Services {
|
for _, svc := range node.Services {
|
||||||
if node.IsV2() {
|
if !node.Disabled {
|
||||||
pending := serviceInstanceToResources(node, svc)
|
syncService = registerServiceAtNode
|
||||||
|
|
||||||
workloadID := topology.NewServiceID(svc.WorkloadIdentity, svc.ID.Namespace, svc.ID.Partition)
|
|
||||||
if _, ok := identityInfo[workloadID]; !ok {
|
|
||||||
identityInfo[workloadID] = pending.WorkloadIdentity
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write workload
|
|
||||||
res, err := pending.Workload.Build()
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("error serializing resource %s: %w", util.IDToString(pending.Workload.Resource.Id), err)
|
|
||||||
}
|
|
||||||
workload, err := s.writeResource(cluster, res)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
// Write check linked to workload
|
|
||||||
for _, check := range pending.HealthStatuses {
|
|
||||||
check.Resource.Owner = workload.Id
|
|
||||||
res, err := check.Build()
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("error serializing resource %s: %w", util.IDToString(check.Resource.Id), err)
|
|
||||||
}
|
|
||||||
if _, err := s.writeResource(cluster, res); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// maybe write destinations
|
|
||||||
if pending.Destinations != nil {
|
|
||||||
res, err := pending.Destinations.Build()
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("error serializing resource %s: %w", util.IDToString(pending.Destinations.Resource.Id), err)
|
|
||||||
}
|
|
||||||
if _, err := s.writeResource(cluster, res); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if pending.ProxyConfiguration != nil {
|
|
||||||
res, err := pending.ProxyConfiguration.Build()
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("error serializing resource %s: %w", util.IDToString(pending.ProxyConfiguration.Resource.Id), err)
|
|
||||||
}
|
|
||||||
if _, err := s.writeResource(cluster, res); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
if err := s.registerCatalogServiceV1(cluster, node, svc); err != nil {
|
syncService = deregisterServiceAtNode
|
||||||
return fmt.Errorf("error registering service: %w", err)
|
}
|
||||||
}
|
syncService(node, svc)
|
||||||
if !svc.DisableServiceMesh {
|
}
|
||||||
if err := s.registerCatalogSidecarServiceV1(cluster, node, svc); err != nil {
|
|
||||||
return fmt.Errorf("error registering sidecar service: %w", err)
|
// Deregister the virtual node if node is disabled
|
||||||
}
|
if node.Disabled {
|
||||||
}
|
if err := s.deregisterCatalogNode(cluster, node); err != nil {
|
||||||
|
return fmt.Errorf("error deregistering virtual node: %w", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, identity := range identityInfo {
|
|
||||||
res, err := identity.Build()
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("error serializing resource %s: %w", util.IDToString(identity.Resource.Id), err)
|
|
||||||
}
|
|
||||||
if _, err := s.writeResource(cluster, res); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if cluster.EnableV2 {
|
if cluster.EnableV2 {
|
||||||
|
for _, identity := range identityInfo {
|
||||||
|
res, err := identity.Build()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error serializing resource %s: %w", util.IDToString(identity.Resource.Id), err)
|
||||||
|
}
|
||||||
|
if _, err := s.writeResource(cluster, res); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for id, svcData := range cluster.Services {
|
for id, svcData := range cluster.Services {
|
||||||
svcInfo := &Resource[*pbcatalog.Service]{
|
svcInfo := &Resource[*pbcatalog.Service]{
|
||||||
Resource: &pbresource.Resource{
|
Resource: &pbresource.Resource{
|
||||||
|
@ -311,6 +355,16 @@ func (s *Sprawl) registerCatalogNode(
|
||||||
return s.registerCatalogNodeV1(cluster, node)
|
return s.registerCatalogNodeV1(cluster, node)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *Sprawl) deregisterCatalogNode(
|
||||||
|
cluster *topology.Cluster,
|
||||||
|
node *topology.Node,
|
||||||
|
) error {
|
||||||
|
if node.IsV2() {
|
||||||
|
panic("deregister V2 node is not implemented")
|
||||||
|
}
|
||||||
|
return s.deregisterCatalogNodeV1(cluster, node)
|
||||||
|
}
|
||||||
|
|
||||||
func (s *Sprawl) registerCatalogNodeV2(
|
func (s *Sprawl) registerCatalogNodeV2(
|
||||||
cluster *topology.Cluster,
|
cluster *topology.Cluster,
|
||||||
node *topology.Node,
|
node *topology.Node,
|
||||||
|
@ -413,6 +467,82 @@ RETRY:
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *Sprawl) deregisterCatalogNodeV1(
|
||||||
|
cluster *topology.Cluster,
|
||||||
|
node *topology.Node,
|
||||||
|
) error {
|
||||||
|
if !node.IsDataplane() {
|
||||||
|
panic("called wrong method type")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
client = s.clients[cluster.Name]
|
||||||
|
logger = s.logger.With("cluster", cluster.Name)
|
||||||
|
)
|
||||||
|
|
||||||
|
dereg := &api.CatalogDeregistration{
|
||||||
|
Node: node.PodName(),
|
||||||
|
Address: node.LocalAddress(),
|
||||||
|
}
|
||||||
|
if cluster.Enterprise {
|
||||||
|
dereg.Partition = node.Partition
|
||||||
|
}
|
||||||
|
|
||||||
|
// deregister synthetic node
|
||||||
|
RETRY:
|
||||||
|
if _, err := client.Catalog().Deregister(dereg, nil); err != nil {
|
||||||
|
if isACLNotFound(err) {
|
||||||
|
time.Sleep(50 * time.Millisecond)
|
||||||
|
goto RETRY
|
||||||
|
}
|
||||||
|
return fmt.Errorf("error deregistering virtual node %s: %w", node.ID(), err)
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.Info("virtual node removed",
|
||||||
|
"node", node.ID(),
|
||||||
|
)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Sprawl) deregisterCatalogServiceV1(
|
||||||
|
cluster *topology.Cluster,
|
||||||
|
node *topology.Node,
|
||||||
|
svc *topology.Service,
|
||||||
|
) error {
|
||||||
|
if !node.IsDataplane() {
|
||||||
|
panic("called wrong method type")
|
||||||
|
}
|
||||||
|
if node.IsV2() {
|
||||||
|
panic("don't call this")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
client = s.clients[cluster.Name]
|
||||||
|
logger = s.logger.With("cluster", cluster.Name)
|
||||||
|
)
|
||||||
|
|
||||||
|
dereg := &api.CatalogDeregistration{
|
||||||
|
Node: node.PodName(),
|
||||||
|
ServiceID: svc.ID.Name,
|
||||||
|
}
|
||||||
|
RETRY:
|
||||||
|
if _, err := client.Catalog().Deregister(dereg, nil); err != nil {
|
||||||
|
if isACLNotFound(err) {
|
||||||
|
time.Sleep(50 * time.Millisecond)
|
||||||
|
goto RETRY
|
||||||
|
}
|
||||||
|
return fmt.Errorf("error deregistering service %s at node %s: %w", svc.ID, node.ID(), err)
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.Info("dataplane service removed",
|
||||||
|
"service", svc.ID,
|
||||||
|
"node", node.ID(),
|
||||||
|
)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (s *Sprawl) registerCatalogServiceV1(
|
func (s *Sprawl) registerCatalogServiceV1(
|
||||||
cluster *topology.Cluster,
|
cluster *topology.Cluster,
|
||||||
node *topology.Node,
|
node *topology.Node,
|
||||||
|
@ -449,6 +579,50 @@ RETRY:
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *Sprawl) deregisterCatalogSidecarServiceV1(
|
||||||
|
cluster *topology.Cluster,
|
||||||
|
node *topology.Node,
|
||||||
|
svc *topology.Service,
|
||||||
|
) error {
|
||||||
|
if !node.IsDataplane() {
|
||||||
|
panic("called wrong method type")
|
||||||
|
}
|
||||||
|
if svc.DisableServiceMesh {
|
||||||
|
panic("not valid")
|
||||||
|
}
|
||||||
|
if node.IsV2() {
|
||||||
|
panic("don't call this")
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
client = s.clients[cluster.Name]
|
||||||
|
logger = s.logger.With("cluster", cluster.Name)
|
||||||
|
)
|
||||||
|
|
||||||
|
pid := svc.ID
|
||||||
|
pid.Name += "-sidecar-proxy"
|
||||||
|
dereg := &api.CatalogDeregistration{
|
||||||
|
Node: node.PodName(),
|
||||||
|
ServiceID: pid.Name,
|
||||||
|
}
|
||||||
|
|
||||||
|
RETRY:
|
||||||
|
if _, err := client.Catalog().Deregister(dereg, nil); err != nil {
|
||||||
|
if isACLNotFound(err) {
|
||||||
|
time.Sleep(50 * time.Millisecond)
|
||||||
|
goto RETRY
|
||||||
|
}
|
||||||
|
return fmt.Errorf("error deregistering service %s to node %s: %w", svc.ID, node.ID(), err)
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.Info("dataplane sidecar service removed",
|
||||||
|
"service", pid,
|
||||||
|
"node", node.ID(),
|
||||||
|
)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (s *Sprawl) registerCatalogSidecarServiceV1(
|
func (s *Sprawl) registerCatalogSidecarServiceV1(
|
||||||
cluster *topology.Cluster,
|
cluster *topology.Cluster,
|
||||||
node *topology.Node,
|
node *topology.Node,
|
||||||
|
|
Loading…
Reference in New Issue