Skip to content

Commit

Permalink
Filter out helm charts based on kubernetes version
Browse files Browse the repository at this point in the history
  • Loading branch information
akashshinde committed Dec 1, 2020
1 parent 01588d5 commit 90f1269
Show file tree
Hide file tree
Showing 17 changed files with 1,219 additions and 36 deletions.
4 changes: 4 additions & 0 deletions docs/helm/endpoints_api.md
Original file line number Diff line number Diff line change
Expand Up @@ -335,6 +335,10 @@ _Returns repository index file containing all entries from all configured reposi

`GET`

* **Supported URL Query Parameter:**
* `onlyCompatible` - `true`/`false` Setting true would return helm charts which are supported in the provided cluster.
Default value is set to true if not provided.

* **Success Response:**

* **Code:** 200 <br />
Expand Down
24 changes: 24 additions & 0 deletions pkg/helm/actions/kube_version.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
package actions

import (
"errors"
"k8s.io/client-go/discovery"
"k8s.io/client-go/rest"
)

func GetKubeVersion(config *rest.Config) (string, error) {
client, err := discovery.NewDiscoveryClientForConfig(config)
if err != nil {
return "", err
}

kubeVersion, err := client.ServerVersion()
if err != nil {
return "", err
}

if kubeVersion != nil {
return kubeVersion.String(), nil
}
return "", errors.New("failed to get kubernetes version")
}
27 changes: 22 additions & 5 deletions pkg/helm/chartproxy/proxy.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package chartproxy

