timflannagan / rukpak

Rukpak runs in a Kubernetes cluster and defines an API for installing cloud native bundle content
Apache License 2.0
0 stars 0 forks source link

refactor NewDefaultUnpacker due to growing parameter list #61

Open github-actions[bot] opened 2 years ago

github-actions[bot] commented 2 years ago

https://github.com/timflannagan/rukpak/blob/2166b326f600c76b73cbb907e41fddf0f81dbae9/internal/source/unpacker.go#L103


package source

import (
    "context"
    "crypto/tls"
    "crypto/x509"
    "fmt"
    "io/fs"
    "net/http"
    "time"

    "k8s.io/client-go/kubernetes"
    ctrl "sigs.k8s.io/controller-runtime"

    rukpakv1alpha1 "github.com/operator-framework/rukpak/api/v1alpha1"
)

const (
    // uploadClientTimeout is the timeout to be used with http connections to upload manager.
    uploadClientTimeout = time.Second * 10
)

// Unpacker unpacks bundle content, either synchronously or asynchronously and
// returns a Result, which conveys information about the progress of unpacking
// the bundle content.
//
// If a Source unpacks content asynchronously, it should register one or more
// watches with a controller to ensure that Bundles referencing this source
// can be reconciled as progress updates are available.
//
// For asynchronous Sources, multiple calls to Unpack should be made until the
// returned result includes state StateUnpacked.
//
// NOTE: A source is meant to be agnostic to specific bundle formats and
// specifications. A source should treat a bundle root directory as an opaque
// file tree and delegate bundle format concerns to bundle parsers.
type Unpacker interface {
    Unpack(context.Context, *rukpakv1alpha1.Bundle) (*Result, error)
}

// Result conveys progress information about unpacking bundle content.
type Result struct {
    // Bundle contains the full filesystem of a bundle's root directory.
    Bundle fs.FS

    // ResolvedSource is a reproducible view of a Bundle's Source.
    // When possible, source implementations should return a ResolvedSource
    // that pins the Source such that future fetches of the bundle content can
    // be guaranteed to fetch the exact same bundle content as the original
    // unpack.
    //
    // For example, resolved image sources should reference a container image
    // digest rather than an image tag, and git sources should reference a
    // commit hash rather than a branch or tag.
    ResolvedSource *rukpakv1alpha1.BundleSource

    // State is the current state of unpacking the bundle content.
    State State

    // Message is contextual information about the progress of unpacking the
    // bundle content.
    Message string
}

type State string

const (
    // StatePending conveys that a request for unpacking a bundle has been
    // acknowledged, but not yet started.
    StatePending State = "Pending"

    // StateUnpacking conveys that the source is currently unpacking a bundle.
    // This state should be used when the bundle contents are being downloaded
    // and processed.
    StateUnpacking State = "Unpacking"

    // StateUnpacked conveys that the bundle has been successfully unpacked.
    StateUnpacked State = "Unpacked"
)

type unpacker struct {
    sources map[rukpakv1alpha1.SourceType]Unpacker
}

// NewUnpacker returns a new composite Source that unpacks bundles using the source
// mapping provided by the configured sources.
func NewUnpacker(sources map[rukpakv1alpha1.SourceType]Unpacker) Unpacker {
    return &unpacker{sources: sources}
}

func (s *unpacker) Unpack(ctx context.Context, bundle *rukpakv1alpha1.Bundle) (*Result, error) {
    source, ok := s.sources[bundle.Spec.Source.Type]
    if !ok {
        return nil, fmt.Errorf("source type %q not supported", bundle.Spec.Source.Type)
    }
    return source.Unpack(ctx, bundle)
}

// NewDefaultUnpacker returns a new composite Source that unpacks bundles using
// a default source mapping with built-in implementations of all of the supported
// source types.
//
// TODO: refactor NewDefaultUnpacker due to growing parameter list
func NewDefaultUnpacker(mgr ctrl.Manager, namespace, unpackImage string, baseUploadManagerURL string, rootCAs *x509.CertPool) (Unpacker, error) {
    cfg := mgr.GetConfig()
    kubeClient, err := kubernetes.NewForConfig(cfg)
    if err != nil {
        return nil, err
    }
    httpTransport := http.DefaultTransport.(*http.Transport).Clone()
    if httpTransport.TLSClientConfig == nil {
        httpTransport.TLSClientConfig = &tls.Config{}
    }
    httpTransport.TLSClientConfig.RootCAs = rootCAs
    return NewUnpacker(map[rukpakv1alpha1.SourceType]Unpacker{
        rukpakv1alpha1.SourceTypeImage: &Image{
            Client:       mgr.GetClient(),
            KubeClient:   kubeClient,
            PodNamespace: namespace,
            UnpackImage:  unpackImage,
        },
        rukpakv1alpha1.SourceTypeGit: &Git{
            Reader:          mgr.GetAPIReader(),
            SecretNamespace: namespace,
        },
        rukpakv1alpha1.SourceTypeLocal: &Local{
            Client: mgr.GetClient(),
            reader: mgr.GetAPIReader(),
        },
        rukpakv1alpha1.SourceTypeUpload: &Upload{
            baseDownloadURL: baseUploadManagerURL,
            bearerToken:     mgr.GetConfig().BearerToken,
            client:          http.Client{Timeout: uploadClientTimeout, Transport: httpTransport},
        },
    }), nil
}
github-actions[bot] commented 1 year ago

This issue has become stale because it has been open 60 days with no activity. The maintainers of this repo will remove this label during issue triage or it will be removed automatically after an update. Adding the lifecycle/frozen label will cause this issue to ignore lifecycle events.