This is an automated email from the ASF dual-hosted git repository.

tsato pushed a commit to branch main
in repository https://gitbox.apache.org/repos/asf/camel-k.git


The following commit(s) were added to refs/heads/main by this push:
     new 37a9fadb9 feat(cli): Add add-repo command to add a repo for custom 
Kamelet catalog
37a9fadb9 is described below

commit 37a9fadb90003aa2344d6d700d90bf6a1bef3849
Author: Nicolas Filotto <nfilo...@talend.com>
AuthorDate: Thu Sep 8 19:18:31 2022 +0200

    feat(cli): Add add-repo command to add a repo for custom Kamelet catalog
---
 .../files/TimerCustomKameletIntegration.java       |  28 ++++++
 .../kamelets/timer-custom-source.kamelet.yaml      |  69 +++++++++++++
 e2e/global/common/kamelet_test.go                  |  43 +++++---
 pkg/cmd/kamelet.go                                 |   1 +
 pkg/cmd/kamelet_add_repo.go                        | 111 +++++++++++++++++++++
 pkg/cmd/kamelet_add_repo_test.go                   |  90 +++++++++++++++++
 pkg/cmd/kamelet_get.go                             |   5 +-
 pkg/cmd/kit_get.go                                 |   5 +-
 8 files changed, 332 insertions(+), 20 deletions(-)

diff --git a/e2e/global/common/files/TimerCustomKameletIntegration.java 
b/e2e/global/common/files/TimerCustomKameletIntegration.java
new file mode 100644
index 000000000..9f749d834
--- /dev/null
+++ b/e2e/global/common/files/TimerCustomKameletIntegration.java
@@ -0,0 +1,28 @@
+/*
+ * 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.
+ */
+
+import java.lang.Exception;
+import java.lang.Override;
+import org.apache.camel.builder.RouteBuilder;
+
+public class TimerCustomKameletIntegration extends RouteBuilder {
+    @Override
+    public void configure() throws Exception {
+        from("kamelet:timer-custom-source?message=great%20message")
+            .to("log:info");
+    }
+}
diff --git a/e2e/global/common/files/kamelets/timer-custom-source.kamelet.yaml 
b/e2e/global/common/files/kamelets/timer-custom-source.kamelet.yaml
new file mode 100644
index 000000000..a47e62e0d
--- /dev/null
+++ b/e2e/global/common/files/kamelets/timer-custom-source.kamelet.yaml
@@ -0,0 +1,69 @@
+# ---------------------------------------------------------------------------
+# 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.
+# ---------------------------------------------------------------------------
+
+apiVersion: camel.apache.org/v1alpha1
+kind: Kamelet
+metadata:
+  name: timer-custom-source
+  annotations:
+    camel.apache.org/kamelet.support.level: "Preview"
+    camel.apache.org/catalog.version: "main-SNAPSHOT"
+    camel.apache.org/kamelet.icon: 

 [...]
