client/pkg/kn/commands/source/binding/binding.go

173 lines
5.0 KiB
Go

// Copyright © 2019 The Knative Authors
//
// Licensed 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 binding
import (
"fmt"
"strings"
"github.com/spf13/cobra"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime/schema"
"k8s.io/client-go/tools/clientcmd"
"knative.dev/eventing/pkg/client/clientset/versioned/typed/sources/v1alpha1"
duckv1 "knative.dev/pkg/apis/duck/v1"
duckv1beta1 "knative.dev/pkg/apis/duck/v1beta1"
"knative.dev/pkg/tracker"
"knative.dev/client/pkg/kn/commands"
sources_v1alpha1 "knative.dev/client/pkg/sources/v1alpha1"
"knative.dev/client/pkg/util"
)
// NewBindingCommand is the root command for all binding related commands
func NewBindingCommand(p *commands.KnParams) *cobra.Command {
bindingCmd := &cobra.Command{
Use: "binding",
Short: "Sink binding command group",
}
bindingCmd.AddCommand(NewBindingCreateCommand(p))
bindingCmd.AddCommand(NewBindingUpdateCommand(p))
bindingCmd.AddCommand(NewBindingDeleteCommand(p))
bindingCmd.AddCommand(NewBindingListCommand(p))
bindingCmd.AddCommand(NewBindingDescribeCommand(p))
return bindingCmd
}
// This var can be used to inject a factory for fake clients when doing
// tests.
var sinkBindingClientFactory func(config clientcmd.ClientConfig, namespace string) (sources_v1alpha1.KnSinkBindingClient, error)
func newSinkBindingClient(p *commands.KnParams, cmd *cobra.Command) (sources_v1alpha1.KnSinkBindingClient, error) {
namespace, err := p.GetNamespace(cmd)
if err != nil {
return nil, err
}
if sinkBindingClientFactory != nil {
config, err := p.GetClientConfig()
if err != nil {
return nil, err
}
return sinkBindingClientFactory(config, namespace)
}
clientConfig, err := p.RestConfig()
if err != nil {
return nil, err
}
client, err := v1alpha1.NewForConfig(clientConfig)
if err != nil {
return nil, err
}
return sources_v1alpha1.NewKnSourcesClient(client, namespace).SinkBindingClient(), nil
}
// Temporary conversions function until we move to duckv1
func toDuckV1(destination *duckv1beta1.Destination) *duckv1.Destination {
return &duckv1.Destination{
Ref: destination.Ref,
URI: destination.URI,
}
}
func toReference(subjectArg string, namespace string) (*tracker.Reference, error) {
parts := strings.SplitN(subjectArg, ":", 3)
if len(parts) < 3 {
return nil, fmt.Errorf("invalid subject argument '%s': not in format kind:api/version:nameOrSelector", subjectArg)
}
kind := parts[0]
gv, err := schema.ParseGroupVersion(parts[1])
if err != nil {
return nil, err
}
reference := &tracker.Reference{
APIVersion: gv.String(),
Kind: kind,
Namespace: namespace,
}
if !strings.Contains(parts[2], "=") {
reference.Name = parts[2]
} else {
selector, err := parseSelector(parts[2])
if err != nil {
return nil, err
}
reference.Selector = &metav1.LabelSelector{MatchLabels: selector}
}
return reference, nil
}
func parseSelector(labelSelector string) (map[string]string, error) {
selector := map[string]string{}
for _, p := range strings.Split(labelSelector, ",") {
keyValue := strings.SplitN(p, "=", 2)
if len(keyValue) != 2 {
return nil, fmt.Errorf("invalid subject label selector '%s', expected format: key1=value,key2=value", labelSelector)
}
selector[keyValue[0]] = keyValue[1]
}
return selector, nil
}
// subjectToString converts a reference to a string representation
func subjectToString(ref tracker.Reference) string {
ret := ref.Kind + ":" + ref.APIVersion
if ref.Name != "" {
return ret + ":" + ref.Name
}
var keyValues []string
selector := ref.Selector
if selector != nil {
for k, v := range selector.MatchLabels {
keyValues = append(keyValues, k+"="+v)
}
return ret + ":" + strings.Join(keyValues, ",")
}
return ret
}
// sinkToString prepares a sink for list output
// This is kept here until we have moved everything to duckv1 (currently the other sources
// are still on duckv1beta1)
func sinkToString(sink duckv1.Destination) string {
if sink.Ref != nil {
if sink.Ref.Kind == "Service" {
return fmt.Sprintf("svc:%s", sink.Ref.Name)
} else {
return fmt.Sprintf("%s:%s", sink.Ref.Kind, sink.Ref.Name)
}
}
if sink.URI != nil {
return sink.URI.String()
}
return ""
}
// updateCeOverrides updates the values of the --ce-override flags if given
func updateCeOverrides(bindingFlags bindingUpdateFlags, bindingBuilder *sources_v1alpha1.SinkBindingBuilder) error {
if bindingFlags.ceOverrides != nil {
ceOverrideMap, err := util.MapFromArray(bindingFlags.ceOverrides, "=")
if err != nil {
return err
}
bindingBuilder.AddCloudEventOverrides(ceOverrideMap)
}
return nil
}