Skip to content

Commit

Permalink
Remove autopilot existing plan before creation
Browse files Browse the repository at this point in the history
Signed-off-by: Alexey Makhov <amakhov@mirantis.com>
  • Loading branch information
makhov committed Dec 2, 2024
1 parent aef0261 commit 61d7732
Show file tree
Hide file tree
Showing 4 changed files with 140 additions and 23 deletions.
27 changes: 27 additions & 0 deletions internal/controller/controlplane/helper.go
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ import (
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"k8s.io/apimachinery/pkg/types"
"k8s.io/client-go/kubernetes"
"k8s.io/utils/ptr"

Check failure on line 34 in internal/controller/controlplane/helper.go

View workflow job for this annotation

GitHub Actions / Lint

"k8s.io/utils/ptr" imported and not used) (typecheck)

Check failure on line 34 in internal/controller/controlplane/helper.go

View workflow job for this annotation

GitHub Actions / Lint

"k8s.io/utils/ptr" imported and not used (typecheck)

Check failure on line 34 in internal/controller/controlplane/helper.go

View workflow job for this annotation

GitHub Actions / Build

"k8s.io/utils/ptr" imported and not used
clusterv1 "sigs.k8s.io/cluster-api/api/v1beta1"
"sigs.k8s.io/cluster-api/controllers/external"
"sigs.k8s.io/cluster-api/util/collections"
Expand Down Expand Up @@ -344,6 +345,28 @@ func (c *K0sController) createAutopilotPlan(ctx context.Context, kcp *cpv1beta1.
return nil
}

var existingPlan unstructured.Unstructured
err := clientset.RESTClient().Get().AbsPath("/apis/autopilot.k0sproject.io/v1beta2/plans/autopilot").Do(ctx).Into(&existingPlan)
if err != nil && !apierrors.IsNotFound(err) {
return fmt.Errorf("error getting autopilot plan: %w", err)
}

state, found, err := unstructured.NestedString(existingPlan.Object, "status", "state")
if err != nil {
return fmt.Errorf("error getting autopilot plan's state: %w", err)
}
if found {
if state == "Schedulable" || state == "SchedulableWait" {
// autopilot is already running
return nil
}
}

err = clientset.RESTClient().Delete().AbsPath("/apis/autopilot.k0sproject.io/v1beta2/plans/autopilot").Do(ctx).Error()
if err != nil && !apierrors.IsNotFound(err) {
return fmt.Errorf("error deleting autopilot plan: %w", err)
}

