pkg/util/olm/operator.go (318 lines of code) (raw):
/*
Licensed to the Apache Software Foundation (ASF) under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The ASF licenses this file to You under the Apache License, Version 2.0
(the "License"); you may not use this file except in compliance with
the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package olm
import (
"context"
"fmt"
"strings"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/kubectl/pkg/cmd/set/env"
k8sclient "k8s.io/client-go/kubernetes"
runtime "sigs.k8s.io/controller-runtime/pkg/client"
"github.com/operator-framework/api/pkg/operators"
operatorsv1 "github.com/operator-framework/api/pkg/operators/v1"
operatorsv1alpha1 "github.com/operator-framework/api/pkg/operators/v1alpha1"
"github.com/apache/camel-k/v2/pkg/client"
"github.com/apache/camel-k/v2/pkg/util/kubernetes"
"github.com/apache/camel-k/v2/pkg/util/openshift"
)
// The following properties can be overridden at build time via ldflags
// DefaultOperatorName is the Camel K operator name in OLM.
var DefaultOperatorName = "camel-k"
// DefaultPackage is the Camel K package in OLM.
var DefaultPackage = "camel-k"
// DefaultChannel is the distribution channel in Operator Hub.
var DefaultChannel = "stable"
// DefaultSource is the name of the operator source where the operator is published.
var DefaultSource = "community-operators"
// DefaultSourceNamespace is the namespace of the operator source.
var DefaultSourceNamespace = "openshift-marketplace"
// DefaultStartingCSV contains the specific version to install.
var DefaultStartingCSV = ""
// DefaultGlobalNamespace indicates a namespace containing an OperatorGroup that enables the operator to watch all namespaces.
// It will be used in global installation mode.
var DefaultGlobalNamespace = "openshift-operators"
// Options contains information about an operator in OLM.
type Options struct {
OperatorName string
Package string
Channel string
Source string
SourceNamespace string
StartingCSV string
GlobalNamespace string
}
// IsOperatorInstalled tells if a OLM CSV or a Subscription is already installed in the namespace.
func IsOperatorInstalled(ctx context.Context, client client.Client, namespace string, global bool, options Options) (bool, error) {
options, err := fillDefaults(options, client)
if err != nil {
return false, err
}
// CSV is present in current namespace for both local and global installation modes
if csv, err := findCSV(ctx, client, namespace, options); err != nil {
return false, err
} else if csv != nil {
return true, nil
}
// A subscription may indicate an in-progress installation
if sub, err := findSubscription(ctx, client, namespace, global, options); err != nil {
return false, err
} else if sub != nil {
return true, nil
}
return false, nil
}
// HasPermissionToInstall checks if the current user/serviceaccount has the right permissions to install camel k via OLM.
func HasPermissionToInstall(ctx context.Context, client client.Client, namespace string, global bool, options Options) (bool, error) {
if ok, err := kubernetes.CheckPermission(ctx, client, operatorsv1alpha1.GroupName, "clusterserviceversions", namespace, options.Package, "list"); err != nil {
return false, err
} else if !ok {
return false, nil
}
targetNamespace := namespace
if global {
targetNamespace = options.GlobalNamespace
}
if ok, err := kubernetes.CheckPermission(ctx, client, operatorsv1alpha1.GroupName, "subscriptions", targetNamespace, options.Package, "create"); err != nil {
return false, err
} else if !ok {
return false, nil
}
if installed, err := IsOperatorInstalled(ctx, client, namespace, global, options); err != nil {
return false, err
} else if installed {
return true, nil
}
if !global {
if ok, err := kubernetes.CheckPermission(ctx, client, operators.GroupName, "operatorgroups", namespace, options.Package, "list"); err != nil {
return false, err
} else if !ok {
return false, nil
}
group, err := findOperatorGroup(ctx, client, namespace)
if err != nil {
return false, err
}
if group == nil {
if ok, err := kubernetes.CheckPermission(ctx, client, operators.GroupName, "operatorgroups", namespace, options.Package, "create"); err != nil {
return false, err
} else if !ok {
return false, nil
}
}
}
return true, nil
}
// Install creates a subscription for the OLM package.
func Install(ctx context.Context, client client.Client, namespace string, global bool, options Options, collection *kubernetes.Collection,
tolerations []string, nodeSelectors []string, resourcesRequirements []string, envVars []string) (bool, error) {
options, err := fillDefaults(options, client)
if err != nil {
return false, err
}
if installed, err := IsOperatorInstalled(ctx, client, namespace, global, options); err != nil {
return false, err
} else if installed {
// Already installed
return false, nil
}
targetNamespace := namespace
if global {
targetNamespace = options.GlobalNamespace
}
sub := operatorsv1alpha1.Subscription{
ObjectMeta: v1.ObjectMeta{
Name: options.Package,
Namespace: targetNamespace,
},
Spec: &operatorsv1alpha1.SubscriptionSpec{
CatalogSource: options.Source,
CatalogSourceNamespace: options.SourceNamespace,
Package: options.Package,
Channel: options.Channel,
StartingCSV: options.StartingCSV,
InstallPlanApproval: operatorsv1alpha1.ApprovalAutomatic,
Config: &operatorsv1alpha1.SubscriptionConfig{},
},
}
// Additional configuration
err = maybeSetTolerations(&sub, tolerations)
if err != nil {
return false, fmt.Errorf("could not set tolerations: %w", err)
}
err = maybeSetNodeSelectors(&sub, nodeSelectors)
if err != nil {
return false, fmt.Errorf("could not set node selectors: %w", err)
}
err = maybeSetResourcesRequirements(&sub, resourcesRequirements)
if err != nil {
return false, fmt.Errorf("could not set resources requirements: %w", err)
}
err = maybeSetEnvVars(&sub, envVars)
if err != nil {
return false, fmt.Errorf("could not set environment variables: %w", err)
}
if collection != nil {
collection.Add(&sub)
} else if err := client.Create(ctx, &sub); err != nil {
return false, err
}
if !global {
group, err := findOperatorGroup(ctx, client, namespace)
if err != nil {
return false, err
}
if group == nil {
group = &operatorsv1.OperatorGroup{
ObjectMeta: v1.ObjectMeta{
Namespace: namespace,
GenerateName: fmt.Sprintf("%s-", namespace),
},
Spec: operatorsv1.OperatorGroupSpec{
TargetNamespaces: []string{namespace},
},
}
if collection != nil {
collection.Add(group)
} else if err := client.Create(ctx, group); err != nil {
return false, fmt.Errorf("namespace %s has no operator group defined and "+
"current user is not able to create it. "+
"Make sure you have the right roles to install operators from OLM"+": %w", namespace, err)
}
}
}
return true, nil
}
func maybeSetTolerations(sub *operatorsv1alpha1.Subscription, tolArray []string) error {
if tolArray != nil {
tolerations, err := kubernetes.NewTolerations(tolArray)
if err != nil {
return err
}
if sub == nil {
panic("sub is nil")
}
if sub.Spec == nil {
panic("sub.Spec is nil")
}
if sub.Spec.Config == nil {
panic("sub.Spec.Config is nil")
}
sub.Spec.Config.Tolerations = tolerations
}
return nil
}
func maybeSetNodeSelectors(sub *operatorsv1alpha1.Subscription, nsArray []string) error {
if nsArray != nil {
nodeSelectors, err := kubernetes.NewNodeSelectors(nsArray)
if err != nil {
return err
}
sub.Spec.Config.NodeSelector = nodeSelectors
}
return nil
}
func maybeSetResourcesRequirements(sub *operatorsv1alpha1.Subscription, reqArray []string) error {
if reqArray != nil {
resourcesReq, err := kubernetes.NewResourceRequirements(reqArray)
if err != nil {
return err
}
sub.Spec.Config.Resources = &resourcesReq
}
return nil
}
func maybeSetEnvVars(sub *operatorsv1alpha1.Subscription, envVars []string) error {
if envVars != nil {
vars, _, _, err := env.ParseEnv(envVars, nil)
if err != nil {
return err
}
sub.Spec.Config.Env = vars
}
return nil
}
// Uninstall removes CSV and subscription from the namespace.
func Uninstall(ctx context.Context, client client.Client, namespace string, global bool, options Options) error {
options, err := fillDefaults(options, client)
if err != nil {
return err
}
sub, err := findSubscription(ctx, client, namespace, global, options)
if err != nil {
return err
}
if sub != nil {
if err := client.Delete(ctx, sub); err != nil {
return err
}
}
csv, err := findCSV(ctx, client, namespace, options)
if err != nil {
return err
}
if csv != nil {
if err := client.Delete(ctx, csv); err != nil {
return err
}
}
return nil
}
func findSubscription(ctx context.Context, client client.Client, namespace string, global bool, options Options) (*operatorsv1alpha1.Subscription, error) {
subNamespace := namespace
if global {
// In case of global installation, global subscription must be removed
subNamespace = options.GlobalNamespace
}
subscriptionList := operatorsv1alpha1.SubscriptionList{}
if err := client.List(ctx, &subscriptionList, runtime.InNamespace(subNamespace)); err != nil {
return nil, err
}
for _, item := range subscriptionList.Items {
if item.Spec.Package == options.Package {
return &item, nil
}
}
return nil, nil
}
func findCSV(ctx context.Context, client client.Client, namespace string, options Options) (*operatorsv1alpha1.ClusterServiceVersion, error) {
csvList := operatorsv1alpha1.ClusterServiceVersionList{}
if err := client.List(ctx, &csvList, runtime.InNamespace(namespace)); err != nil {
return nil, err
}
for _, item := range csvList.Items {
if strings.HasPrefix(item.Name, options.OperatorName) {
return &item, nil
}
}
return nil, nil
}
func findOperatorGroup(ctx context.Context, client client.Client, namespace string) (*operatorsv1.OperatorGroup, error) {
opGroupList := operatorsv1.OperatorGroupList{}
if err := client.List(ctx, &opGroupList, runtime.InNamespace(namespace)); err != nil {
return nil, err
}
if len(opGroupList.Items) > 0 {
return &opGroupList.Items[0], nil
}
return nil, nil
}
func fillDefaults(o Options, client k8sclient.Interface) (Options, error) {
if o.OperatorName == "" {
o.OperatorName = DefaultOperatorName
}
if o.Package == "" {
o.Package = DefaultPackage
}
if o.Channel == "" {
o.Channel = DefaultChannel
}
if o.StartingCSV == "" {
o.StartingCSV = DefaultStartingCSV
}
isOCP, err := openshift.IsOpenShift(client)
if err != nil {
return o, err
}
if isOCP {
if o.Source == "" {
o.Source = DefaultSource
}
if o.SourceNamespace == "" {
o.SourceNamespace = DefaultSourceNamespace
}
if o.GlobalNamespace == "" {
o.GlobalNamespace = DefaultGlobalNamespace
}
} else {
// Use a different set of defaults value
if o.Source == "" {
o.Source = "operatorhubio-catalog"
}
if o.SourceNamespace == "" {
o.SourceNamespace = "olm"
}
if o.GlobalNamespace == "" {
o.GlobalNamespace = "operators"
}
}
return o, nil
}