+    camel.apache.org/provider: "Apache Software Foundation"
+    camel.apache.org/kamelet.group: "Timer"
+  labels:
+    camel.apache.org/kamelet.type: source
+    camel.apache.org/kamelet.verified: "true"
+spec:
+  definition:
+    title: Timer Custom Source
+    description: Produces periodic events with a custom payload in a custom 
way.
+    required:
+      - message
+    type: object
+    properties:
+      period:
+        title: Period
+        description: The interval between two events in milliseconds
+        type: integer
+        default: 1000
+      message:
+        title: Message
+        description: The message to generate
+        type: string
+        example: hello world
+      contentType:
+        title: Content Type
+        description: The content type of the message being generated
+        type: string
+        default: text/plain
+  dependencies:
+    - "camel:core"
+    - "camel:timer"
+    - "camel:kamelet"
+  template:
+    from:
+      uri: timer:tick
+      parameters:
+        period: "{{period}}"
+      steps:
+        - set-body:
+            constant: "{{message}}"
+        - set-header:
+            name: "Content-Type"
+            constant: "{{contentType}}"
+        - to: kamelet:sink
diff --git a/e2e/global/common/kamelet_test.go 
b/e2e/global/common/kamelet_test.go
index d255f7319..de91fbe80 100644
--- a/e2e/global/common/kamelet_test.go
+++ b/e2e/global/common/kamelet_test.go
@@ -41,18 +41,37 @@ func TestKameletClasspathLoading(t *testing.T) {
 
                Eventually(Kamelet(kameletName, ns)).Should(BeNil())
 
-               Expect(KamelRunWithID(operatorID, ns, 
"files/TimerKameletIntegration.java", "-t", "kamelets.enabled=false",
-                       "--resource", 
"file:files/timer-source.kamelet.yaml@/kamelets/timer-source.kamelet.yaml",
-                       "-p camel.component.kamelet.location=file:/kamelets",
-                       "-d", "camel:yaml-dsl",
-                       // kamelet dependencies
-                       "-d", "camel:timer").Execute()).To(Succeed())
-               Eventually(IntegrationPodPhase(ns, 
"timer-kamelet-integration"), TestTimeoutLong).Should(Equal(corev1.PodRunning))
-
-               Eventually(IntegrationLogs(ns, 
"timer-kamelet-integration")).Should(ContainSubstring("important message"))
-
-               // Cleanup
-               Expect(Kamel("delete", "--all", "-n", 
ns).Execute()).Should(BeNil())
+               // Basic
+               t.Run("test basic case", func(t *testing.T) {
+
+                       Expect(KamelRunWithID(operatorID, ns, 
"files/TimerKameletIntegration.java", "-t", "kamelets.enabled=false",
+                               "--resource", 
"file:files/timer-source.kamelet.yaml@/kamelets/timer-source.kamelet.yaml",
+                               "-p 
camel.component.kamelet.location=file:/kamelets",
+                               "-d", "camel:yaml-dsl",
+                               // kamelet dependencies
+                               "-d", "camel:timer").Execute()).To(Succeed())
+                       Eventually(IntegrationPodPhase(ns, 
"timer-kamelet-integration"), TestTimeoutLong).Should(Equal(corev1.PodRunning))
+
+                       Eventually(IntegrationLogs(ns, 
"timer-kamelet-integration")).Should(ContainSubstring("important message"))
+
+                       // Cleanup
+                       Expect(Kamel("delete", "--all", "-n", 
ns).Execute()).Should(BeNil())
+               })
+
+               // Custom repo
+               t.Run("test custom Kamelet repository", func(t *testing.T) {
+
+                       // Add the custom repository
+                       Expect(Kamel("kamelet", "add-repo", 
"github:essobedo/camel-k-test/kamelets", "-n", ns, "-x", 
operatorID).Execute()).To(Succeed())
+
+                       Expect(KamelRunWithID(operatorID, ns, 
"files/TimerCustomKameletIntegration.java").Execute()).To(Succeed())
+                       Eventually(IntegrationPodPhase(ns, 
"timer-custom-kamelet-integration"), 
TestTimeoutLong).Should(Equal(corev1.PodRunning))
+
+                       Eventually(IntegrationLogs(ns, 
"timer-custom-kamelet-integration")).Should(ContainSubstring("great message"))
+
+                       // Cleanup
+                       Expect(Kamel("delete", "--all", "-n", 
ns).Execute()).Should(BeNil())
+               })
        })
 }
 
diff --git a/pkg/cmd/kamelet.go b/pkg/cmd/kamelet.go
index 89a09062c..7645fd0c2 100644
--- a/pkg/cmd/kamelet.go
+++ b/pkg/cmd/kamelet.go
@@ -30,6 +30,7 @@ func newCmdKamelet(rootCmdOptions *RootCmdOptions) 
*cobra.Command {
 
        cmd.AddCommand(cmdOnly(newKameletGetCmd(rootCmdOptions)))
        cmd.AddCommand(cmdOnly(newKameletDeleteCmd(rootCmdOptions)))
+       cmd.AddCommand(cmdOnly(newKameletAddRepoCmd(rootCmdOptions)))
 
        return &cmd
 }