machines, err := collections.GetFilteredMachinesForCluster(ctx, c, cluster, collections.ControlPlaneMachines(cluster.Name), collections.ActiveMachines)
if err != nil {
return fmt.Errorf("error getting control plane machines: %w", err)
Expand Down Expand Up @@ -409,14 +432,18 @@ func minVersion(machines collections.Machines) (string, error) {
if machines == nil || machines.Len() == 0 {
return "", nil
}

versions := make([]*version.Version, 0, len(machines))
for _, m := range machines {
v, err := version.NewVersion(*m.Spec.Version)
if err != nil {
return "", fmt.Errorf("failed to parse version %s: %w", *m.Spec.Version, err)
}

versions = append(versions, v)
}

sort.Sort(version.Collection(versions))

return versions[0].String(), nil
}
Original file line number Diff line number Diff line change
Expand Up @@ -298,9 +298,13 @@ func (c *K0sController) reconcileMachines(ctx context.Context, cluster *clusterv

var clusterIsUpdating bool
for _, m := range machines.SortedByCreationTimestamp() {
if m.Spec.Version == nil || !versionMatches(m, kcp.Spec.Version) {
if m.Spec.Version == nil || (!versionMatches(m, kcp.Spec.Version)) {
clusterIsUpdating = true
machineNamesToDelete[m.Name] = true
if kcp.Spec.UpdateStrategy == cpv1beta1.UpdateInPlace {
desiredMachineNames[m.Name] = true
} else {
machineNamesToDelete[m.Name] = true
}
} else if !matchesTemplateClonedFrom(infraMachines, kcp, m) {
machineNamesToDelete[m.Name] = true
} else if machines.Len() > int(kcp.Spec.Replicas)+len(machineNamesToDelete) {
Expand All @@ -322,7 +326,6 @@ func (c *K0sController) reconcileMachines(ctx context.Context, cluster *clusterv
}
}
}

} else {
kubeClient, err := c.getKubeClient(ctx, cluster)
if err != nil {
Expand Down
4 changes: 2 additions & 2 deletions inttest/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -46,8 +46,8 @@ check-capi-controlplane-docker-tunneling: TIMEOUT=10m
check-capi-remote-machine: TIMEOUT=12m
check-capi-remote-machine-template: TIMEOUT=12m
check-capi-remote-machine-template-update: TIMEOUT=10m
check-capi-docker-machine-change-template: TIMEOUT=15m
check-capi-docker-machine-template-update: TIMEOUT=12m
check-capi-docker-machine-template-update: TIMEOUT=15m
check-capi-docker-machine-template-update-recreate: TIMEOUT=15m
check-capi-docker-machine-change-template: TIMEOUT=15m
check-capi-remote-machine-job-provision: TIMEOUT=10m
check-upgrade: TIMEOUT=20m
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,8 @@ package capicontolplanedockerdownscaling
import (
"context"
"fmt"
autopilot "github.com/k0sproject/k0s/pkg/apis/autopilot/v1beta2"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"os"
"os/exec"
"strconv"
Expand All @@ -39,11 +41,12 @@ import (

type CAPIControlPlaneDockerDownScalingSuite struct {
suite.Suite
client *kubernetes.Clientset
restConfig *rest.Config
clusterYamlsPath string
clusterYamlsUpdatePath string
ctx context.Context
client *kubernetes.Clientset
restConfig *rest.Config
clusterYamlsPath string
clusterYamlsUpdatePath string
clusterYamlsSecondUpdatePath string
ctx context.Context
}

func TestCAPIControlPlaneDockerDownScalingSuite(t *testing.T) {
Expand Down Expand Up @@ -71,6 +74,8 @@ func (s *CAPIControlPlaneDockerDownScalingSuite) SetupSuite() {
s.Require().NoError(os.WriteFile(s.clusterYamlsPath, []byte(dockerClusterYaml), 0644))
s.clusterYamlsUpdatePath = tmpDir + "/update.yaml"
s.Require().NoError(os.WriteFile(s.clusterYamlsUpdatePath, []byte(controlPlaneUpdate), 0644))
s.clusterYamlsSecondUpdatePath = tmpDir + "/update2.yaml"
s.Require().NoError(os.WriteFile(s.clusterYamlsSecondUpdatePath, []byte(controlPlaneSecondUpdate), 0644))

s.ctx, _ = util.NewSuiteContext(s.T())
}
Expand All @@ -93,8 +98,7 @@ func (s *CAPIControlPlaneDockerDownScalingSuite) TestCAPIControlPlaneDockerDownS
s.T().Log("cluster objects applied, waiting for cluster to be ready")

var localPort int
// nolint:staticcheck
err := wait.PollImmediateUntilWithContext(s.ctx, 1*time.Second, func(ctx context.Context) (bool, error) {
err := wait.PollUntilContextCancel(s.ctx, 1*time.Second, true, func(ctx context.Context) (bool, error) {
localPort, _ = getLBPort("docker-test-lb")
return localPort > 0, nil
})
Expand All @@ -104,8 +108,7 @@ func (s *CAPIControlPlaneDockerDownScalingSuite) TestCAPIControlPlaneDockerDownS
kmcKC, err := util.GetKMCClientSet(s.ctx, s.client, "docker-test", "default", localPort)
s.Require().NoError(err)

// nolint:staticcheck
err = wait.PollImmediateUntilWithContext(s.ctx, 1*time.Second, func(ctx context.Context) (bool, error) {
err = wait.PollUntilContextCancel(s.ctx, 1*time.Second, true, func(ctx context.Context) (bool, error) {
b, _ := s.client.RESTClient().
Get().
AbsPath("/healthz").
Expand All @@ -116,8 +119,7 @@ func (s *CAPIControlPlaneDockerDownScalingSuite) TestCAPIControlPlaneDockerDownS
s.Require().NoError(err)

for i := 0; i < 3; i++ {
// nolint:staticcheck
err = wait.PollImmediateUntilWithContext(s.ctx, 1*time.Second, func(ctx context.Context) (bool, error) {
err = wait.PollUntilContextCancel(s.ctx, 1*time.Second, true, func(ctx context.Context) (bool, error) {
nodeName := fmt.Sprintf("docker-test-%d", i)
output, err := exec.Command("docker", "exec", nodeName, "k0s", "status").Output()
if err != nil {
Expand All @@ -129,18 +131,57 @@ func (s *CAPIControlPlaneDockerDownScalingSuite) TestCAPIControlPlaneDockerDownS
s.Require().NoError(err)
}

var cnList autopilot.ControlNodeList
err = wait.PollUntilContextCancel(s.ctx, 1*time.Second, true, func(ctx context.Context) (bool, error) {
err = kmcKC.RESTClient().Get().AbsPath("/apis/autopilot.k0sproject.io/v1beta2/controlnodes").Do(ctx).Into(&cnList)
if err != nil {
return false, nil
}

return len(cnList.Items) == 3, nil
})
s.Require().NoError(err)

s.T().Log("waiting for node to be ready")
s.Require().NoError(util.WaitForNodeReadyStatus(s.ctx, kmcKC, "docker-test-worker-0", corev1.ConditionTrue))

s.T().Log("updating cluster objects")
s.updateClusterObjects()
// nolint:staticcheck
err = wait.PollImmediateUntilWithContext(s.ctx, 1*time.Second, func(ctx context.Context) (bool, error) {
err = wait.PollUntilContextCancel(s.ctx, 1*time.Second, true, func(ctx context.Context) (bool, error) {
output, err := exec.Command("docker", "exec", "docker-test-0", "k0s", "status").CombinedOutput()
if err != nil {
return false, nil
}

return strings.Contains(string(output), "Version: v1.30.2"), nil
})

s.Require().NoError(err)
err = wait.PollUntilContextCancel(s.ctx, 1*time.Second, true, func(ctx context.Context) (bool, error) {
var existingPlan unstructured.Unstructured
err = kmcKC.RESTClient().Get().AbsPath("/apis/autopilot.k0sproject.io/v1beta2/plans/autopilot").Do(ctx).Into(&existingPlan)
if err != nil {
return false, nil
}

state, _, err := unstructured.NestedString(existingPlan.Object, "status", "state")
if err != nil {
return false, nil
}

return state == "Completed", nil
})
s.Require().NoError(err)

s.T().Log("updating cluster objects again")
s.updateClusterObjectsAgain()
err = wait.PollUntilContextCancel(s.ctx, 1*time.Second, true, func(ctx context.Context) (bool, error) {
output, err := exec.Command("docker", "exec", "docker-test-0", "k0s", "status").CombinedOutput()
if err != nil {
return false, nil
}

return strings.Contains(string(output), "Version: v1.28"), nil
return strings.Contains(string(output), "Version: v1.31"), nil
})
s.Require().NoError(err)

Expand All @@ -159,6 +200,12 @@ func (s *CAPIControlPlaneDockerDownScalingSuite) updateClusterObjects() {
s.Require().NoError(err, "failed to update cluster objects: %s", string(out))
}

func (s *CAPIControlPlaneDockerDownScalingSuite) updateClusterObjectsAgain() {
// Exec via kubectl
out, err := exec.Command("kubectl", "apply", "-f", s.clusterYamlsSecondUpdatePath).CombinedOutput()
s.Require().NoError(err, "failed to update cluster objects: %s", string(out))
}

func (s *CAPIControlPlaneDockerDownScalingSuite) deleteCluster() {
// Exec via kubectl
out, err := exec.Command("kubectl", "delete", "-f", s.clusterYamlsPath).CombinedOutput()
Expand Down Expand Up @@ -220,8 +267,14 @@ metadata:
name: docker-test
spec:
replicas: 3
version: v1.27.1+k0s.0
version: v1.30.1+k0s.0
k0sConfigSpec:
postStartCommands:
- sed -i 's/RestartSec=120/RestartSec=1/' /etc/systemd/system/k0scontroller.service
- systemctl daemon-reload
args:
- --enable-worker
- --no-taints
k0s:
apiVersion: k0s.k0sproject.io/v1beta1
kind: ClusterConfig
Expand Down Expand Up @@ -253,7 +306,7 @@ metadata:
name: docker-test-worker-0
namespace: default
spec:
version: v1.27.1
version: v1.30.1
clusterName: docker-test
bootstrap:
configRef:
Expand All @@ -272,7 +325,7 @@ metadata:
namespace: default
spec:
# version is deliberately different to be able to verify we actually pick it up :)
version: v1.27.1+k0s.0
version: v1.30.1+k0s.0
---
apiVersion: infrastructure.cluster.x-k8s.io/v1beta1
kind: DockerMachine
Expand All @@ -290,8 +343,42 @@ metadata:
name: docker-test
spec:
replicas: 3
version: v1.28.7+k0s.0
version: v1.30.2+k0s.0
k0sConfigSpec:
args:
- --enable-worker
- --no-taints
k0s:
apiVersion: k0s.k0sproject.io/v1beta1
kind: ClusterConfig
metadata:
name: k0s
spec:
api:
extraArgs:
anonymous-auth: "true"
telemetry:
enabled: false
machineTemplate:
infrastructureRef:
apiVersion: infrastructure.cluster.x-k8s.io/v1beta1
kind: DockerMachineTemplate
name: docker-test-cp-template
namespace: default
`

var controlPlaneSecondUpdate = `
apiVersion: controlplane.cluster.x-k8s.io/v1beta1
kind: K0sControlPlane
metadata:
name: docker-test
spec:
replicas: 3
version: v1.31.2+k0s.0
k0sConfigSpec:
args:
- --enable-worker
- --no-taints
k0s:
apiVersion: k0s.k0sproject.io/v1beta1
kind: ClusterConfig
Expand Down

0 comments on commit 61d7732

Please sign in to comment.