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

Reconcile OperatorSource type #4

Merged
merged 1 commit into from
Sep 26, 2018
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
311 changes: 286 additions & 25 deletions Gopkg.lock

Large diffs are not rendered by default.

4 changes: 4 additions & 0 deletions Gopkg.toml
Original file line number Diff line number Diff line change
Expand Up @@ -60,3 +60,7 @@ required = [
[[constraint]]
name = "github.com/operator-framework/operator-lifecycle-manager"
version = "0.7.1"

[[constraint]]
branch = "master"
name = "github.com/operator-framework/go-appr"
8 changes: 5 additions & 3 deletions cmd/marketplace-operator/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -31,10 +31,12 @@ func main() {
if err != nil {
logrus.Fatalf("failed to get watch namespace: %v", err)
}
resyncPeriod := time.Duration(5) * time.Second
logrus.Infof("Watching %s, %s, %s", resource, catalogSourceConfigKind, namespace)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Remove blank lines.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

resyncPeriod is shared by the two types here, hence it gets its own spacing.

// No reason to resync until we implement updating of CatalogSourceConfig CRs
sdk.Watch(resource, catalogSourceConfigKind, namespace, 0)
resyncPeriod := time.Duration(0) * time.Second

logrus.Infof("Watching %s, %s, %s", resource, catalogSourceConfigKind, namespace)
sdk.Watch(resource, catalogSourceConfigKind, namespace, resyncPeriod)

operatorSourceKind := "OperatorSource"
logrus.Infof("Watching %s, %s, %s, %d", resource, operatorSourceKind, namespace, resyncPeriod)
Expand Down
5 changes: 4 additions & 1 deletion deploy/operatorsource.cr.yaml
Original file line number Diff line number Diff line change
@@ -1,4 +1,7 @@
apiVersion: "marketplace.redhat.com/v1alpha1"
kind: "OperatorSource"
metadata:
name: "operatorsource-example"
name: "localhost"
spec:
type: appregistry
endpoint: "http://localhost:5000/cnr"
4 changes: 3 additions & 1 deletion pkg/apis/marketplace/v1alpha1/register.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,9 @@ var (
SchemeBuilder = runtime.NewSchemeBuilder(addKnownTypes)
AddToScheme = SchemeBuilder.AddToScheme
// SchemeGroupVersion is the group version used to register these objects.
SchemeGroupVersion = schema.GroupVersion{Group: groupName, Version: version}
SchemeGroupVersion = schema.GroupVersion{Group: groupName, Version: version}
OperatorSourceKind = "OperatorSource"
CatalogSourceConfigKind = "CatalogSourceConfig"
)

func init() {
Expand Down
7 changes: 6 additions & 1 deletion pkg/apis/marketplace/v1alpha1/types.go
Original file line number Diff line number Diff line change
Expand Up @@ -54,8 +54,13 @@ type OperatorSource struct {
}

type OperatorSourceSpec struct {
// Fill me
// Type of operator source
Type string `json:"type"`

// Endpoint points to the URL from where operator manifests can be fetched
Endpoint string `json:"endpoint"`
}

type OperatorSourceStatus struct {
// Fill me
}
71 changes: 71 additions & 0 deletions pkg/appregistry/adapter.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
package appregistry

import (
"bytes"

appr "github.com/operator-framework/go-appr/appregistry"
appr_blobs "github.com/operator-framework/go-appr/appregistry/blobs"
appr_package "github.com/operator-framework/go-appr/appregistry/package_appr"
appr_models "github.com/operator-framework/go-appr/models"
)

const (
mediaType string = "helm"
)

// This interface (internal to this package) encapsulates nitty gritty details of go-appr client bindings
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Comment could have been a bit more clear on why 3 methods.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That's all we need for this iteration. Do you want more clarity on why this interface exists? I don't think we need to explain why an interface has N number of methods.

type apprApiAdapter interface {
// ListPackages returns a list of package(s) available to the user
ListPackages() (appr_models.Packages, error)

// GetPackageMetadata returns metadata associated with a given package
GetPackageMetadata(namespace string, repository string, release string) (*appr_models.Package, error)

// DownloadOperatorManifest downloads the blob associated with a given digest that directly corresponds to a package release
DownloadOperatorManifest(namespace string, repository string, digest string) ([]byte, error)
}

type apprApiAdapterImpl struct {
client *appr.Appregistry
}

func (a *apprApiAdapterImpl) ListPackages() (appr_models.Packages, error) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nit: Please add docs to functions. It rarely gets filled in later. This applies to all places where docs have not be filled.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please add a task to add docs if it is not going to be done as part of this commit.

params := appr_package.NewListPackagesParams()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Don't we allow listing packages within a namespace? I think we should allow some filtering here

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I already have a new PR in review that allows to filter by namespace.


packages, err := a.client.PackageAppr.ListPackages(params)
if err != nil {
return nil, err
}

return packages.Payload, nil
}

func (a *apprApiAdapterImpl) GetPackageMetadata(namespace string, repository string, release string) (*appr_models.Package, error) {
params := appr_package.NewShowPackageParams().
WithNamespace(namespace).
WithPackage(repository).
WithRelease(release).
WithMediaType(mediaType)

pkg, err := a.client.PackageAppr.ShowPackage(params)
if err != nil {
return nil, err
}

return pkg.Payload, nil
}

func (a *apprApiAdapterImpl) DownloadOperatorManifest(namespace string, repository string, digest string) ([]byte, error) {
params := appr_blobs.NewPullBlobParams().
WithNamespace(namespace).
WithPackage(repository).
WithDigest(digest)

writer := &bytes.Buffer{}
_, err := a.client.Blobs.PullBlob(params, writer)
if err != nil {
return nil, err
}

return writer.Bytes(), nil
}
48 changes: 48 additions & 0 deletions pkg/appregistry/appregistry.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
package appregistry
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This whole file could be named as factory.go

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, good point. I thought about it. I also like the idea of a {package name}.go file which has the entry level function for the given package. NewFactory is the entry level function one must call to use the appregistry package.


import (
"net/url"

"github.com/go-openapi/runtime"
httptransport "github.com/go-openapi/runtime/client"
"github.com/go-openapi/strfmt"
apprclient "github.com/operator-framework/go-appr/appregistry"
)

// NewClientFactory return a factory which can be used to instantiate a new appregistry client
func NewClientFactory() ClientFactory {
return &factory{}
}

type ClientFactory interface {
// New returns a new instance of appregistry Client from given source and type
New(sourceType, source string) (Client, error)
}

// Client exposes the functionality of app registry server
type Client interface {
// RetrieveAll retrieves all visible packages from the given source
RetrieveAll() ([]*OperatorMetadata, error)

// RetrieveOneretrieves a given package from the source
RetrieveOne(name, release string) (*OperatorMetadata, error)
}

type factory struct{}

func (f *factory) New(sourceType, source string) (Client, error) {
u, err := url.Parse(source)
if err != nil {
return nil, err
}

transport := httptransport.New(u.Host, u.Path, []string{u.Scheme})
transport.Consumers["application/x-gzip"] = runtime.ByteStreamConsumer()
c := apprclient.New(transport, strfmt.Default)

return &client{
adapter: &apprApiAdapterImpl{client: c},
decoder: &blobDecoderImpl{},
unmarshaller: &blobUnmarshallerImpl{},
}, nil
}
22 changes: 22 additions & 0 deletions pkg/appregistry/appregistry_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
package appregistry_test
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why a new package for testing?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am treating this test as a black box test, hence a new package for test(s) and explicit import of package being tested.


import (
"testing"

"github.com/stretchr/testify/assert"

"github.com/operator-framework/operator-marketplace/pkg/appregistry"
"github.com/stretchr/testify/require"
)

func TestRetrieveAll(t *testing.T) {
factory := appregistry.NewClientFactory()

client, err := factory.New("appregistry", "http://localhost:5000/cnr")
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not sure, if the assumption is we have some service running on 5000? Or did you mock it somewhere?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This test is hanging around here for debugging purposes. It will be replaced by a unit test.

require.NoError(t, err)

packages, err := client.RetrieveAll()

assert.NoError(t, err)
assert.NotNil(t, packages)
}
105 changes: 105 additions & 0 deletions pkg/appregistry/client.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
package appregistry

import (
"errors"
"fmt"
"strings"
)

// OperatorMetadata encapsulates operator metadata and manifest assocated with a package
type OperatorMetadata struct {
// Namespace is the namespace in app registry server under which the package is hosted.
Namespace string

// Repository is the repository name for the specified package in app registry
Repository string

// Release represents the release or version number of the given package
Release string

// Digest is the sha256 hash value that uniquely corresponds to the blob associated with the release
Digest string

// Manifest encapsulates operator manifest
Manifest *Manifest
}

func (om *OperatorMetadata) ID() string {
return fmt.Sprintf("%s/%s", om.Namespace, om.Repository)
}

type client struct {
adapter apprApiAdapter
decoder blobDecoder
unmarshaller blobUnmarshaller
}

func (c *client) RetrieveAll() ([]*OperatorMetadata, error) {
packages, err := c.adapter.ListPackages()
if err != nil {
return nil, err
}

list := make([]*OperatorMetadata, len(packages))
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

operatorMetaDataList probably would have made more sense.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have to implement a operatorMetaDataList. This is not a k8s type.

for i, pkg := range packages {
manifest, err := c.RetrieveOne(pkg.Name, pkg.Default)
if err != nil {
return nil, err
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you want to return error, if retrieving one package is erroring? It could that one package is misbehaving, probably aggregated errorlist would make more sense.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is a rudimentary version geared toward the proof of concept. I am working on prod ready reconciliation process now.

}

list[i] = manifest
}

return list, nil
}

func (c *client) RetrieveOne(name, release string) (*OperatorMetadata, error) {
namespace, repository, err := split(name)
if err != nil {
return nil, err
}

metadata, err := c.adapter.GetPackageMetadata(namespace, repository, release)
if err != nil {
return nil, err
}

digest := metadata.Content.Digest
blob, err := c.adapter.DownloadOperatorManifest(namespace, repository, digest)
if err != nil {
return nil, err
}

decoded, err := c.decoder.Decode(blob)
if err != nil {
return nil, err
}

manifest, err := c.unmarshaller.Unmarshal(decoded)
if err != nil {
return nil, err
}

om := &OperatorMetadata{
Namespace: namespace,
Repository: repository,
Release: release,
Manifest: manifest,
Digest: digest,
}

return om, nil
}

func split(name string) (namespace string, repository string, err error) {
// we expect package name to comply to this format - {namespace}/{repository}
split := strings.Split(name, "/")
if len(split) != 2 {
return "", "", errors.New(fmt.Sprintf("package name should be specified in this format {namespace}/{repository}"))
}

namespace = split[0]
repository = split[1]

return namespace, repository, nil
}
61 changes: 61 additions & 0 deletions pkg/appregistry/client_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
package appregistry

import (
"fmt"
"testing"

"github.com/stretchr/testify/assert"

"github.com/golang/mock/gomock"
appr_models "github.com/operator-framework/go-appr/models"
)

func TestRetrieveOne_PackageExists_SuccessExpected(t *testing.T) {
controller := gomock.NewController(t)
defer controller.Finish()

adapter := NewMockapprApiAdapter(controller)
decoder := NewMockblobDecoder(controller)
unmarshaller := NewMockblobUnmarshaller(controller)

client := client{
adapter: adapter,
decoder: decoder,
unmarshaller: unmarshaller,
}

namespace := "redhat"
repository := "foo"
release := "1.0"
digest := "abcdefgh"

pkg := &appr_models.Package{Content: &appr_models.OciDescriptor{
Digest: digest,
}}
adapter.EXPECT().GetPackageMetadata(namespace, repository, release).Return(pkg, nil).Times(1)

blobExpected := []byte{'e', 'n', 'c', 'o', 'd', 'e', 'd'}
adapter.EXPECT().DownloadOperatorManifest(namespace, repository, digest).Return(blobExpected, nil).Times(1)

decodedExpected := []byte{'d', 'e', 'c', 'o', 'd', 'e', 'd'}
decoder.EXPECT().Decode(blobExpected).Return(decodedExpected, nil).Times(1)

manifestExpected := &Manifest{
Publisher: "redhat",
Data: Data{
CRDs: "my crds",
CSVs: "my csvs",
Packages: "my packages",
},
}
unmarshaller.EXPECT().Unmarshal(decodedExpected).Return(manifestExpected, nil)

metadata, err := client.RetrieveOne(fmt.Sprintf("%s/%s", namespace, repository), release)

assert.NoError(t, err)
assert.Equal(t, namespace, metadata.Namespace)
assert.Equal(t, repository, metadata.Repository)
assert.Equal(t, release, metadata.Release)
assert.Equal(t, digest, metadata.Digest)
assert.Equal(t, manifestExpected, metadata.Manifest)
}
Loading