diff --git a/pkg/cmd/kamelet_add_repo.go b/pkg/cmd/kamelet_add_repo.go
new file mode 100644
index 000000000..92591823d
--- /dev/null
+++ b/pkg/cmd/kamelet_add_repo.go
@@ -0,0 +1,111 @@
+/*
+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 cmd
+
+import (
+       "errors"
+       "fmt"
+       "regexp"
+
+       v1 "github.com/apache/camel-k/pkg/apis/camel/v1"
+       "github.com/spf13/cobra"
+       k8serrors "k8s.io/apimachinery/pkg/api/errors"
+       "sigs.k8s.io/controller-runtime/pkg/client"
+)
+
+// kameletRepositoryURIRegexp is the regular expression used to validate the 
URI of a Kamelet repository.
+var kameletRepositoryURIRegexp = 
regexp.MustCompile(`^github:[^/]+/[^/]+((/[^/]+)*)?$`)
+
+func newKameletAddRepoCmd(rootCmdOptions *RootCmdOptions) (*cobra.Command, 
*kameletAddRepoCommandOptions) {
+       options := kameletAddRepoCommandOptions{
+               RootCmdOptions: rootCmdOptions,
+       }
+
+       cmd := cobra.Command{
+               Use:     "add-repo 
github:owner/repo[/path_to_kamelets_folder][@version] ...",
+               Short:   "Add a Kamelet repository",
+               Long:    `Add a Kamelet repository.`,
+               PreRunE: decode(&options),
+               RunE: func(cmd *cobra.Command, args []string) error {
+                       if err := options.validate(args); err != nil {
+                               return err
+                       }
+                       return options.run(cmd, args)
+               },
+       }
+
+       cmd.Flags().StringP("operator-id", "x", "camel-k", "Id of the Operator 
to update.")
+
+       return &cmd, &options
+}
+
+type kameletAddRepoCommandOptions struct {
+       *RootCmdOptions
+       OperatorID string `mapstructure:"operator-id" yaml:",omitempty"`
+}
+
+func (o *kameletAddRepoCommandOptions) validate(args []string) error {
+       if len(args) == 0 {
+               return errors.New("at least one Kamelet repository is expected")
+       }
+       if o.OperatorID == "" {
+               return fmt.Errorf("cannot use empty operator id")
+       }
+       return nil
+}
+
+func (o *kameletAddRepoCommandOptions) run(cmd *cobra.Command, args []string) 
error {
+       c, err := o.GetCmdClient()
+       if err != nil {
+               return err
+       }
+       key := client.ObjectKey{
+               Namespace: o.Namespace,
+               Name:      o.OperatorID,
+       }
+       platform := v1.IntegrationPlatform{}
+       if err := c.Get(o.Context, key, &platform); err != nil {
+               if k8serrors.IsNotFound(err) {
+                       // IntegrationPlatform may be in the operator 
namespace, but we currently don't have a way to determine it: we just warn
+                       fmt.Fprintf(cmd.ErrOrStderr(), "Warning: 
IntegrationPlatform %q not found in namespace %q\n", key.Name, key.Namespace)
+                       return nil
+               }
+               return err
+       }
+       for _, uri := range args {
+               if err := checkURI(uri, platform.Spec.Kamelet.Repositories); 
err != nil {
+                       return err
+               }
+               platform.Spec.Kamelet.Repositories = 
append(platform.Spec.Kamelet.Repositories, 
v1.IntegrationPlatformKameletRepositorySpec{
+                       URI: uri,
+               })
+       }
+       return c.Update(o.Context, &platform)
+}
+
+func checkURI(uri string, repositories 
[]v1.IntegrationPlatformKameletRepositorySpec) error {
+       if !kameletRepositoryURIRegexp.MatchString(uri) {
+               return fmt.Errorf("malformed Kamelet repository uri %s, the 
expected format is github:owner/repo[/path_to_kamelets_folder][@version]", uri)
+       }
+       for _, repo := range repositories {
+               if repo.URI == uri {
+                       return fmt.Errorf("duplicate Kamelet repository uri 
%s", uri)
+               }
+       }
+       return nil
+}
diff --git a/pkg/cmd/kamelet_add_repo_test.go b/pkg/cmd/kamelet_add_repo_test.go
new file mode 100644
index 000000000..fcaeeac6c
--- /dev/null
+++ b/pkg/cmd/kamelet_add_repo_test.go
@@ -0,0 +1,90 @@
+/*
+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 cmd
+
+import (
+       "testing"
+
+       "github.com/spf13/cobra"
+       "github.com/stretchr/testify/assert"
+
+       v1 "github.com/apache/camel-k/pkg/apis/camel/v1"
+       "github.com/apache/camel-k/pkg/util/test"
+)
+
+const cmdKameletAddRepo = "add-repo"
+
+// nolint: unparam
+func initializeKameletAddRepoCmdOptions(t *testing.T) 
(*kameletAddRepoCommandOptions, *cobra.Command, RootCmdOptions) {
+       t.Helper()
+
+       options, rootCmd := kamelTestPreAddCommandInit()
+       kameletAddRepoCommandOptions := addTestKameletAddRepoCmd(*options, 
rootCmd)
+       kamelTestPostAddCommandInit(t, rootCmd)
+
+       return kameletAddRepoCommandOptions, rootCmd, *options
+}
+
+func addTestKameletAddRepoCmd(options RootCmdOptions, rootCmd *cobra.Command) 
*kameletAddRepoCommandOptions {
+       // Add a testing version of kamelet add-repo Command
+       kameletAddRepoCmd, kameletAddRepoOptions := 
newKameletAddRepoCmd(&options)
+       kameletAddRepoCmd.RunE = func(c *cobra.Command, args []string) error {
+               return nil
+       }
+       kameletAddRepoCmd.PostRunE = func(c *cobra.Command, args []string) 
error {
+               return nil
+       }
+       kameletAddRepoCmd.Args = test.ArbitraryArgs
+       rootCmd.AddCommand(kameletAddRepoCmd)
+       return kameletAddRepoOptions
+}
+
+func TestKameletAddRepoNoFlag(t *testing.T) {
+       _, rootCmd, _ := initializeKameletAddRepoCmdOptions(t)
+       _, err := test.ExecuteCommand(rootCmd, cmdKameletAddRepo, "foo")
+       assert.Nil(t, err)
+}
+
+func TestKameletAddRepoNonExistingFlag(t *testing.T) {
+       _, rootCmd, _ := initializeKameletAddRepoCmdOptions(t)
+       _, err := test.ExecuteCommand(rootCmd, cmdKameletAddRepo, 
"--nonExistingFlag", "foo")
+       assert.NotNil(t, err)
+}
+
+func TestKameletAddRepoInvalidRepositoryURI(t *testing.T) {
+       repositories := []v1.IntegrationPlatformKameletRepositorySpec{}
+       assert.NotNil(t, checkURI("foo", repositories))
+       assert.NotNil(t, checkURI("github", repositories))
+       assert.NotNil(t, checkURI("github:", repositories))
+       assert.NotNil(t, checkURI("github:foo", repositories))
+       assert.NotNil(t, checkURI("github:foo/", repositories))
+}
+
+func TestKameletAddRepoValidRepositoryURI(t *testing.T) {
+       repositories := []v1.IntegrationPlatformKameletRepositorySpec{}
+       assert.Nil(t, checkURI("github:foo/bar", repositories))
+       assert.Nil(t, checkURI("github:foo/bar/some/path", repositories))
+       assert.Nil(t, checkURI("github:foo/bar@1.0", repositories))
+       assert.Nil(t, checkURI("github:foo/bar/some/path@1.0", repositories))
+}
+
+func TestKameletAddRepoDuplicateRepositoryURI(t *testing.T) {
+       repositories := []v1.IntegrationPlatformKameletRepositorySpec{{URI: 
"github:foo/bar"}}
+       assert.NotNil(t, checkURI("github:foo/bar", repositories))
+       assert.Nil(t, checkURI("github:foo/bar2", repositories))
+}
diff --git a/pkg/cmd/kamelet_get.go b/pkg/cmd/kamelet_get.go
index 013e92c0f..ced23216a 100644
--- a/pkg/cmd/kamelet_get.go
+++ b/pkg/cmd/kamelet_get.go
@@ -44,11 +44,8 @@ func newKameletGetCmd(rootCmdOptions *RootCmdOptions) 
(*cobra.Command, *kameletG
                        if err := options.validate(); err != nil {
                                return err
                        }
-                       if err := options.run(cmd); err != nil {
-                               fmt.Fprintln(cmd.ErrOrStderr(), err.Error())
-                       }
 
-                       return nil
+                       return options.run(cmd)
                },
        }
 
diff --git a/pkg/cmd/kit_get.go b/pkg/cmd/kit_get.go
index e74864897..47f78268a 100644
--- a/pkg/cmd/kit_get.go
+++ b/pkg/cmd/kit_get.go
@@ -42,11 +42,8 @@ func newKitGetCmd(rootCmdOptions *RootCmdOptions) 
(*cobra.Command, *kitGetComman
                        if err := options.validate(cmd, args); err != nil {
                                return err
                        }
-                       if err := options.run(cmd); err != nil {
-                               fmt.Fprintln(cmd.ErrOrStderr(), err.Error())
-                       }
 
-                       return nil
+                       return options.run(cmd)
                },
        }
 

Reply via email to