Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix getOpenApiHighestElevatedVersion bug for unsupported API version #683

Merged
merged 5 commits into from
Jun 12, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions .changes/v2.25.0/683-bug-fixes.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
* Patched bug in core OpenAPI handling function `getOpenApiHighestElevatedVersion` that could
sometimes choose unsupported API versions in future VCD versions [GH-683]
3 changes: 2 additions & 1 deletion govcd/api_vcd.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,8 @@ func init() {
requestCounter = &counter
}

var minVcdApiVersion = "37.0" // supported by 10.4+
Didainius marked this conversation as resolved.
Show resolved Hide resolved

// VCDClientOption defines signature for customizing VCDClient using
// functional options pattern.
type VCDClientOption func(*VCDClient) error
Expand Down Expand Up @@ -136,7 +138,6 @@ func (vcdClient *VCDClient) vcdCloudApiAuthorize(user, pass, org string) (*http.
// NewVCDClient initializes VMware VMware Cloud Director client with reasonable defaults.
// It accepts functions of type VCDClientOption for adjusting defaults.
func NewVCDClient(vcdEndpoint url.URL, insecure bool, options ...VCDClientOption) *VCDClient {
minVcdApiVersion := "37.0" // supported by 10.4+
userDefinedApiVersion := os.Getenv("GOVCD_API_VERSION")
if userDefinedApiVersion != "" {
_, err := semver.NewVersion(userDefinedApiVersion)
Expand Down
8 changes: 7 additions & 1 deletion govcd/openapi_endpoints.go
Original file line number Diff line number Diff line change
Expand Up @@ -297,13 +297,19 @@ func (client *Client) getOpenApiHighestElevatedVersion(endpoint string) (string,
util.Logger.Printf("[DEBUG] Checking if elevated version '%s' is supported by VCD instance for endpoint '%s'",
elevatedVersion.Original(), endpoint)
// Check if maximum VCD API version supported is greater or equal to elevated version
if client.APIVCDMaxVersionIs(fmt.Sprintf(">= %s", elevatedVersion.Original())) {

if client.APIVCDMaxVersionIs(fmt.Sprintf(">= %s", elevatedVersion.Original())) &&
!client.APIClientVersionIs(fmt.Sprintf("> %s", elevatedVersion.Original())) {
Didainius marked this conversation as resolved.
Show resolved Hide resolved
util.Logger.Printf("[DEBUG] Elevated version '%s' is supported by VCD instance for endpoint '%s'",
elevatedVersion.Original(), endpoint)
// highest version found - store it and abort the loop
supportedElevatedVersion = elevatedVersion.Original()
break
} else {
util.Logger.Printf("[DEBUG] Skipped Elevated version '%s' for endpoint '%s', Default minimum version '%s'",
elevatedVersion.Original(), endpoint, client.APIVersion)
}

util.Logger.Printf("[DEBUG] API version '%s' is not supported by VCD instance for endpoint '%s'",
elevatedVersion.Original(), endpoint)
}
Expand Down
84 changes: 69 additions & 15 deletions govcd/openapi_endpoints_unit_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@ import (
"fmt"
"testing"

semver "github.com/hashicorp/go-version"

"github.com/vmware/go-vcloud-director/v2/types/v56"
)

Expand All @@ -16,12 +18,20 @@ import (
// * Automatically generated tests where available VCD version does not satisfy it
// * Automatically generated tests to check if each elevated API version is returned for endpoints that have it defined
func TestClient_getOpenApiHighestElevatedVersion(t *testing.T) {
semverMinVcdApiVersion, err := semver.NewSemver(minVcdApiVersion)
if err != nil {
t.Fatalf("error parsing 'minVcdApiVersion': %s", err)
}

type testCase struct {
name string
supportedVersions SupportedVersions
endpoint string
wantVersion string
wantErr bool
// overrideClientMinApiVersion is an option to override default expected version that is
// defined in global variable`minVcdApiVersion`
overrideClientMinApiVersion string
}

// Start with some statically defined tests for known endpoints
Expand All @@ -34,40 +44,63 @@ func TestClient_getOpenApiHighestElevatedVersion(t *testing.T) {
wantErr: true, // NAT requires at least version 34.0
},
{
name: "VCD_minimum_required_version_only",
supportedVersions: renderSupportedVersions([]string{"28.0", "29.0", "30.0", "31.0", "32.0", "33.0", "34.0"}),
endpoint: types.OpenApiPathVersion1_0_0 + types.OpenApiEndpointNsxtNatRules,
wantVersion: "34.0",
wantErr: false, // NAT minimum requirement is version 34.0
name: "VCD_minimum_required_version_only",
supportedVersions: renderSupportedVersions([]string{"28.0", "29.0", "30.0", "31.0", "32.0", "33.0", "34.0"}),
endpoint: types.OpenApiPathVersion1_0_0 + types.OpenApiEndpointNsxtNatRules,
wantVersion: "34.0",
wantErr: false, // NAT minimum requirement is version 34.0
overrideClientMinApiVersion: "34.0",
},
{
name: "VCD_minimum_required_version_only_unordered",
// Explicitly pass in unordered VCD API supported versions to ensure that ordering and matching works well
supportedVersions: renderSupportedVersions([]string{"33.0", "34.0", "30.0", "31.0", "32.0", "28.0", "29.0"}),
endpoint: types.OpenApiPathVersion1_0_0 + types.OpenApiEndpointNsxtNatRules,
wantVersion: "34.0",
wantErr: false, // NAT minimum requirement is version 34.0
supportedVersions: renderSupportedVersions([]string{"33.0", "34.0", "30.0", "31.0", "32.0", "28.0", "29.0"}),
endpoint: types.OpenApiPathVersion1_0_0 + types.OpenApiEndpointNsxtNatRules,
wantVersion: "34.0",
wantErr: false, // NAT minimum requirement is version 34.0
overrideClientMinApiVersion: "34.0",
},
{
name: "VCD_version_higher_than_elevated_version_entries",
supportedVersions: renderSupportedVersions([]string{"37.0", "37.1", "37.2", "37.3", "38.0", "38.1", "39.0"}),
endpoint: types.OpenApiPathVersion1_0_0 + types.OpenApiEndpointFirewallGroups,
wantVersion: minVcdApiVersion,
wantErr: false,
Didainius marked this conversation as resolved.
Show resolved Hide resolved
},
}

// Generate unit tests for each defined endpoint in endpointMinApiVersions which does not have an elevated
// version entry in endpointElevatedApiVersions.
// Always expect to get minimal version returned without error
// Expect to get minimal supported version returned without error
for endpointName, minimumRequiredVersion := range endpointMinApiVersions {
// Do not create a test case for those endpoints which explicitly have elevated versions defined in
// endpointElevatedApiVersions
if _, hasEntry := endpointElevatedApiVersions[endpointName]; hasEntry {
continue
}

wantVersion := minimumRequiredVersion
semverWantVersion, err := semver.NewSemver(wantVersion)
if err != nil {
t.Fatalf("error parsing 'singleElevatedApiVersion': %s", err)
}

if semverWantVersion.LessThan(semverMinVcdApiVersion) {
semverMinVcdApiVersionSegments := semverMinVcdApiVersion.Segments()
wantVersion = fmt.Sprintf("%d.%d", semverMinVcdApiVersionSegments[0], semverMinVcdApiVersionSegments[1])
Didainius marked this conversation as resolved.
Show resolved Hide resolved
if testVerbose {
fmt.Printf("# Overriding wanted version to %s for endpoint %s\n", wantVersion, endpointName)
}
}

tCase := testCase{
name: fmt.Sprintf("%s_minimum_version_%s", minimumRequiredVersion, endpointName),
// Put a list of versions which always satisfied minimum requirement
supportedVersions: renderSupportedVersions([]string{
"27.0", "28.0", "29.0", "30.0", "31.0", "32.0", "33.0", "34.0", "35.0", "35.1", "35.2", "36.0", "37.0", "38.0",
}),
endpoint: endpointName,
wantVersion: minimumRequiredVersion,
wantVersion: wantVersion,
wantErr: false,
}
tests = append(tests, tCase)
Expand All @@ -89,23 +122,38 @@ func TestClient_getOpenApiHighestElevatedVersion(t *testing.T) {
}

// Generate tests for each elevated API version in endpoints which do have elevated rights defined
// Expect to get either that version or minimum supported version
for endpointName := range endpointMinApiVersions {
// Do not create a test case for those endpoints which do not have endpointElevatedApiVersions specified
if _, hasEntry := endpointElevatedApiVersions[endpointName]; !hasEntry {
continue
}

// generate tests for all elevated rights and expect to get
for _, singleElevatedApiVerion := range endpointElevatedApiVersions[endpointName] {
for _, singleElevatedApiVersion := range endpointElevatedApiVersions[endpointName] {
wantVersion := singleElevatedApiVersion

semverWantVersion, err := semver.NewSemver(wantVersion)
if err != nil {
t.Fatalf("error parsing 'singleElevatedApiVersion': %s", err)
}

if semverWantVersion.LessThan(semverMinVcdApiVersion) {
semverMinVcdApiVersionSegments := semverMinVcdApiVersion.Segments()
wantVersion = fmt.Sprintf("%d.%d", semverMinVcdApiVersionSegments[0], semverMinVcdApiVersionSegments[1])
if testVerbose {
fmt.Printf("# Overriding wanted version to %s for endpoint %s\n", wantVersion, endpointName)
}
}

tCase := testCase{
name: fmt.Sprintf("elevated_up_to_%s_for_%s", singleElevatedApiVerion, endpointName),
name: fmt.Sprintf("elevated_up_to_%s_for_%s", singleElevatedApiVersion, endpointName),
// Insert some older versions and make it so that the highest elevated API version matches
supportedVersions: renderSupportedVersions([]string{
"27.0", singleElevatedApiVerion, "23.0", "30.0",
"27.0", singleElevatedApiVersion, "23.0", "30.0",
}),
endpoint: endpointName,
wantVersion: singleElevatedApiVerion,
wantVersion: wantVersion,
wantErr: false,
}
tests = append(tests, tCase)
Expand All @@ -117,7 +165,13 @@ func TestClient_getOpenApiHighestElevatedVersion(t *testing.T) {
t.Run(tt.name, func(t *testing.T) {
client := &Client{
supportedVersions: tt.supportedVersions,
APIVersion: minVcdApiVersion,
}

if tt.overrideClientMinApiVersion != "" {
client.APIVersion = tt.overrideClientMinApiVersion
}

got, err := client.getOpenApiHighestElevatedVersion(tt.endpoint)
if (err != nil) != tt.wantErr {
t.Errorf("getOpenApiHighestElevatedVersion() error = %v, wantErr %v", err, tt.wantErr)
Expand Down
Loading