Skip to content

Commit

Permalink
clusterctl v1alpha4 should not install v1alpha3 providers
Browse files Browse the repository at this point in the history
  • Loading branch information
fabriziopandini committed Feb 18, 2021
1 parent 4a6cccd commit c62b77b
Show file tree
Hide file tree
Showing 3 changed files with 206 additions and 61 deletions.
5 changes: 5 additions & 0 deletions cmd/clusterctl/client/cluster/installer.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ import (
"github.com/pkg/errors"
"k8s.io/apimachinery/pkg/util/sets"
"k8s.io/apimachinery/pkg/util/version"
clusterv1 "sigs.k8s.io/cluster-api/api/v1alpha4"
clusterctlv1 "sigs.k8s.io/cluster-api/cmd/clusterctl/api/v1alpha3"
"sigs.k8s.io/cluster-api/cmd/clusterctl/client/config"
"sigs.k8s.io/cluster-api/cmd/clusterctl/client/repository"
Expand Down Expand Up @@ -244,6 +245,10 @@ func (i *providerInstaller) getProviderContract(providerInstanceContracts map[st
return "", errors.Errorf("invalid provider metadata: version %s for the provider %s does not match any release series", provider.Version, provider.InstanceName())
}

if releaseSeries.Contract != clusterv1.GroupVersion.Version {
return "", errors.Errorf("current version of clusterctl could install only %s providers, detected %s for provider %s", clusterv1.GroupVersion.Version, releaseSeries.Contract, provider.ManifestLabel())
}

providerInstanceContracts[provider.InstanceName()] = releaseSeries.Contract
return releaseSeries.Contract, nil
}
Expand Down
164 changes: 117 additions & 47 deletions cmd/clusterctl/client/cluster/installer_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,12 +22,19 @@ import (
. "github.com/onsi/gomega"

"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
clusterv1 "sigs.k8s.io/cluster-api/api/v1alpha4"
clusterctlv1 "sigs.k8s.io/cluster-api/cmd/clusterctl/api/v1alpha3"
"sigs.k8s.io/cluster-api/cmd/clusterctl/client/config"
"sigs.k8s.io/cluster-api/cmd/clusterctl/client/repository"
"sigs.k8s.io/cluster-api/cmd/clusterctl/internal/test"
)

const PreviousContractNotSupported = "v1alpha3"

var CurrentContract = clusterv1.GroupVersion.Version

const NextContractNotSupported = "v1alpha5"

func Test_providerInstaller_Validate(t *testing.T) {
fakeReader := test.NewFakeReader().
WithProvider("cluster-api", clusterctlv1.CoreProviderType, "https://somewhere.com").
Expand All @@ -36,42 +43,61 @@ func Test_providerInstaller_Validate(t *testing.T) {

repositoryMap := map[string]repository.Repository{
"cluster-api": test.NewFakeRepository().
WithVersions("v1.0.0", "v1.0.1").
WithVersions("v0.9.0", "v1.0.0", "v1.0.1", "v2.0.0").
WithMetadata("v0.9.0", &clusterctlv1.Metadata{
ReleaseSeries: []clusterctlv1.ReleaseSeries{
{Major: 0, Minor: 9, Contract: PreviousContractNotSupported},
},
}).
WithMetadata("v1.0.0", &clusterctlv1.Metadata{
ReleaseSeries: []clusterctlv1.ReleaseSeries{
{Major: 1, Minor: 0, Contract: "v1alpha3"},
{Major: 0, Minor: 9, Contract: PreviousContractNotSupported},
{Major: 1, Minor: 0, Contract: CurrentContract},
},
}).
WithMetadata("v2.0.0", &clusterctlv1.Metadata{
ReleaseSeries: []clusterctlv1.ReleaseSeries{
{Major: 1, Minor: 0, Contract: "v1alpha3"},
{Major: 2, Minor: 0, Contract: "v1alpha4"},
{Major: 0, Minor: 9, Contract: PreviousContractNotSupported},
{Major: 1, Minor: 0, Contract: CurrentContract},
{Major: 2, Minor: 0, Contract: NextContractNotSupported},
},
}),
"infrastructure-infra1": test.NewFakeRepository().
WithVersions("v1.0.0", "v1.0.1").
WithVersions("v0.9.0", "v1.0.0", "v1.0.1", "v2.0.0").
WithMetadata("v0.9.0", &clusterctlv1.Metadata{
ReleaseSeries: []clusterctlv1.ReleaseSeries{
{Major: 0, Minor: 9, Contract: PreviousContractNotSupported},
},
}).
WithMetadata("v1.0.0", &clusterctlv1.Metadata{
ReleaseSeries: []clusterctlv1.ReleaseSeries{
{Major: 1, Minor: 0, Contract: "v1alpha3"},
{Major: 0, Minor: 9, Contract: PreviousContractNotSupported},
{Major: 1, Minor: 0, Contract: CurrentContract},
},
}).
WithMetadata("v2.0.0", &clusterctlv1.Metadata{
ReleaseSeries: []clusterctlv1.ReleaseSeries{
{Major: 1, Minor: 0, Contract: "v1alpha3"},
{Major: 2, Minor: 0, Contract: "v1alpha4"},
{Major: 0, Minor: 9, Contract: PreviousContractNotSupported},
{Major: 1, Minor: 0, Contract: CurrentContract},
{Major: 2, Minor: 0, Contract: NextContractNotSupported},
},
}),
"infrastructure-infra2": test.NewFakeRepository().
WithVersions("v1.0.0", "v1.0.1").
WithVersions("v0.9.0", "v1.0.0", "v1.0.1", "v2.0.0").
WithMetadata("v0.9.0", &clusterctlv1.Metadata{
ReleaseSeries: []clusterctlv1.ReleaseSeries{
{Major: 0, Minor: 9, Contract: PreviousContractNotSupported},
},
}).
WithMetadata("v1.0.0", &clusterctlv1.Metadata{
ReleaseSeries: []clusterctlv1.ReleaseSeries{
{Major: 1, Minor: 0, Contract: "v1alpha3"},
{Major: 1, Minor: 0, Contract: CurrentContract},
},
}).
WithMetadata("v2.0.0", &clusterctlv1.Metadata{
ReleaseSeries: []clusterctlv1.ReleaseSeries{
{Major: 1, Minor: 0, Contract: "v1alpha3"},
{Major: 2, Minor: 0, Contract: "v1alpha4"},
{Major: 1, Minor: 0, Contract: CurrentContract},
{Major: 2, Minor: 0, Contract: NextContractNotSupported},
},
}),
}
Expand All @@ -86,105 +112,149 @@ func Test_providerInstaller_Validate(t *testing.T) {
wantErr bool
}{
{
name: "install core + infra1 on an empty cluster",
name: "install core/current contract + infra1/current contract on an empty cluster",
fields: fields{
proxy: test.NewFakeProxy(), //empty cluster
installQueue: []repository.Components{ // install core + infra1, v1alpha3 contract
installQueue: []repository.Components{
newFakeComponents("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "cluster-api-system", ""),
newFakeComponents("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "infra1-system", ""),
},
},
wantErr: false,
},
{
name: "install infra2 on a cluster already initialized with core + infra1",
name: "install infra2/current contract on a cluster already initialized with core/current contract + infra1/current contract",
fields: fields{
proxy: test.NewFakeProxy(). // cluster with core + infra1, v1alpha3 contract
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "cluster-api-system", "").
WithProviderInventory("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "infra1-system", ""),
installQueue: []repository.Components{ // install infra2, v1alpha3 contract
proxy: test.NewFakeProxy().
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "cluster-api-system", "").
WithProviderInventory("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "infra1-system", ""),
installQueue: []repository.Components{
newFakeComponents("infra2", clusterctlv1.InfrastructureProviderType, "v1.0.0", "infra2-system", ""),
},
},
wantErr: false,
},
{
name: "install another instance of infra1 on a cluster already initialized with core + infra1, no overlaps",
name: "install another instance of infra1/current contract on a cluster already initialized with core/current contract + infra1/current contract, no overlaps",
fields: fields{
proxy: test.NewFakeProxy(). // cluster with core + infra1, v1alpha3 contract
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "cluster-api-system", "").
WithProviderInventory("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "ns1", "ns1"),
installQueue: []repository.Components{ // install infra2, v1alpha3 contract
proxy: test.NewFakeProxy().
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "cluster-api-system", "").
WithProviderInventory("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "ns1", "ns1"),
installQueue: []repository.Components{
newFakeComponents("infra2", clusterctlv1.InfrastructureProviderType, "v1.0.0", "ns2", "ns2"),
},
},
wantErr: false,
},
{
name: "install another instance of infra1 on a cluster already initialized with core + infra1, same namespace of the existing infra1",
name: "install another instance of infra1/current contract on a cluster already initialized with core/current contract + infra1/current contract, same namespace of the existing infra1",
fields: fields{
proxy: test.NewFakeProxy(). // cluster with core + infra1, v1alpha3 contract
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "cluster-api-system", "").
WithProviderInventory("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "n1", ""),
installQueue: []repository.Components{ // install infra1, v1alpha3 contract
proxy: test.NewFakeProxy().
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "cluster-api-system", "").
WithProviderInventory("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "n1", ""),
installQueue: []repository.Components{
newFakeComponents("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "n1", ""),
},
},
wantErr: true,
},
{
name: "install another instance of infra1 on a cluster already initialized with core + infra1, watching overlap with the existing infra1",
name: "install another instance of infra1/current contract on a cluster already initialized with core/current contract + infra1/current contract, watching overlap with the existing infra1",
fields: fields{
proxy: test.NewFakeProxy(). // cluster with core + infra1, v1alpha3 contract
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "cluster-api-system", "").
WithProviderInventory("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "infra1-system", ""),
installQueue: []repository.Components{ // install infra1, v1alpha3 contract
proxy: test.NewFakeProxy().
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "cluster-api-system", "").
WithProviderInventory("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "infra1-system", ""),
installQueue: []repository.Components{
newFakeComponents("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "infra2-system", ""),
},
},
wantErr: true,
},
{
name: "install another instance of infra1 on a cluster already initialized with core + infra1, not part of the existing management group",
name: "install another instance of infra1/current contract on a cluster already initialized with core/current contract + infra1/current contract, not part of the existing management group",
fields: fields{
proxy: test.NewFakeProxy(). // cluster with core + infra1, v1alpha3 contract
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "ns1", "ns1").
WithProviderInventory("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "ns1", "ns1"),
installQueue: []repository.Components{ // install infra1, v1alpha3 contract
proxy: test.NewFakeProxy().
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "ns1", "ns1").
WithProviderInventory("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "ns1", "ns1"),
installQueue: []repository.Components{
newFakeComponents("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "ns2", "ns2"),
},
},
wantErr: true,
},
{
name: "install an instance of infra1 on a cluster already initialized with two core, but it is part of two management group",
name: "install an instance of infra1/current contract on a cluster already initialized with two core/current contract, but it is part of two management group",
fields: fields{
proxy: test.NewFakeProxy(). // cluster with two core (two management groups)
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "ns1", "ns1").
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "ns2", "ns2"),
installQueue: []repository.Components{ // install infra1, v1alpha3 contract
installQueue: []repository.Components{
newFakeComponents("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "infra1-system", ""),
},
},
wantErr: true,
},
{
name: "install core/previous contract + infra1/previous contract on an empty cluster (not supported)",
fields: fields{
proxy: test.NewFakeProxy(), //empty cluster
installQueue: []repository.Components{
newFakeComponents("cluster-api", clusterctlv1.CoreProviderType, "v0.9.0", "cluster-api-system", ""),
newFakeComponents("infra1", clusterctlv1.InfrastructureProviderType, "v0.9.0", "infra1-system", ""),
},
},
wantErr: true,
},
{
name: "install core/previous contract + infra1/current contract on an empty cluster (not supported)",
fields: fields{
proxy: test.NewFakeProxy(), //empty cluster
installQueue: []repository.Components{
newFakeComponents("cluster-api", clusterctlv1.CoreProviderType, "v0.9.0", "cluster-api-system", ""),
newFakeComponents("infra1", clusterctlv1.InfrastructureProviderType, "v1.0.0", "infra1-system", ""),
},
},
wantErr: true,
},
{
name: "install core@v1alpha3 + infra1@v1alpha4 on an empty cluster",
name: "install infra1/previous contract (not supported) on a cluster already initialized with core/current contract",
fields: fields{
proxy: test.NewFakeProxy().
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "ns1", "ns1"),
installQueue: []repository.Components{
newFakeComponents("infra1", clusterctlv1.InfrastructureProviderType, "v0.9.0", "infra1-system", ""),
},
},
wantErr: true,
},
{
name: "install core/next contract + infra1/next contract on an empty cluster (not supported)",
fields: fields{
proxy: test.NewFakeProxy(), //empty cluster
installQueue: []repository.Components{
newFakeComponents("cluster-api", clusterctlv1.CoreProviderType, "v2.0.0", "cluster-api-system", ""),
newFakeComponents("infra1", clusterctlv1.InfrastructureProviderType, "v2.0.0", "infra1-system", ""),
},
},
wantErr: true,
},
{
name: "install core/current contract + infra1/next contract on an empty cluster (not supported)",
fields: fields{
proxy: test.NewFakeProxy(), //empty cluster
installQueue: []repository.Components{ // install core + infra1, v1alpha3 contract
installQueue: []repository.Components{
newFakeComponents("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "cluster-api-system", ""),
newFakeComponents("infra1", clusterctlv1.InfrastructureProviderType, "v2.0.0", "infra1-system", ""),
},
},
wantErr: true,
},
{
name: "install infra1@v1alpha4 on a cluster already initialized with core@v1alpha3 +",
name: "install infra1/next contract (not supported) on a cluster already initialized with core/current contract",
fields: fields{
proxy: test.NewFakeProxy(). // cluster with one core, v1alpha3 contract
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "ns1", "ns1"),
installQueue: []repository.Components{ // install infra1, v1alpha4 contract
proxy: test.NewFakeProxy().
WithProviderInventory("cluster-api", clusterctlv1.CoreProviderType, "v1.0.0", "ns1", "ns1"),
installQueue: []repository.Components{
newFakeComponents("infra1", clusterctlv1.InfrastructureProviderType, "v2.0.0", "infra1-system", ""),
},
},
Expand Down
Loading

0 comments on commit c62b77b

Please sign in to comment.