import (
"helm.sh/helm/v3/pkg/chartutil"
"helm.sh/helm/v3/pkg/repo"
"k8s.io/client-go/dynamic"
"k8s.io/client-go/kubernetes"
Expand All @@ -14,10 +15,11 @@ type proxy struct {
dynamicClient dynamic.Interface
coreV1Client v1.CoreV1Interface
helmRepoGetter HelmRepoGetter
kubeVersion string
}

type Proxy interface {
IndexFile() (*repo.IndexFile, error)
IndexFile(onlyCompatible bool) (*repo.IndexFile, error)
}

type RestConfigProvider func() (*rest.Config, error)
Expand All @@ -44,13 +46,16 @@ func coreClientProvider(p *proxy) error {

var defaultOptions = []ProxyOption{dynamicKubeClientProvider, coreClientProvider}

func New(k8sConfig RestConfigProvider, opts ...ProxyOption) (Proxy, error) {
func New(k8sConfig RestConfigProvider, kubeVersion string, opts ...ProxyOption) (Proxy, error) {
config, err := k8sConfig()

if err != nil {
return nil, err
}

p := &proxy{
config: config,
config: config,
kubeVersion: kubeVersion,
}

if len(opts) == 0 {
Expand All @@ -64,7 +69,7 @@ func New(k8sConfig RestConfigProvider, opts ...ProxyOption) (Proxy, error) {
return p, nil
}

func (p *proxy) IndexFile() (*repo.IndexFile, error) {
func (p *proxy) IndexFile(onlyCompatible bool) (*repo.IndexFile, error) {
helmRepos, err := p.helmRepoGetter.List()
if err != nil {
return nil, err
Expand All @@ -77,8 +82,20 @@ func (p *proxy) IndexFile() (*repo.IndexFile, error) {
klog.Errorf("Error retrieving index file for %v: %v", helmRepo, err)
continue
}

for key, entry := range idxFile.Entries {
indexFile.Entries[key+"--"+helmRepo.Name] = entry
if onlyCompatible {
for i := len(entry) - 1; i >= 0; i-- {
if entry[i].Metadata.KubeVersion != "" && p.kubeVersion != "" {
if !chartutil.IsCompatibleRange(entry[i].Metadata.KubeVersion, p.kubeVersion) {
entry = append(entry[:i], entry[i+1:]...)
}
}
}
}
if len(entry) > 0 {
indexFile.Entries[key+"--"+helmRepo.Name] = entry
}
}
}
return indexFile, nil
Expand Down
72 changes: 62 additions & 10 deletions pkg/helm/chartproxy/proxy_test.go
Original file line number Diff line number Diff line change
@@ -1,31 +1,82 @@
package chartproxy

import (
"golang.org/x/net/context"
"helm.sh/helm/v3/pkg/repo"
"io/ioutil"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"reflect"
"testing"

"golang.org/x/net/context"
"helm.sh/helm/v3/pkg/repo"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"k8s.io/client-go/rest"

"github.com/openshift/console/pkg/helm/actions/fake"
)

func TestProxy_IndexFile(t *testing.T) {
tests := []struct {
name string
indexFiles []string
mergedFile string
helmCRS []*unstructured.Unstructured
name string
indexFiles []string
mergedFile string
kubeVersion string
helmCRS []*unstructured.Unstructured
onlyCompatible bool
}{
{
name: "returned index file for configured helm repo",
indexFiles: []string{"testdata/sampleRepoIndex.yaml"},
mergedFile: "testdata/mergedSampleRepoIndex2.yaml",
},
{
name: "returned index file for configured helm repo contains only charts compatible for given cluster v1.16.0",
indexFiles: []string{"testdata/sampleRepoIndex3.yaml"},
mergedFile: "testdata/sampleRepoIndex3FilteredKube1-16-0.yaml",
kubeVersion: "v1.16.0",
onlyCompatible: true,
},
{
name: "returned index file for configured helm repo contains only charts compatible for given cluster v1.15.0",
indexFiles: []string{"testdata/sampleRepoIndex3.yaml"},
mergedFile: "testdata/sampleRepoIndexFilteredKube1-15-0.yaml",
kubeVersion: "v1.15.0",
onlyCompatible: true,
},
{
name: "returned index file for configured helm repo contains only charts compatible for given cluster v1.14.0",
indexFiles: []string{"testdata/sampleRepoIndex3.yaml"},
mergedFile: "testdata/sampleRepoIndexFilteredKube1-14-0.yaml",
kubeVersion: "v1.14.0",
onlyCompatible: true,
},
{
name: "return empty index file if not charts are compatible with given cluster",
indexFiles: []string{"testdata/incompatibleRepoIndex.yaml"},
mergedFile: "",
kubeVersion: "v1.15.0",
onlyCompatible: true,
},
{
name: "returned index file for configured helm repo contains only charts compatible for given pre-release cluster v1.20.0-beta2",
indexFiles: []string{"testdata/RepoIndexPreRelease.yaml"},
mergedFile: "testdata/mergedRepoIndexPreReleaseV1-20-0-beta2.yaml",
kubeVersion: "v1.20.0-beta.2",
onlyCompatible: true,
},
{
name: "returned index file for configured helm repo contains only charts compatible for given pre-release cluster v1.20.0-alpha2",
indexFiles: []string{"testdata/RepoIndexPreRelease.yaml"},
mergedFile: "testdata/mergedRepoIndexPreReleaseV1-20-0-alpha2.yaml",
kubeVersion: "v1.20.0-alpha.2",
onlyCompatible: true,
},
{
name: "returned index file for configured helm repo contains only charts compatible for given pre-release cluster v1.20.0-beta.2",
indexFiles: []string{"testdata/sampleRepoIndex3.yaml"},
mergedFile: "testdata/mergedRepoIndexPreReleaseV1-20-0-0.yaml",
kubeVersion: "v1.20.0-beta.2",
onlyCompatible: true,
},
{
name: "returned merged index file for configured helm repos",
indexFiles: []string{"testdata/azureRepoIndex.yaml", "testdata/sampleRepoIndex.yaml"},
Expand Down Expand Up @@ -82,18 +133,19 @@ func TestProxy_IndexFile(t *testing.T) {
t.Error(err)
}
}

fakeProxyOption := func(p *proxy) error {
p.dynamicClient = dynamicClient
return nil
}
p, err := New(func() (r *rest.Config, err error) {
return &rest.Config{}, nil
}, fakeProxyOption)
}, tt.kubeVersion, fakeProxyOption)
if err != nil {
t.Error(err)
}

indexFile, err := p.IndexFile()
indexFile, err := p.IndexFile(tt.onlyCompatible)
if err != nil {
t.Error(err)
}
Expand Down
11 changes: 6 additions & 5 deletions pkg/helm/chartproxy/repos_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,17 +3,18 @@ package chartproxy
import (
"errors"
"fmt"
helmrepo "helm.sh/helm/v3/pkg/repo"
"io/ioutil"
"net/http"
"net/http/httptest"
"reflect"
"testing"

helmrepo "helm.sh/helm/v3/pkg/repo"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"k8s.io/apimachinery/pkg/runtime"
fakeclient "k8s.io/client-go/dynamic/fake"
k8sfake "k8s.io/client-go/kubernetes/fake"
fakeclienttest "k8s.io/client-go/testing"
"net/http"
"net/http/httptest"
"reflect"
"testing"

"github.com/openshift/console/pkg/helm/actions/fake"
)
Expand Down
78 changes: 78 additions & 0 deletions pkg/helm/chartproxy/testdata/RepoIndexPreRelease.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
apiVersion: v1
entries:
ibm-cpq-prod:
- apiVersion: v2
appVersion: 10.0.0.6
created: "2020-07-15T16:32:34.607411698+05:30"
description: |-
IBM Sterling Configure, Price, Quote (CPQ) solution enables you to quickly configure, price, quote and order the right products and services.\n
To sell competitively in today’s multichannel environment, organizations need a way to easily manage product and service configuration and pricing rules that allow prospects, customers, sales staff, call center representatives and Business Partners to quickly find, configure and order the right products and services.\n
IBM® Sterling Configure, Price, Quote solution automates every step of the configure, price and quote process to help organizations easily create Web storefronts, offer dynamic catalog and pricing information, and direct customers and partners to find, configure and order the right products and services.\n
This enables you to transform how you sell complex products and services by removing the internal complexity of multi-tiered selling within your organization and with your partners.\n
Documentation\n
Additional information about installation can be found at https://ibm.biz/BdqqUS.
License\n
Sterling Configure Price Quote Software is licensed under https://ibm.biz/Bdqqgd which must be accepted during the install of the Product.
digest: 2d93c570ff47b27858901dacafc22fe64d868e9a8ecb4742867396f2912eded8
icon: https://raw.githubusercontent.com/IBM/charts/master/logo/ibm-oms-logo.png
keywords:
- cpq
- ifs
- sterling
- yantra
- order
- fulfillment
- om
- amd64
- framework
- Commercial
- RHOCP
- All items
- Languages
- Databases
- Middleware
- CI/CD
- Other
kubeVersion: '>=1.20.0-alpha.2'
maintainers:
- name: IBM
name: ibm-cpq-prod
response:
- https://redhat-developer.github.com/redhat-helm-charts/charts/ibm-cpq-prod-2.0.0.tgz
version: 2.0.0
ibm-operator-catalog-enablement:
- apiVersion: v2
appVersion: 1.0.0
created: "2020-07-15T16:32:34.697730479+05:30"
description: |
IBM Operator Catalog enablement deploys custom CatalogSources for operators
available to deploy and maintain IBM products. The operators are publically
available, but products they install may require purchase and entitlement keys.
Documentation
For additional details regarding installation see https://ibm.biz/operator-catalog-readme
License
By installing this catalog you accept the license terms https://www.apache.org/licenses/LICENSE-2.0
digest: 5a765d2f3f36f1b35c403a546df3590e3d6014cbf86bd36775f8febb54baa58c
home: http://ibm.biz/oprcatalog
icon: 
keywords:
- amd64
- deploy
- Catalog
- Commercial
- Limited
- Other
- RHOCP
kubeVersion: '>=1.20.0-beta.2'
maintainers:
- name: IBM
name: ibm-operator-catalog-enablement
type: application
response:
- https://redhat-developer.github.com/redhat-helm-charts/charts/ibm-operator-catalog-enablement-1.0.0.tgz
version: 1.0.0
generated: "2020-03-30T16:27:13.4275217-05:00"
Loading

0 comments on commit 90f1269

Please sign in to comment.