mirror of https://github.com/knative/client.git
162 lines
4.8 KiB
Go
162 lines
4.8 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 flags
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/spf13/cobra"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
"knative.dev/pkg/apis"
|
|
duckv1 "knative.dev/pkg/apis/duck/v1"
|
|
|
|
clientdynamic "knative.dev/client/pkg/dynamic"
|
|
"knative.dev/client/pkg/kn/config"
|
|
)
|
|
|
|
type SinkFlags struct {
|
|
sink string
|
|
}
|
|
|
|
// AddWithFlagName configures sink flag with given flag name and a short flag name
|
|
// pass empty short flag name if you dont want to set one
|
|
func (i *SinkFlags) AddWithFlagName(cmd *cobra.Command, fname, short string) {
|
|
flag := "--" + fname
|
|
if short == "" {
|
|
cmd.Flags().StringVar(&i.sink, fname, "", "")
|
|
} else {
|
|
cmd.Flags().StringVarP(&i.sink, fname, short, "", "")
|
|
}
|
|
cmd.Flag(fname).Usage = "Addressable sink for events. " +
|
|
"You can specify a broker, channel, Knative service or URI. " +
|
|
"Examples: '" + flag + " broker:nest' for a broker 'nest', " +
|
|
"'" + flag + " channel:pipe' for a channel 'pipe', " +
|
|
"'" + flag + " ksvc:mysvc:mynamespace' for a Knative service 'mysvc' in another namespace 'mynamespace', " +
|
|
"'" + flag + " https://event.receiver.uri' for an URI with an 'http://' or 'https://' schema, " +
|
|
"'" + flag + " ksvc:receiver' or simply '" + flag + " receiver' for a Knative service 'receiver'. " +
|
|
"If a prefix is not provided, it is considered as a Knative service."
|
|
|
|
for _, p := range config.GlobalConfig.SinkMappings() {
|
|
//user configuration might override the default configuration
|
|
sinkMappings[p.Prefix] = schema.GroupVersionResource{
|
|
Resource: p.Resource,
|
|
Group: p.Group,
|
|
Version: p.Version,
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add configures sink flag with name 'sink' amd short name 's'
|
|
func (i *SinkFlags) Add(cmd *cobra.Command) {
|
|
i.AddWithFlagName(cmd, "sink", "s")
|
|
}
|
|
|
|
// sinkPrefixes maps prefixes used for sinks to their GroupVersionResources.
|
|
var sinkMappings = map[string]schema.GroupVersionResource{
|
|
"broker": {
|
|
Resource: "brokers",
|
|
Group: "eventing.knative.dev",
|
|
Version: "v1beta1",
|
|
},
|
|
// Shorthand alias for service
|
|
"ksvc": {
|
|
Resource: "services",
|
|
Group: "serving.knative.dev",
|
|
Version: "v1",
|
|
},
|
|
"channel": {
|
|
Resource: "channels",
|
|
Group: "messaging.knative.dev",
|
|
Version: "v1beta1",
|
|
},
|
|
}
|
|
|
|
// ResolveSink returns the Destination referred to by the flags in the acceptor.
|
|
// It validates that any object the user is referring to exists.
|
|
func (i *SinkFlags) ResolveSink(knclient clientdynamic.KnDynamicClient, namespace string) (*duckv1.Destination, error) {
|
|
client := knclient.RawClient()
|
|
if i.sink == "" {
|
|
return nil, nil
|
|
}
|
|
prefix, name, ns := parseSink(i.sink)
|
|
if prefix == "" {
|
|
// URI target
|
|
uri, err := apis.ParseURL(name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &duckv1.Destination{URI: uri}, nil
|
|
}
|
|
typ, ok := sinkMappings[prefix]
|
|
if !ok {
|
|
if prefix == "svc" || prefix == "service" {
|
|
return nil, fmt.Errorf("unsupported sink prefix: '%s', please use prefix 'ksvc' for knative service", prefix)
|
|
}
|
|
return nil, fmt.Errorf("unsupported sink prefix: '%s'", prefix)
|
|
}
|
|
if ns != "" {
|
|
namespace = ns
|
|
}
|
|
obj, err := client.Resource(typ).Namespace(namespace).Get(context.TODO(), name, metav1.GetOptions{})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
destination := &duckv1.Destination{
|
|
Ref: &duckv1.KReference{
|
|
Kind: obj.GetKind(),
|
|
APIVersion: obj.GetAPIVersion(),
|
|
Name: obj.GetName(),
|
|
Namespace: namespace,
|
|
},
|
|
}
|
|
return destination, nil
|
|
}
|
|
|
|
// parseSink takes the string given by the user into the prefix, name and namespace of
|
|
// the object. If the user put a URI instead, the prefix is empty and the name
|
|
// is the whole URI.
|
|
func parseSink(sink string) (string, string, string) {
|
|
parts := strings.SplitN(sink, ":", 3)
|
|
switch {
|
|
case len(parts) == 1:
|
|
return "ksvc", parts[0], ""
|
|
case parts[0] == "http" || parts[0] == "https":
|
|
return "", sink, ""
|
|
case len(parts) == 3:
|
|
return parts[0], parts[1], parts[2]
|
|
default:
|
|
return parts[0], parts[1], ""
|
|
}
|
|
}
|
|
|
|
// SinkToString prepares a sink for list output
|
|
func SinkToString(sink duckv1.Destination) string {
|
|
if sink.Ref != nil {
|
|
if sink.Ref.Kind == "Service" {
|
|
return fmt.Sprintf("ksvc:%s", sink.Ref.Name)
|
|
} else {
|
|
return fmt.Sprintf("%s:%s", strings.ToLower(sink.Ref.Kind), sink.Ref.Name)
|
|
}
|
|
}
|
|
if sink.URI != nil {
|
|
return sink.URI.String()
|
|
}
|
|
return ""
|
|
}
|