diff --git a/Makefile b/Makefile
index 7a00b79c..adfc623d 100644
--- a/Makefile
+++ b/Makefile
@@ -20,8 +20,8 @@ OPERATOR_VERSION := $(subst -SNAPSHOT,,$(VERSION))
LAST_RELEASED_IMAGE_NAME := yaks-operator
LAST_RELEASED_VERSION := 0.19.0
-CONTROLLER_GEN_VERSION := v0.6.1
-CODEGEN_VERSION := v0.25.6
+CONTROLLER_GEN_VERSION := v0.15.0
+CODEGEN_VERSION := v0.30.2
OPERATOR_SDK_VERSION := v1.28.0
KUSTOMIZE_VERSION := v4.5.4
DEFAULT_IMAGE := docker.io/citrusframework/yaks
diff --git a/config/crd/bases/yaks.citrusframework.org_instances.yaml b/config/crd/bases/yaks.citrusframework.org_instances.yaml
index f08a7bd5..674aa37d 100644
--- a/config/crd/bases/yaks.citrusframework.org_instances.yaml
+++ b/config/crd/bases/yaks.citrusframework.org_instances.yaml
@@ -20,8 +20,7 @@ metadata:
labels:
app: yaks
annotations:
- controller-gen.kubebuilder.io/version: v0.6.1
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.15.0
name: instances.yaks.citrusframework.org
spec:
group: yaks.citrusframework.org
@@ -54,14 +53,19 @@ spec:
description: Instance is the Schema for the yaks instance.
properties:
apiVersion:
- description: 'APIVersion defines the versioned schema of this representation
- of an object. Servers should convert recognized schemas to the latest
- internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ description: |-
+ APIVersion defines the versioned schema of this representation of an object.
+ Servers should convert recognized schemas to the latest internal value, and
+ may reject unrecognized values.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources
type: string
kind:
- description: 'Kind is a string value representing the REST resource this
- object represents. Servers may infer this from the endpoint the client
- submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ description: |-
+ Kind is a string value representing the REST resource this object represents.
+ Servers may infer this from the endpoint the client submits requests to.
+ Cannot be updated.
+ In CamelCase.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds
type: string
metadata:
type: object
diff --git a/config/crd/bases/yaks.citrusframework.org_tests.yaml b/config/crd/bases/yaks.citrusframework.org_tests.yaml
index f96a1822..6c7d2915 100644
--- a/config/crd/bases/yaks.citrusframework.org_tests.yaml
+++ b/config/crd/bases/yaks.citrusframework.org_tests.yaml
@@ -20,8 +20,7 @@ metadata:
labels:
app: yaks
annotations:
- controller-gen.kubebuilder.io/version: v0.6.1
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.15.0
name: tests.yaks.citrusframework.org
spec:
group: yaks.citrusframework.org
@@ -66,14 +65,19 @@ spec:
description: Test is the Schema for the tests API.
properties:
apiVersion:
- description: 'APIVersion defines the versioned schema of this representation
- of an object. Servers should convert recognized schemas to the latest
- internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ description: |-
+ APIVersion defines the versioned schema of this representation of an object.
+ Servers should convert recognized schemas to the latest internal value, and
+ may reject unrecognized values.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources
type: string
kind:
- description: 'Kind is a string value representing the REST resource this
- object represents. Servers may infer this from the endpoint the client
- submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ description: |-
+ Kind is a string value representing the REST resource this object represents.
+ Servers may infer this from the endpoint the client submits requests to.
+ Cannot be updated.
+ In CamelCase.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds
type: string
metadata:
type: object
@@ -120,8 +124,10 @@ spec:
verbose:
type: boolean
type: object
- secret:
- type: string
+ secrets:
+ items:
+ type: string
+ type: array
selenium:
description: SeleniumSpec --.
properties:
diff --git a/deploy/olm-catalog/yaks/0.19.2/yaks.citrusframework.org_instances.yaml b/deploy/olm-catalog/yaks/0.19.2/yaks.citrusframework.org_instances.yaml
new file mode 100644
index 00000000..3b2ae087
--- /dev/null
+++ b/deploy/olm-catalog/yaks/0.19.2/yaks.citrusframework.org_instances.yaml
@@ -0,0 +1,100 @@
+# ---------------------------------------------------------------------------
+# Copyright the original author or 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.
+# ---------------------------------------------------------------------------
+
+apiVersion: apiextensions.k8s.io/v1
+kind: CustomResourceDefinition
+metadata:
+ annotations:
+ controller-gen.kubebuilder.io/version: v0.6.1
+ creationTimestamp: null
+ labels:
+ app: yaks
+ name: instances.yaks.citrusframework.org
+spec:
+ group: yaks.citrusframework.org
+ names:
+ categories:
+ - yaks
+ - testing
+ kind: Instance
+ listKind: InstanceList
+ plural: instances
+ singular: instance
+ scope: Namespaced
+ versions:
+ - additionalPrinterColumns:
+ - description: True if YAKS instance is global
+ jsonPath: .spec.operator.global
+ name: Global
+ type: boolean
+ - description: The YAKS operator pod name
+ jsonPath: .spec.operator.pod
+ name: Pod
+ type: string
+ - description: The YAKS version
+ jsonPath: .status.version
+ name: Version
+ type: string
+ name: v1alpha1
+ schema:
+ openAPIV3Schema:
+ description: Instance is the Schema for the yaks instance.
+ properties:
+ apiVersion:
+ description: 'APIVersion defines the versioned schema of this representation
+ of an object. Servers should convert recognized schemas to the latest
+ internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ type: string
+ kind:
+ description: 'Kind is a string value representing the REST resource this
+ object represents. Servers may infer this from the endpoint the client
+ submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ type: string
+ metadata:
+ type: object
+ spec:
+ description: InstanceSpec provides the state of a yaks instance
+ properties:
+ operator:
+ description: OperatorSpec--.
+ properties:
+ global:
+ type: boolean
+ namespace:
+ type: string
+ pod:
+ type: string
+ required:
+ - global
+ type: object
+ type: object
+ status:
+ description: InstanceStatus defines the observed state of a yaks instance
+ properties:
+ version:
+ type: string
+ type: object
+ type: object
+ served: true
+ storage: true
+ subresources:
+ status: {}
+status:
+ acceptedNames:
+ kind: ""
+ plural: ""
+ conditions: null
+ storedVersions: null
diff --git a/deploy/olm-catalog/yaks/0.19.2/yaks.citrusframework.org_tests.yaml b/deploy/olm-catalog/yaks/0.19.2/yaks.citrusframework.org_tests.yaml
new file mode 100644
index 00000000..fe330ea2
--- /dev/null
+++ b/deploy/olm-catalog/yaks/0.19.2/yaks.citrusframework.org_tests.yaml
@@ -0,0 +1,214 @@
+# ---------------------------------------------------------------------------
+# Copyright the original author or 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.
+# ---------------------------------------------------------------------------
+
+apiVersion: apiextensions.k8s.io/v1
+kind: CustomResourceDefinition
+metadata:
+ annotations:
+ controller-gen.kubebuilder.io/version: v0.6.1
+ creationTimestamp: null
+ labels:
+ app: yaks
+ name: tests.yaks.citrusframework.org
+spec:
+ group: yaks.citrusframework.org
+ names:
+ categories:
+ - yaks
+ - testing
+ kind: Test
+ listKind: TestList
+ plural: tests
+ singular: test
+ scope: Namespaced
+ versions:
+ - additionalPrinterColumns:
+ - description: The test phase
+ jsonPath: .status.phase
+ name: Phase
+ type: string
+ - description: The total amount of tests
+ jsonPath: .status.results.summary.total
+ name: Total
+ type: string
+ - description: Passed tests
+ jsonPath: .status.results.summary.passed
+ name: Passed
+ type: string
+ - description: Failed tests
+ jsonPath: .status.results.summary.failed
+ name: Failed
+ type: string
+ - description: Skipped tests
+ jsonPath: .status.results.summary.skipped
+ name: Skipped
+ type: string
+ - description: Test error details
+ jsonPath: .status.errors
+ name: Errors
+ type: string
+ name: v1alpha1
+ schema:
+ openAPIV3Schema:
+ description: Test is the Schema for the tests API.
+ properties:
+ apiVersion:
+ description: 'APIVersion defines the versioned schema of this representation
+ of an object. Servers should convert recognized schemas to the latest
+ internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ type: string
+ kind:
+ description: 'Kind is a string value representing the REST resource this
+ object represents. Servers may infer this from the endpoint the client
+ submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ type: string
+ metadata:
+ type: object
+ spec:
+ description: TestSpec defines the desired state of Test
+ properties:
+ config:
+ description: SettingsSpec --.
+ properties:
+ content:
+ type: string
+ name:
+ type: string
+ type: object
+ env:
+ items:
+ type: string
+ type: array
+ kubedock:
+ description: KubeDockSpec --.
+ properties:
+ image:
+ type: string
+ runAsUser:
+ type: integer
+ type: object
+ resources:
+ items:
+ description: ResourceSpec --.
+ properties:
+ content:
+ type: string
+ name:
+ type: string
+ type: object
+ type: array
+ runtime:
+ description: RuntimeSpec --.
+ properties:
+ logger:
+ items:
+ type: string
+ type: array
+ verbose:
+ type: boolean
+ type: object
+ secret:
+ type: string
+ selenium:
+ description: SeleniumSpec --.
+ properties:
+ env:
+ items:
+ type: string
+ type: array
+ image:
+ type: string
+ noVNC:
+ type: boolean
+ runAsUser:
+ type: integer
+ type: object
+ source:
+ description: SourceSpec --.
+ properties:
+ content:
+ type: string
+ language:
+ type: string
+ name:
+ type: string
+ type: object
+ type: object
+ status:
+ description: TestStatus defines the observed state of Test
+ properties:
+ digest:
+ type: string
+ errors:
+ type: string
+ phase:
+ description: TestPhase --.
+ type: string
+ results:
+ properties:
+ errors:
+ items:
+ type: string
+ type: array
+ suiteName:
+ type: string
+ summary:
+ properties:
+ errors:
+ type: integer
+ failed:
+ type: integer
+ passed:
+ type: integer
+ pending:
+ type: integer
+ skipped:
+ type: integer
+ total:
+ type: integer
+ undefined:
+ type: integer
+ type: object
+ tests:
+ items:
+ properties:
+ classname:
+ type: string
+ errorMessage:
+ type: string
+ errorType:
+ type: string
+ name:
+ type: string
+ type: object
+ type: array
+ type: object
+ testID:
+ type: string
+ version:
+ type: string
+ type: object
+ type: object
+ served: true
+ storage: true
+ subresources:
+ status: {}
+status:
+ acceptedNames:
+ kind: ""
+ plural: ""
+ conditions: null
+ storedVersions: null
diff --git a/deploy/olm-catalog/yaks/0.19.2/yaks.v0.19.2.clusterserviceversion.yaml b/deploy/olm-catalog/yaks/0.19.2/yaks.v0.19.2.clusterserviceversion.yaml
new file mode 100644
index 00000000..787afd24
--- /dev/null
+++ b/deploy/olm-catalog/yaks/0.19.2/yaks.v0.19.2.clusterserviceversion.yaml
@@ -0,0 +1,437 @@
+# ---------------------------------------------------------------------------
+# Copyright the original author or 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.
+# ---------------------------------------------------------------------------
+
+apiVersion: operators.coreos.com/v1alpha1
+kind: ClusterServiceVersion
+metadata:
+ annotations:
+ alm-examples: |-
+ [
+ {
+ "apiVersion": "yaks.citrusframework.org/v1alpha1",
+ "kind": "Instance",
+ "metadata": {
+ "labels": {
+ "app": "yaks"
+ },
+ "name": "example-instance"
+ },
+ "spec": {
+ "operator": {
+ "global": true
+ }
+ }
+ },
+ {
+ "apiVersion": "yaks.citrusframework.org/v1alpha1",
+ "kind": "Test",
+ "metadata": {
+ "labels": {
+ "app": "yaks"
+ },
+ "name": "example-test"
+ },
+ "spec": {
+ "source": {
+ "content": "Feature: hello world\n\n Scenario: print slogan\n Given YAKS does Cloud-Native BDD testing\n Then YAKS rocks!",
+ "language": "feature",
+ "name": "example.feature"
+ }
+ }
+ }
+ ]
+ capabilities: Full Lifecycle
+ categories: Integration & Delivery
+ certified: "false"
+ containerImage: docker.io/citrusframework/yaks:0.19.2
+ createdAt: "2024-05-17T10:45:05Z"
+ description: YAKS is a platform to enable Cloud Native BDD testing on Kubernetes.
+ operators.operatorframework.io/builder: operator-sdk-v1.28.0
+ operators.operatorframework.io/internal-objects: '["instances.yaks.citrusframework.org"]'
+ operators.operatorframework.io/project_layout: go.kubebuilder.io/v2
+ repository: https://github.com/citrusframework/yaks
+ support: Citrus Framework
+ name: yaks-operator.v0.19.2
+ namespace: placeholder
+spec:
+ apiservicedefinitions: {}
+ customresourcedefinitions:
+ owned:
+ - description: Instance is the Schema for the yaks instance
+ displayName: Instance
+ kind: Instance
+ name: instances.yaks.citrusframework.org
+ version: v1alpha1
+ - description: Test is the Schema for the tests API
+ displayName: Test
+ kind: Test
+ name: tests.yaks.citrusframework.org
+ version: v1alpha1
+ description: |
+ YAKS
+ ====
+
+ YAKS is a platform to enable Cloud Native BDD testing on Kubernetes.
+
+ ## Running a YAKS test
+
+ With the YAKS operator installed, you can run tests by creating a `Test` custom resource on the cluster.
+
+ Tests are defined using [Gherkin](https://cucumber.io/docs/gherkin/) syntax. YAKS provides a set of predefined steps which
+ help to connect with different messaging transports (Http REST, JMS, Kafka, Knative eventing) and verify responses with
+ assertions on message header and body content.
+
+ The example below defines a simple test resource.
+
+ ```
+ apiVersion: yaks.citrusframework.org/v1alpha1
+ kind: Test
+ metadata:
+ name: example
+ spec:
+ source:
+ name: example.feature
+ content: |-
+ Feature: hello world
+
+ Scenario: print slogan
+ Given YAKS does Cloud-Native BDD testing
+ Then YAKS rocks!
+ ```
+
+ Refer to the [YAKS repository](https://github.com/citrusframework/yaks) for more complex examples.
+ displayName: YAKS Operator
+ icon:
+ - base64data: <?xml version="1.0" encoding="UTF-8" standalone="no"?>
<!-- Created with Inkscape (http://www.inkscape.org/) -->

<svg
   xmlns:dc="http://purl.org/dc/elements/1.1/"
   xmlns:cc="http://creativecommons.org/ns#"
   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
   xmlns:svg="http://www.w3.org/2000/svg"
   xmlns="http://www.w3.org/2000/svg"
   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
   width="49.070492mm"
   height="47.512463mm"
   viewBox="0 0 49.070492 47.512463"
   version="1.1"
   id="svg8"
   inkscape:version="0.92.2 5c3e80d, 2017-08-06"
   sodipodi:docname="logo.svg"
   inkscape:export-xdpi="368.63"
   inkscape:export-ydpi="368.63">
  <defs
     id="defs2" />
  <sodipodi:namedview
     id="base"
     pagecolor="#ffffff"
     bordercolor="#666666"
     borderopacity="1.0"
     inkscape:pageopacity="0.0"
     inkscape:pageshadow="2"
     inkscape:zoom="4.4104216"
     inkscape:cx="92.731641"
     inkscape:cy="89.787328"
     inkscape:document-units="mm"
     inkscape:current-layer="g874"
     showgrid="false"
     inkscape:window-width="1680"
     inkscape:window-height="1005"
     inkscape:window-x="2560"
     inkscape:window-y="194"
     inkscape:window-maximized="1" />
  <metadata
     id="metadata5">
    <rdf:RDF>
      <cc:Work
         rdf:about="">
        <dc:format>image/svg+xml</dc:format>
        <dc:type
           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
        <dc:title />
      </cc:Work>
    </rdf:RDF>
  </metadata>
  <g
     inkscape:label="Layer 1"
     inkscape:groupmode="layer"
     id="layer1"
     transform="translate(-37.230254,-97.640813)">
    <g
       id="g874"
       transform="matrix(0.5344693,0,0,0.5344693,39.331637,98.273668)">
      <path
         style="fill:#009100;fill-opacity:1;stroke:#8cd490;stroke-width:4.3669486;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
         id="path10349"
         d="m 78.047208,19.800131 c -0.5,-1.6 -1.7,-2.9 -3.2,-3.7 l -30.5,-14.6000005 c -0.8,-0.4 -1.7,-0.5 -2.5,-0.5 -0.8,0 -1.7,0 -2.5,0.2 L 8.8472076,15.900131 c -1.5000002,0.7 -2.6000002,2 -3.0000002,3.7 l -7.5,32.9 c -0.3,1.7 0.1,3.4 1.1,4.8 l 21.1000006,26.1 c 1.2,1.2 2.9,2 4.6,2.1 h 33.6 c 1.8,0.2 3.5,-0.6 4.6,-2.1 l 21.1,-26.1 c 1,-1.4 1.4,-3.1 1.2,-4.8 z"
         inkscape:connector-curvature="0" />
      <path
         style="fill:#92fa91;fill-opacity:1;stroke:#30b039;stroke-width:0.56130445;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
         d="m 32.79796,11.993484 c 1.280335,6.503176 8.905546,24.020196 8.836134,28.224729 4.206855,-9.633566 5.662953,-19.731107 5.412979,-28.339338 C 41.700177,7.7998697 36.053855,8.7850567 32.79796,11.993484 Z"
         id="path895"
         inkscape:connector-curvature="0"
         sodipodi:nodetypes="cccc" />
      <path
         style="fill:#92fa91;fill-opacity:1;stroke:#30b039;stroke-width:0.56130445;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
         d="m 8.882214,41.94944 c 4.956952,1.64244 17.7688,0.584632 27.093024,2.230083 C 31.123115,37.081583 23.286352,31.673161 12.219786,26.213735 7.042222,28.884156 6.9488727,36.883298 8.882214,41.94944 Z"
         id="path895-3"
         inkscape:connector-curvature="0"
         sodipodi:nodetypes="cccc"
         inkscape:transform-center-x="0.89613942"
         inkscape:transform-center-y="-0.68634753" />
      <path
         style="fill:#92fa91;fill-opacity:1;stroke:#30b039;stroke-width:0.56130445;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
         d="M 14.645242,63.804032 C 35.175028,51.326484 36.576698,47.241077 36.576698,47.241077 28.663032,49.134324 8.882214,41.94944 5.7455616,49.964623 6.1226673,53.959456 8.3655943,62.242831 14.645242,63.804032 Z"
         id="path895-3-6"
         inkscape:connector-curvature="0"
         sodipodi:nodetypes="cccc"
         inkscape:transform-center-x="0.94118156"
         inkscape:transform-center-y="2.0550388" />
      <path
         style="fill:#92fa91;fill-opacity:1;stroke:#30b039;stroke-width:0.56130445;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
         d="m 31.944563,78.12219 c 6.3276,-23.175911 6.88043,-28.493246 6.88043,-28.493246 -4.342432,8.572248 -19.391309,11.68134 -20.209203,20.569138 3.020575,4.141592 8.097247,9.350544 13.328773,7.924108 z"
         id="path895-3-6-7"
         inkscape:connector-curvature="0"
         sodipodi:nodetypes="cccc"
         inkscape:transform-center-x="1.2239014"
         inkscape:transform-center-y="1.3740268" />
      <path
         style="fill:#92fa91;fill-opacity:1;stroke:#30b039;stroke-width:0.56130445;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
         d="M 56.723474,76.908681 C 57.572691,70.776051 47.965868,57.466839 42.099047,50.34489 40.575204,60.940955 37.060794,71.723836 43.816303,78.710932 47.794164,78.184157 55.093435,80.53074 56.723474,76.908681 Z"
         id="path895-3-6-7-5"
         inkscape:connector-curvature="0"
         sodipodi:nodetypes="cccc"
         inkscape:transform-center-x="-2.0422851"
         inkscape:transform-center-y="0.97210317" />
      <path
         style="fill:#92fa91;fill-opacity:1;stroke:#30b039;stroke-width:0.56130445;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
         d="M 73.371152,58.512599 C 71.273386,54.452983 44.817075,48.252994 44.817075,48.252994 c 4.197377,5.356987 13.44743,20.829807 19.980264,23.516018 2.954194,-2.71544 9.430681,-7.902036 8.573813,-13.256413 z"
         id="path895-3-6-7-5-3"
         inkscape:connector-curvature="0"
         sodipodi:nodetypes="cccc"
         inkscape:transform-center-x="-1.4877682"
         inkscape:transform-center-y="1.2480337" />
      <path
         style="fill:#92fa91;fill-opacity:1;stroke:#30b039;stroke-width:0.56130445;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
         d="m 72.113774,36.72904 c -11.944599,1.876561 -25.820977,8.837081 -25.820977,8.837081 7.559776,1.019707 21.778826,9.012105 29.835012,6.738144 0.472098,-3.98472 0.119305,-12.065377 -4.014035,-15.575225 z"
         id="path895-3-6-7-5-3-5"
         inkscape:connector-curvature="0"
         sodipodi:nodetypes="cccc"
         inkscape:transform-center-x="-0.90631737"
         inkscape:transform-center-y="-1.6885587" />
      <path
         style="fill:#92fa91;fill-opacity:1;stroke:#30b039;stroke-width:0.56130445;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
         d="m 16.846897,23.023821 c 3.496742,5.520767 15.036547,10.557814 22.262532,18.339644 -0.822047,-8.580319 -7.157207,-20.944286 -12.718872,-27.519393 -3.812735,1.250581 -8.601907,4.202472 -9.54366,9.179749 z"
         id="path895-6"
         inkscape:connector-curvature="0"
         sodipodi:nodetypes="cccc" />
    </g>
  </g>
</svg>

+ mediatype: image/svg+xml
+ install:
+ spec:
+ clusterPermissions:
+ - rules:
+ - apiGroups:
+ - console.openshift.io
+ resources:
+ - consoleclidownloads
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ serviceAccountName: yaks-operator
+ deployments:
+ - label:
+ app: yaks
+ yaks.citrusframework.org/component: operator
+ name: yaks-operator
+ spec:
+ replicas: 1
+ selector:
+ matchLabels:
+ app: yaks
+ name: yaks-operator
+ strategy:
+ type: Recreate
+ template:
+ metadata:
+ labels:
+ app: yaks
+ name: yaks-operator
+ yaks.citrusframework.org/component: operator
+ spec:
+ containers:
+ - command:
+ - yaks
+ - operator
+ env:
+ - name: WATCH_NAMESPACE
+ valueFrom:
+ fieldRef:
+ fieldPath: metadata.annotations['olm.targetNamespaces']
+ - name: OPERATOR_NAME
+ value: yaks
+ - name: POD_NAME
+ valueFrom:
+ fieldRef:
+ fieldPath: metadata.name
+ - name: NAMESPACE
+ valueFrom:
+ fieldRef:
+ fieldPath: metadata.namespace
+ image: docker.io/citrusframework/yaks:0.19.1
+ imagePullPolicy: IfNotPresent
+ name: yaks-operator
+ resources: {}
+ serviceAccountName: yaks-operator
+ permissions:
+ - rules:
+ - apiGroups:
+ - yaks.citrusframework.org
+ resources:
+ - '*'
+ verbs:
+ - '*'
+ - apiGroups:
+ - batch
+ resources:
+ - jobs
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ - apiGroups:
+ - batch
+ resources:
+ - jobs/status
+ verbs:
+ - get
+ - list
+ - watch
+ - apiGroups:
+ - ""
+ resources:
+ - pods
+ - pods/exec
+ - services
+ - endpoints
+ - persistentvolumeclaims
+ - configmaps
+ - secrets
+ - serviceaccounts
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ - apiGroups:
+ - ""
+ resources:
+ - pods/log
+ - pods/status
+ verbs:
+ - get
+ - list
+ - watch
+ - apiGroups:
+ - rbac.authorization.k8s.io
+ resources:
+ - roles
+ - rolebindings
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ - apiGroups:
+ - rbac.authorization.k8s.io
+ resources:
+ - clusterroles
+ - clusterrolebindings
+ verbs:
+ - get
+ - list
+ - apiGroups:
+ - ""
+ resources:
+ - events
+ verbs:
+ - create
+ - patch
+ - get
+ - list
+ - watch
+ - apiGroups:
+ - apps
+ resources:
+ - deployments
+ - replicasets
+ - statefulsets
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ - apiGroups:
+ - coordination.k8s.io
+ resources:
+ - leases
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ - apiGroups:
+ - camel.apache.org
+ resources:
+ - integrations
+ - kamelets
+ - kameletbindings
+ - pipes
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ - apiGroups:
+ - eventing.knative.dev
+ resources:
+ - brokers
+ - triggers
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ - apiGroups:
+ - messaging.knative.dev
+ resources:
+ - channels
+ - inmemorychannels
+ - subscriptions
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ - apiGroups:
+ - sources.knative.dev
+ resources:
+ - '*'
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ - apiGroups:
+ - route.openshift.io
+ resources:
+ - routes
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ - apiGroups:
+ - monitoring.coreos.com
+ resources:
+ - servicemonitors
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ - apiGroups:
+ - kafka.strimzi.io
+ resources:
+ - kafkatopics
+ verbs:
+ - create
+ - delete
+ - deletecollection
+ - get
+ - list
+ - patch
+ - update
+ - watch
+ serviceAccountName: yaks-operator
+ strategy: deployment
+ installModes:
+ - supported: true
+ type: OwnNamespace
+ - supported: true
+ type: SingleNamespace
+ - supported: false
+ type: MultiNamespace
+ - supported: true
+ type: AllNamespaces
+ keywords:
+ - yaks
+ - testing
+ - microservices
+ - knative
+ - kafka
+ - serverless
+ - camel
+ - camel-k
+ labels:
+ name: yaks-operator
+ links:
+ - name: Project page
+ url: https://citrusframework.org/
+ - name: YAKS source code repository
+ url: https://github.com/citrusframework/yaks
+ maintainers:
+ - email: cdeppisch@redhat.com
+ name: Christoph Deppisch
+ - email: nferraro@redhat.com
+ name: Nicola Ferraro
+ maturity: alpha
+ minKubeVersion: 1.22.0
+ provider:
+ name: Citrus Framework
+ replaces: yaks-operator.v0.19.1
+ selector:
+ matchLabels:
+ name: yaks-operator
+ version: 0.19.2
diff --git a/deploy/olm-catalog/yaks/0.20.0-snapshot/yaks.citrusframework.org_instances.yaml b/deploy/olm-catalog/yaks/0.20.0-snapshot/yaks.citrusframework.org_instances.yaml
index 0add3b47..674aa37d 100644
--- a/deploy/olm-catalog/yaks/0.20.0-snapshot/yaks.citrusframework.org_instances.yaml
+++ b/deploy/olm-catalog/yaks/0.20.0-snapshot/yaks.citrusframework.org_instances.yaml
@@ -17,11 +17,10 @@
apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
- annotations:
- controller-gen.kubebuilder.io/version: v0.6.1
- creationTimestamp: null
labels:
app: yaks
+ annotations:
+ controller-gen.kubebuilder.io/version: v0.15.0
name: instances.yaks.citrusframework.org
spec:
group: yaks.citrusframework.org
@@ -54,14 +53,19 @@ spec:
description: Instance is the Schema for the yaks instance.
properties:
apiVersion:
- description: 'APIVersion defines the versioned schema of this representation
- of an object. Servers should convert recognized schemas to the latest
- internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ description: |-
+ APIVersion defines the versioned schema of this representation of an object.
+ Servers should convert recognized schemas to the latest internal value, and
+ may reject unrecognized values.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources
type: string
kind:
- description: 'Kind is a string value representing the REST resource this
- object represents. Servers may infer this from the endpoint the client
- submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ description: |-
+ Kind is a string value representing the REST resource this object represents.
+ Servers may infer this from the endpoint the client submits requests to.
+ Cannot be updated.
+ In CamelCase.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds
type: string
metadata:
type: object
@@ -92,9 +96,3 @@ spec:
storage: true
subresources:
status: {}
-status:
- acceptedNames:
- kind: ""
- plural: ""
- conditions: null
- storedVersions: null
diff --git a/deploy/olm-catalog/yaks/0.20.0-snapshot/yaks.citrusframework.org_tests.yaml b/deploy/olm-catalog/yaks/0.20.0-snapshot/yaks.citrusframework.org_tests.yaml
index 7d4c5f85..6c7d2915 100644
--- a/deploy/olm-catalog/yaks/0.20.0-snapshot/yaks.citrusframework.org_tests.yaml
+++ b/deploy/olm-catalog/yaks/0.20.0-snapshot/yaks.citrusframework.org_tests.yaml
@@ -17,11 +17,10 @@
apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
- annotations:
- controller-gen.kubebuilder.io/version: v0.6.1
- creationTimestamp: null
labels:
app: yaks
+ annotations:
+ controller-gen.kubebuilder.io/version: v0.15.0
name: tests.yaks.citrusframework.org
spec:
group: yaks.citrusframework.org
@@ -66,14 +65,19 @@ spec:
description: Test is the Schema for the tests API.
properties:
apiVersion:
- description: 'APIVersion defines the versioned schema of this representation
- of an object. Servers should convert recognized schemas to the latest
- internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ description: |-
+ APIVersion defines the versioned schema of this representation of an object.
+ Servers should convert recognized schemas to the latest internal value, and
+ may reject unrecognized values.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources
type: string
kind:
- description: 'Kind is a string value representing the REST resource this
- object represents. Servers may infer this from the endpoint the client
- submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ description: |-
+ Kind is a string value representing the REST resource this object represents.
+ Servers may infer this from the endpoint the client submits requests to.
+ Cannot be updated.
+ In CamelCase.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds
type: string
metadata:
type: object
@@ -120,8 +124,10 @@ spec:
verbose:
type: boolean
type: object
- secret:
- type: string
+ secrets:
+ items:
+ type: string
+ type: array
selenium:
description: SeleniumSpec --.
properties:
@@ -206,9 +212,3 @@ spec:
storage: true
subresources:
status: {}
-status:
- acceptedNames:
- kind: ""
- plural: ""
- conditions: null
- storedVersions: null
diff --git a/examples/kamelets/hello-source.kamelet.yaml b/examples/kamelets/hello-source.kamelet.yaml
index 32a77da3..e3afcf5a 100644
--- a/examples/kamelets/hello-source.kamelet.yaml
+++ b/examples/kamelets/hello-source.kamelet.yaml
@@ -14,7 +14,7 @@
# limitations under the License.
# ---------------------------------------------------------------------------
-apiVersion: camel.apache.org/v1alpha1
+apiVersion: camel.apache.org/v1
kind: Kamelet
metadata:
name: hello-source
@@ -41,5 +41,5 @@ spec:
period: "{{period}}"
steps:
- set-body:
- constant: "Hello world!"
+ constant: "Hello World!"
- to: "kamelet:sink"
diff --git a/examples/kamelets/hello-to-log-binding.yaml b/examples/kamelets/hello-to-log-binding.yaml
index 9cd9ce75..d8b30fa6 100644
--- a/examples/kamelets/hello-to-log-binding.yaml
+++ b/examples/kamelets/hello-to-log-binding.yaml
@@ -22,7 +22,7 @@ spec:
source:
ref:
kind: Kamelet
- apiVersion: camel.apache.org/v1alpha1
+ apiVersion: camel.apache.org/v1
name: hello-source
sink:
uri: log:info
diff --git a/examples/kamelets/kamelet-binding.feature b/examples/kamelets/kamelet-pipe.feature
similarity index 50%
rename from examples/kamelets/kamelet-binding.feature
rename to examples/kamelets/kamelet-pipe.feature
index b2c162f4..627c0302 100644
--- a/examples/kamelets/kamelet-binding.feature
+++ b/examples/kamelets/kamelet-pipe.feature
@@ -1,11 +1,4 @@
-Feature: Kamelet resource
-
- Background:
- Given Disable auto removal of Kamelet resources
- Given Disable auto removal of Kubernetes resources
- Given Camel K resource polling configuration
- | maxAttempts | 20 |
- | delayBetweenAttempts | 1000 |
+Feature: Kamelet pipe
Scenario: Bind Kamelet to service
# Create Kamelet from file
@@ -14,24 +7,24 @@ Feature: Kamelet resource
# Create the binding
Given create Kubernetes service greeting-service with target port 8080
- And KameletBinding source properties
+ And Pipe source properties
| message | Hello World |
And bind Kamelet hello-source to uri http://greeting-service.${YAKS_NAMESPACE}/greeting
- When create KameletBinding hello-source-uri
- Then KameletBinding hello-source-uri should be available
+ When create Pipe hello-source-uri
+ Then Pipe hello-source-uri should be available
# Verify binding
Given HTTP server "greeting-service"
And HTTP server timeout is 600000 ms
- Then expect HTTP request body: Hello World
+ Then expect HTTP request body: Hello World!
And receive POST /greeting
Scenario: Create binding from YAML
- Given load KameletBinding hello-to-log-binding.yaml
- Then KameletBinding hello-to-log-binding should be available
- And Camel K integration hello-to-log-binding should print Hello world!
+ Given load Pipe hello-to-log-binding.yaml
+ Then Pipe hello-to-log-binding should be available
+ And Camel K integration hello-to-log-binding should print Hello World!
Scenario: Remove Camel K resources
Given delete Kamelet hello-source
- Given delete KameletBinding hello-source-uri
- Given delete KameletBinding hello-to-log-binding
+ Given delete Pipe hello-source-uri
+ Given delete Pipe hello-to-log-binding
diff --git a/examples/kamelets/kamelet-resource.feature b/examples/kamelets/kamelet-resource.feature
index dcc957c5..1bd4a384 100644
--- a/examples/kamelets/kamelet-resource.feature
+++ b/examples/kamelets/kamelet-resource.feature
@@ -1,12 +1,5 @@
Feature: Kamelet resource
- Background:
- Given Disable auto removal of Camel K resources
- Given Disable auto removal of Kamelet resources
- Given Camel K resource polling configuration
- | maxAttempts | 200 |
- | delayBetweenAttempts | 1000 |
-
Scenario: Use Kamelet
Given load Camel K integration timer-to-log.groovy
Then Camel K integration timer-to-log should be running
diff --git a/examples/kamelets/kamelet.feature b/examples/kamelets/kamelet.feature
index 0740a042..0930a19b 100644
--- a/examples/kamelets/kamelet.feature
+++ b/examples/kamelets/kamelet.feature
@@ -1,12 +1,5 @@
Feature: Kamelet
- Background:
- Given Disable auto removal of Camel K resources
- Given Disable auto removal of Kamelet resources
- Given Camel K resource polling configuration
- | maxAttempts | 200 |
- | delayBetweenAttempts | 1000 |
-
Scenario: Create Kamelet
Given Kamelet property definition
| name | message |
diff --git a/examples/kamelets/yaks-config.yaml b/examples/kamelets/yaks-config.yaml
index 942b46af..1c23bad6 100644
--- a/examples/kamelets/yaks-config.yaml
+++ b/examples/kamelets/yaks-config.yaml
@@ -17,8 +17,14 @@
config:
runtime:
env:
- - name: YAKS_KAMELET_API_VERSION
- value: v1alpha1
+ - name: YAKS_CAMELK_MAX_ATTEMPTS
+ value: 100
+ - name: YAKS_CAMELK_DELAY_BETWEEN_ATTEMPTS
+ value: 1000
+ - name: YAKS_CAMELK_AUTO_REMOVE_RESOURCES
+ value: false
+ - name: YAKS_KAMELETS_AUTO_REMOVE_RESOURCES
+ value: false
settings:
loggers:
- name: INTEGRATION_STATUS
diff --git a/go.mod b/go.mod
index 5f6e2855..b52a31ac 100644
--- a/go.mod
+++ b/go.mod
@@ -1,6 +1,6 @@
module github.com/citrusframework/yaks
-go 1.21
+go 1.22
require (
github.com/Masterminds/semver v1.5.0
@@ -11,30 +11,30 @@ require (
github.com/mitchellh/go-homedir v1.1.0
github.com/mitchellh/mapstructure v1.5.0
// go get github.com/openshift/api@release-4.14
- github.com/openshift/api v0.0.0-20240304080513-3e8192a10b13
+ github.com/openshift/api v0.0.0-20240228005710-4511c790cc60
github.com/operator-framework/api v0.20.0
github.com/pkg/errors v0.9.1
github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring v0.67.1
- github.com/rs/xid v1.4.0
+ github.com/rs/xid v1.5.0
github.com/shurcooL/httpfs v0.0.0-20190707220628-8d4bc4ba7749
github.com/shurcooL/vfsgen v0.0.0-20181202132449-6a9ea43bcacd
github.com/sirupsen/logrus v1.9.3
github.com/spf13/cobra v1.8.0
github.com/spf13/pflag v1.0.5
- github.com/spf13/viper v1.18.2
+ github.com/spf13/viper v1.19.0
github.com/stoewer/go-strcase v1.3.0
github.com/stretchr/testify v1.9.0
- go.uber.org/multierr v1.10.0
- go.uber.org/zap v1.26.0
+ go.uber.org/multierr v1.11.0
+ go.uber.org/zap v1.27.0
gopkg.in/yaml.v2 v2.4.0
- k8s.io/api v0.27.7
- k8s.io/apiextensions-apiserver v0.27.7
- k8s.io/apimachinery v0.27.7
- k8s.io/client-go v0.27.7
+ k8s.io/api v0.28.8
+ k8s.io/apiextensions-apiserver v0.28.8
+ k8s.io/apimachinery v0.28.8
+ k8s.io/client-go v0.28.8
k8s.io/klog/v2 v2.120.1
- k8s.io/kubectl v0.27.7
- knative.dev/eventing v0.39.3
- sigs.k8s.io/controller-runtime v0.15.2
+ k8s.io/kubectl v0.28.8
+ knative.dev/eventing v0.40.3
+ sigs.k8s.io/controller-runtime v0.16.5
)
require (
@@ -49,21 +49,21 @@ require (
github.com/cloudevents/sdk-go/sql/v2 v2.13.0 // indirect
github.com/cloudevents/sdk-go/v2 v2.15.2 // indirect
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect
- github.com/emicklei/go-restful/v3 v3.10.2 // indirect
- github.com/evanphx/json-patch v4.12.0+incompatible // indirect
- github.com/evanphx/json-patch/v5 v5.7.0 // indirect
+ github.com/emicklei/go-restful/v3 v3.11.0 // indirect
+ github.com/evanphx/json-patch v5.6.0+incompatible // indirect
+ github.com/evanphx/json-patch/v5 v5.8.0 // indirect
github.com/fsnotify/fsnotify v1.7.0 // indirect
github.com/go-ini/ini v1.67.0 // indirect
github.com/go-kit/log v0.2.1 // indirect
github.com/go-logfmt/logfmt v0.5.1 // indirect
github.com/go-logr/zapr v1.2.4 // indirect
github.com/go-openapi/jsonpointer v0.19.6 // indirect
- github.com/go-openapi/jsonreference v0.20.1 // indirect
+ github.com/go-openapi/jsonreference v0.20.2 // indirect
github.com/go-openapi/swag v0.22.3 // indirect
github.com/gogo/protobuf v1.3.2 // indirect
github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect
- github.com/golang/protobuf v1.5.3 // indirect
- github.com/google/gnostic v0.6.9 // indirect
+ github.com/golang/protobuf v1.5.4 // indirect
+ github.com/google/gnostic-models v0.6.8 // indirect
github.com/google/go-cmp v0.6.0 // indirect
github.com/google/gofuzz v1.2.0 // indirect
github.com/grpc-ecosystem/grpc-gateway/v2 v2.11.3 // indirect
@@ -76,18 +76,17 @@ require (
github.com/kelseyhightower/envconfig v1.4.0 // indirect
github.com/magiconair/properties v1.8.7 // indirect
github.com/mailru/easyjson v0.7.7 // indirect
- github.com/matttproud/golang_protobuf_extensions v1.0.4 // indirect
github.com/minio/minio-go v6.0.14+incompatible // indirect
github.com/moby/spdystream v0.2.0 // indirect
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
github.com/modern-go/reflect2 v1.0.2 // indirect
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect
- github.com/pelletier/go-toml/v2 v2.1.0 // indirect
+ github.com/pelletier/go-toml/v2 v2.2.2 // indirect
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect
- github.com/prometheus/client_golang v1.17.0 // indirect
- github.com/prometheus/client_model v0.4.1-0.20230718164431-9a2bf3000d16 // indirect
- github.com/prometheus/common v0.44.0 // indirect
- github.com/prometheus/procfs v0.11.1 // indirect
+ github.com/prometheus/client_golang v1.18.0 // indirect
+ github.com/prometheus/client_model v0.5.0 // indirect
+ github.com/prometheus/common v0.46.0 // indirect
+ github.com/prometheus/procfs v0.12.0 // indirect
github.com/prometheus/statsd_exporter v0.22.7 // indirect
github.com/rickb777/date v1.13.0 // indirect
github.com/rickb777/plural v1.2.1 // indirect
@@ -102,31 +101,31 @@ require (
go.opencensus.io v0.24.0 // indirect
golang.org/x/crypto v0.21.0 // indirect
golang.org/x/exp v0.0.0-20230905200255-921286631fa9 // indirect
- golang.org/x/net v0.22.0 // indirect
- golang.org/x/oauth2 v0.15.0 // indirect
- golang.org/x/sync v0.5.0 // indirect
+ golang.org/x/net v0.23.0 // indirect
+ golang.org/x/oauth2 v0.18.0 // indirect
+ golang.org/x/sync v0.6.0 // indirect
golang.org/x/sys v0.18.0 // indirect
golang.org/x/term v0.18.0 // indirect
golang.org/x/text v0.14.0 // indirect
golang.org/x/time v0.5.0 // indirect
gomodules.xyz/jsonpatch/v2 v2.4.0 // indirect
- google.golang.org/api v0.153.0 // indirect
- google.golang.org/appengine v1.6.7 // indirect
- google.golang.org/genproto v0.0.0-20231106174013-bbf56f31fb17 // indirect
- google.golang.org/genproto/googleapis/api v0.0.0-20231106174013-bbf56f31fb17 // indirect
- google.golang.org/genproto/googleapis/rpc v0.0.0-20231120223509-83a465c0220f // indirect
- google.golang.org/grpc v1.59.0 // indirect
+ google.golang.org/api v0.171.0 // indirect
+ google.golang.org/appengine v1.6.8 // indirect
+ google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9 // indirect
+ google.golang.org/genproto/googleapis/api v0.0.0-20240311132316-a219d84964c2 // indirect
+ google.golang.org/genproto/googleapis/rpc v0.0.0-20240314234333-6e1732d8331c // indirect
+ google.golang.org/grpc v1.62.1 // indirect
google.golang.org/protobuf v1.33.0 // indirect
gopkg.in/inf.v0 v0.9.1 // indirect
gopkg.in/ini.v1 v1.67.0 // indirect
gopkg.in/yaml.v3 v3.0.1 // indirect
- k8s.io/component-base v0.27.7 // indirect
- k8s.io/kube-openapi v0.0.0-20230501164219-8b0f38b5fd1f // indirect
+ k8s.io/component-base v0.28.8 // indirect
+ k8s.io/kube-openapi v0.0.0-20230717233707-2695361300d9 // indirect
k8s.io/utils v0.0.0-20230505201702-9f6742963106 // indirect
- knative.dev/pkg v0.0.0-20231023151236-29775d7c9e5c // indirect
+ knative.dev/pkg v0.0.0-20240116073220-b488e7be5902 // indirect
sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd // indirect
sigs.k8s.io/structured-merge-diff/v4 v4.2.3 // indirect
- sigs.k8s.io/yaml v1.3.0 // indirect
+ sigs.k8s.io/yaml v1.4.0 // indirect
)
replace github.com/docker/docker => github.com/moby/moby v0.7.3-0.20190826074503-38ab9da00309 // Required by Helm
diff --git a/go.sum b/go.sum
index 97a2cff2..e43eea77 100644
--- a/go.sum
+++ b/go.sum
@@ -39,7 +39,6 @@ github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03
github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo=
github.com/Masterminds/semver v1.5.0 h1:H65muMkzWKEuNDnfl9d70GUjFniHKHRbFPGBuZ3QEww=
github.com/Masterminds/semver v1.5.0/go.mod h1:MB6lktGJrhw8PrUyiEoblNEGEQ+RzHPF078ddwwvV3Y=
-github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU=
github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
@@ -60,11 +59,9 @@ github.com/blang/semver/v4 v4.0.0 h1:1PFHFE6yCCTv8C1TeyNNarDzntLi7wMI5i/pzqYIsAM
github.com/blang/semver/v4 v4.0.0/go.mod h1:IbckMUScFkM3pff0VJDNKRiT6TG/YpiHIM2yvyW5YoQ=
github.com/blendle/zapdriver v1.3.1 h1:C3dydBOWYRiOk+B8X9IVZ5IOe+7cl+tGOexN4QqHfpE=
github.com/blendle/zapdriver v1.3.1/go.mod h1:mdXfREi6u5MArG4j9fewC+FGnXaBR+T4Ox4J2u4eHCc=
-github.com/buger/jsonparser v1.1.1/go.mod h1:6RYKKt7H4d4+iWqouImQ9R2FZql3VbhNgx27UK13J/0=
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
github.com/census-instrumentation/opencensus-proto v0.4.1 h1:iKLQ0xPNFxR/2hzXZMrBo8f1j86j5WHzznCCQxV/b8g=
github.com/census-instrumentation/opencensus-proto v0.4.1/go.mod h1:4T9NM4+4Vw91VeyqjLS6ao50K5bOcLKN6Q42XnYaRYw=
-github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc=
github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44=
@@ -78,30 +75,26 @@ github.com/cloudevents/sdk-go/sql/v2 v2.13.0/go.mod h1:XZRQBCgRreddIpQrdjBJQUrRg
github.com/cloudevents/sdk-go/v2 v2.15.2 h1:54+I5xQEnI73RBhWHxbI1XJcqOFOVJN85vb41+8mHUc=
github.com/cloudevents/sdk-go/v2 v2.15.2/go.mod h1:lL7kSWAE/V8VI4Wh0jbL2v/jvqsm6tjmaQBSvxcv4uE=
github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc=
-github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk=
-github.com/cncf/xds/go v0.0.0-20210312221358-fbca930ec8ed/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/container-tools/snap v0.0.10 h1:osH6xx67USIME5tz0HQ/EBZc7jgbPQwBG5aFwpDKYy4=
github.com/container-tools/snap v0.0.10/go.mod h1:oc/WJPTvUx+p4opnHu0S3hA0b//4zelB35lWt7xGNQ0=
+github.com/coreos/go-oidc/v3 v3.9.0 h1:0J/ogVOd4y8P0f0xUh8l9t07xRP/d8tccvjHl2dcsSo=
+github.com/coreos/go-oidc/v3 v3.9.0/go.mod h1:rTKz2PYwftcrtoCzV5g5kvfJoWcm0Mk8AF8y1iAQro4=
github.com/cpuguy83/go-md2man/v2 v2.0.3/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o=
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM=
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
-github.com/docopt/docopt-go v0.0.0-20180111231733-ee0de3bc6815/go.mod h1:WwZ+bS3ebgob9U8Nd0kOddGdZWjyMGR8Wziv+TBNwSE=
-github.com/emicklei/go-restful/v3 v3.10.2 h1:hIovbnmBTLjHXkqEBUz3HGpXZdM7ZrE9fJIZIqlJLqE=
-github.com/emicklei/go-restful/v3 v3.10.2/go.mod h1:6n3XBCmQQb25CM2LCACGz8ukIrRry+4bhvbpWn3mrbc=
+github.com/emicklei/go-restful/v3 v3.11.0 h1:rAQeMHw1c7zTmncogyy8VvRZwtkmkZ4FxERmMY4rD+g=
+github.com/emicklei/go-restful/v3 v3.11.0/go.mod h1:6n3XBCmQQb25CM2LCACGz8ukIrRry+4bhvbpWn3mrbc=
github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98=
-github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk=
-github.com/envoyproxy/go-control-plane v0.9.9-0.20210512163311-63b5d3c536b0/go.mod h1:hliV/p42l8fGbc6Y9bQ70uLwIvmJyVE5k4iMKlh8wCQ=
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
-github.com/evanphx/json-patch v4.12.0+incompatible h1:4onqiflcdA9EOZ4RxV643DvftH5pOlLGNtQ5lPWQu84=
-github.com/evanphx/json-patch v4.12.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk=
-github.com/evanphx/json-patch/v5 v5.7.0 h1:nJqP7uwL84RJInrohHfW0Fx3awjbm8qZeFv0nW9SYGc=
-github.com/evanphx/json-patch/v5 v5.7.0/go.mod h1:VNkHZ/282BpEyt/tObQO8s5CMPmYYq14uClGH4abBuQ=
-github.com/flowstack/go-jsonschema v0.1.1/go.mod h1:yL7fNggx1o8rm9RlgXv7hTBWxdBM0rVwpMwimd3F3N0=
+github.com/evanphx/json-patch v5.6.0+incompatible h1:jBYDEEiFBPxA0v50tFdvOzQQTCvpL6mnFh5mB2/l16U=
+github.com/evanphx/json-patch v5.6.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk=
+github.com/evanphx/json-patch/v5 v5.8.0 h1:lRj6N9Nci7MvzrXuX6HFzU8XjmhPiXPlsKEy1u0KQro=
+github.com/evanphx/json-patch/v5 v5.8.0/go.mod h1:VNkHZ/282BpEyt/tObQO8s5CMPmYYq14uClGH4abBuQ=
github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8=
github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
@@ -115,6 +108,8 @@ github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8=
github.com/go-ini/ini v1.67.0 h1:z6ZrTEZqSWOTyH2FlglNbNgARyHG8oLW9gMELqKr06A=
github.com/go-ini/ini v1.67.0/go.mod h1:ByCAeIL28uOIIG0E3PJtZPDL8WnHpFKFOtgjp+3Ies8=
+github.com/go-jose/go-jose/v3 v3.0.1 h1:pWmKFVtt+Jl0vBZTIpz/eAKwsm6LkIxDVVbFHKkchhA=
+github.com/go-jose/go-jose/v3 v3.0.1/go.mod h1:RNkWWRld676jZEYoV3+XK8L2ZnNSvIsxFMht0mSX+u8=
github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY=
@@ -133,8 +128,8 @@ github.com/go-logr/zapr v1.2.4 h1:QHVo+6stLbfJmYGkQ7uGHUCu5hnAFAj6mDe6Ea0SeOo=
github.com/go-logr/zapr v1.2.4/go.mod h1:FyHWQIzQORZ0QVE1BtVHv3cKtNLuXsbNLtpuhNapBOA=
github.com/go-openapi/jsonpointer v0.19.6 h1:eCs3fxoIi3Wh6vtgmLTOjdhSpiqphQ+DaPn38N2ZdrE=
github.com/go-openapi/jsonpointer v0.19.6/go.mod h1:osyAmYz/mB/C3I+WsTTSgw1ONzaLJoLCyoi6/zppojs=
-github.com/go-openapi/jsonreference v0.20.1 h1:FBLnyygC4/IZZr893oiomc9XaghoveYTrLC1F86HID8=
-github.com/go-openapi/jsonreference v0.20.1/go.mod h1:Bl1zwGIM8/wsvqjsOQLJ/SH+En5Ap4rVB5KVcIDZG2k=
+github.com/go-openapi/jsonreference v0.20.2 h1:3sVjiK66+uXK/6oQ8xgcRKcFgQ5KXa2KvnJRumpMGbE=
+github.com/go-openapi/jsonreference v0.20.2/go.mod h1:Bl1zwGIM8/wsvqjsOQLJ/SH+En5Ap4rVB5KVcIDZG2k=
github.com/go-openapi/swag v0.22.3 h1:yMBqmnQ0gyZvEb/+KzuWZOXgllrXT4SADYbvDaXHv/g=
github.com/go-openapi/swag v0.22.3/go.mod h1:UzaqsxGiab7freDnrUUra0MwWfN/q7tE4j+VcZ0yl14=
github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY=
@@ -144,8 +139,8 @@ github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7a
github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q=
github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q=
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
-github.com/golang/glog v1.1.2 h1:DVjP2PbBOzHyzA+dn3WhHIq4NdVu3Q+pvivFICf/7fo=
-github.com/golang/glog v1.1.2/go.mod h1:zR+okUeTbrL6EL3xHUDxZuEtGv04p5shwip1+mL/rLQ=
+github.com/golang/glog v1.2.0 h1:uCdmnmatrKCgMBlM4rMuJZWOkPDqdbZPnrMXDY4gI68=
+github.com/golang/glog v1.2.0/go.mod h1:6AhwSGph0fcJtXVM/PEHPqZlFeoLxhs7/t5UDAwmO+w=
github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
@@ -174,12 +169,12 @@ github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw
github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY=
-github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg=
-github.com/golang/protobuf v1.5.3/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY=
+github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek=
+github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps=
github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
-github.com/google/gnostic v0.6.9 h1:ZK/5VhkoX835RikCHpSUJV9a+S3e1zLh59YnyWeBW+0=
-github.com/google/gnostic v0.6.9/go.mod h1:Nm8234We1lq6iB9OmlgNv3nH91XLLVZHCDayfA3xq+E=
+github.com/google/gnostic-models v0.6.8 h1:yo/ABAfM5IMRsS1VnXjTBvUb61tFIHozhlYvRgGre9I=
+github.com/google/gnostic-models v0.6.8/go.mod h1:5n7qKqH0f5wFt+aWF8CW6pZLLNOfYuF5OpfBSENuI8U=
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
@@ -191,6 +186,7 @@ github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/
github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
+github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
@@ -215,9 +211,12 @@ github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+
github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk=
github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
github.com/grpc-ecosystem/grpc-gateway v1.14.6/go.mod h1:zdiPV4Yse/1gnckTHtghG4GkDEdKCRJduHpTxT3/jcw=
-github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw=
github.com/grpc-ecosystem/grpc-gateway/v2 v2.11.3 h1:lLT7ZLSzGLI08vc9cpd+tYmNWjdKDqyr/2L+f6U12Fk=
github.com/grpc-ecosystem/grpc-gateway/v2 v2.11.3/go.mod h1:o//XUCC/F+yRGJoPO/VU0GSB0f8Nhgmxx0VIRUvaC0w=
+github.com/hashicorp/go-cleanhttp v0.5.2 h1:035FKYIWjmULyFRBKPs8TBQoi0x6d9G4xc9neXJWAZQ=
+github.com/hashicorp/go-cleanhttp v0.5.2/go.mod h1:kO/YDlP8L1346E6Sodw+PrpBSV4/SoxCXGY6BqNFT48=
+github.com/hashicorp/go-retryablehttp v0.6.7 h1:8/CAEZt/+F7kR7GevNHulKkUjLht3CPmn7egmhieNKo=
+github.com/hashicorp/go-retryablehttp v0.6.7/go.mod h1:vAew36LZh98gCBJNLH42IQ1ER/9wtLZZ8meHqQvEYWY=
github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
github.com/hashicorp/golang-lru v1.0.2 h1:dV3g9Z/unq5DpblPpw+Oqcv4dU/1omnb4Ok8iPY6p1c=
@@ -250,7 +249,6 @@ github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxv
github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc=
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
-github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
@@ -264,8 +262,6 @@ github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0
github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc=
github.com/mattn/goveralls v0.0.2/go.mod h1:8d1ZMHsd7fW6IRPKQh46F2WRpyib5/X4FOpevwGNQEw=
github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
-github.com/matttproud/golang_protobuf_extensions v1.0.4 h1:mmDVorXM7PCGKw94cs5zkfA9PSy5pEvNWRP0ET0TIVo=
-github.com/matttproud/golang_protobuf_extensions v1.0.4/go.mod h1:BSXmuO+STAnVfrANrmjBb36TMTDstsz7MSK+HVaYKv4=
github.com/minio/minio-go v6.0.14+incompatible h1:fnV+GD28LeqdN6vT2XdGKW8Qe/IfjJDswNVuni6km9o=
github.com/minio/minio-go v6.0.14+incompatible/go.mod h1:7guKYtitv8dktvNUGrhzmNlA5wrAABTQXCoesZdFQO8=
github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y=
@@ -289,22 +285,18 @@ github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI
github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
github.com/onsi/ginkgo v1.12.1 h1:mFwc4LvZ0xpSvDZ3E+k8Yte0hLOMxXUlP+yXtJqkYfQ=
github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk=
-github.com/onsi/ginkgo/v2 v2.9.5 h1:+6Hr4uxzP4XIUyAkg61dWBw8lb/gc4/X5luuxN/EC+Q=
-github.com/onsi/ginkgo/v2 v2.9.5/go.mod h1:tvAoo1QUJwNEU2ITftXTpR7R1RbCzoZUOs3RonqW57k=
+github.com/onsi/ginkgo/v2 v2.11.0 h1:WgqUCUt/lT6yXoQ8Wef0fsNn5cAuMK7+KT9UFRz2tcU=
+github.com/onsi/ginkgo/v2 v2.11.0/go.mod h1:ZhrRA5XmEE3x3rhlzamx/JJvujdZoJ2uvgI7kR0iZvM=
github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY=
github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo=
-github.com/onsi/gomega v1.27.7 h1:fVih9JD6ogIiHUN6ePK7HJidyEDpWGVB5mzM7cWNXoU=
-github.com/onsi/gomega v1.27.7/go.mod h1:1p8OOlwo2iUUDsHnOrjE5UKYJ+e3W8eQ3qSlRahPmr4=
-github.com/openshift/api v0.0.0-20240103200955-7ca3a4634e46 h1:mnrBzHjjqYKw2uinOVXL9Eplj3+QaQwJ3SaWAs8l6cc=
-github.com/openshift/api v0.0.0-20240103200955-7ca3a4634e46/go.mod h1:aQ6LDasvHMvHZXqLHnX2GRmnfTWCF/iIwz8EMTTIE9A=
-github.com/openshift/api v0.0.0-20240304080513-3e8192a10b13 h1:KNaEkpcVi4XGb86cA6FMJ8Wia7KWAembCUv8blIksTY=
-github.com/openshift/api v0.0.0-20240304080513-3e8192a10b13/go.mod h1:yimSGmjsI+XF1mr+AKBs2//fSXIOhhetHGbMlBEfXbs=
-github.com/openshift/api v3.9.1-0.20190927182313-d4a64ec2cbd8+incompatible h1:YwFnUQ5RQ17CmkxHyjpQnWAQOGkLKXY0shOUEyqaCGk=
-github.com/openshift/api v3.9.1-0.20190927182313-d4a64ec2cbd8+incompatible/go.mod h1:dh9o4Fs58gpFXGSYfnVxGR9PnV53I8TW84pQaJDdGiY=
+github.com/onsi/gomega v1.27.10 h1:naR28SdDFlqrG6kScpT8VWpu1xWY5nJRCF3XaYyBjhI=
+github.com/onsi/gomega v1.27.10/go.mod h1:RsS8tutOdbdgzbPtzzATp12yT7kM5I5aElG3evPbQ0M=
+github.com/openshift/api v0.0.0-20240228005710-4511c790cc60 h1:BfN2JThYjmpXhULHahY1heyfct+fsj4fhkUo3tVIGH4=
+github.com/openshift/api v0.0.0-20240228005710-4511c790cc60/go.mod h1:qNtV0315F+f8ld52TLtPvrfivZpdimOzTi3kn9IVbtU=
github.com/operator-framework/api v0.20.0 h1:A2YCRhr+6s0k3pRJacnwjh1Ue8BqjIGuQ2jvPg9XCB4=
github.com/operator-framework/api v0.20.0/go.mod h1:rXPOhrQ6mMeXqCmpDgt1ALoar9ZlHL+Iy5qut9R99a4=
-github.com/pelletier/go-toml/v2 v2.1.0 h1:FnwAJ4oYMvbT/34k9zzHuZNrhlz48GB3/s6at6/MHO4=
-github.com/pelletier/go-toml/v2 v2.1.0/go.mod h1:tJU2Z3ZkXwnxa4DPO899bsyIoywizdUvyaeZurnPPDc=
+github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM=
+github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs=
github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
@@ -321,30 +313,30 @@ github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqr
github.com/prometheus/client_golang v1.12.1/go.mod h1:3Z9XVyYiZYEO+YQWt3RD2R3jrbd179Rt297l4aS6nDY=
github.com/prometheus/client_golang v1.12.2/go.mod h1:3Z9XVyYiZYEO+YQWt3RD2R3jrbd179Rt297l4aS6nDY=
github.com/prometheus/client_golang v1.13.0/go.mod h1:vTeo+zgvILHsnnj/39Ou/1fPN5nJFOEMgftOUOmlvYQ=
-github.com/prometheus/client_golang v1.17.0 h1:rl2sfwZMtSthVU752MqfjQozy7blglC+1SOtjMAMh+Q=
-github.com/prometheus/client_golang v1.17.0/go.mod h1:VeL+gMmOAxkS2IqfCq0ZmHSL+LjWfWDUmp1mBz9JgUY=
+github.com/prometheus/client_golang v1.18.0 h1:HzFfmkOzH5Q8L8G+kSJKUx5dtG87sewO+FoDDqP5Tbk=
+github.com/prometheus/client_golang v1.18.0/go.mod h1:T+GXkCk5wSJyOqMIzVgvvjFDlkOQntgjkJWKrN5txjA=
github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
-github.com/prometheus/client_model v0.4.1-0.20230718164431-9a2bf3000d16 h1:v7DLqVdK4VrYkVD5diGdl4sxJurKJEMnODWRJlxV9oM=
-github.com/prometheus/client_model v0.4.1-0.20230718164431-9a2bf3000d16/go.mod h1:oMQmHW1/JoDwqLtg57MGgP/Fb1CJEYF2imWWhWtMkYU=
+github.com/prometheus/client_model v0.5.0 h1:VQw1hfvPvk3Uv6Qf29VrPF32JB6rtbgI6cYPYQjL0Qw=
+github.com/prometheus/client_model v0.5.0/go.mod h1:dTiFglRmd66nLR9Pv9f0mZi7B7fk5Pm3gvsjB5tr+kI=
github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4=
github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo=
github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc=
github.com/prometheus/common v0.32.1/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls=
github.com/prometheus/common v0.35.0/go.mod h1:phzohg0JFMnBEFGxTDbfu3QyL5GI8gTQJFhYO5B3mfA=
github.com/prometheus/common v0.37.0/go.mod h1:phzohg0JFMnBEFGxTDbfu3QyL5GI8gTQJFhYO5B3mfA=
-github.com/prometheus/common v0.44.0 h1:+5BrQJwiBB9xsMygAB3TNvpQKOwlkc25LbISbrdOOfY=
-github.com/prometheus/common v0.44.0/go.mod h1:ofAIvZbQ1e/nugmZGz4/qCb9Ap1VoSTIO7x0VV9VvuY=
+github.com/prometheus/common v0.46.0 h1:doXzt5ybi1HBKpsZOL0sSkaNHJJqkyfEWZGGqqScV0Y=
+github.com/prometheus/common v0.46.0/go.mod h1:Tp0qkxpb9Jsg54QMe+EAmqXkSV7Evdy1BTn+g2pa/hQ=
github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk=
github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU=
github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA=
github.com/prometheus/procfs v0.7.3/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA=
github.com/prometheus/procfs v0.8.0/go.mod h1:z7EfXMXOkbkqb9IINtpCn86r/to3BnA0uaxHdg830/4=
-github.com/prometheus/procfs v0.11.1 h1:xRC8Iq1yyca5ypa9n1EZnWZkt7dwcoRPQwX/5gwaUuI=
-github.com/prometheus/procfs v0.11.1/go.mod h1:eesXgaPo1q7lBpVMoMy0ZOFTth9hBn4W/y0/p/ScXhY=
+github.com/prometheus/procfs v0.12.0 h1:jluTpSng7V9hY0O2R9DzzJHYb2xULk9VTR1V1R/k6Bo=
+github.com/prometheus/procfs v0.12.0/go.mod h1:pcuDEFsWDnvcgNzo4EEweacyhjeA9Zk3cnaOZAZEfOo=
github.com/prometheus/statsd_exporter v0.22.7 h1:7Pji/i2GuhK6Lu7DHrtTkFmNBCudCPT1pX2CziuyQR0=
github.com/prometheus/statsd_exporter v0.22.7/go.mod h1:N/TevpjkIh9ccs6nuzY3jQn9dFqnUakOjnEuMPJJJnI=
github.com/rickb777/date v1.13.0 h1:+8AmwLuY1d/rldzdqvqTEg7107bZ8clW37x4nsdG3Hs=
@@ -355,10 +347,10 @@ github.com/robfig/cron/v3 v3.0.1 h1:WdRxkvbJztn8LMz/QEvLN5sBU+xKpSqwwUO1Pjr4qDs=
github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro=
github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ=
github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
-github.com/rogpeppe/go-internal v1.11.0 h1:cWPaGQEPrBb5/AsnsZesgZZ9yb1OQ+GOISoDNXVBh4M=
-github.com/rogpeppe/go-internal v1.11.0/go.mod h1:ddIwULY96R17DhadqLgMfk9H9tvdUzkipdSkR5nkCZA=
-github.com/rs/xid v1.4.0 h1:qd7wPTDkN6KQx2VmMBLrpHkiyQwgFXRnkOLacUiaSNY=
-github.com/rs/xid v1.4.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg=
+github.com/rogpeppe/go-internal v1.10.0 h1:TMyTOH3F/DB16zRVcYyreMH6GnZZrwQVAoYjRBZyWFQ=
+github.com/rogpeppe/go-internal v1.10.0/go.mod h1:UQnix2H7Ngw/k4C5ijL5+65zddjncjaFoBhdsK/akog=
+github.com/rs/xid v1.5.0 h1:mKX4bl4iPYJtEIxp6CYiUuLQ/8DYMoz0PUdtGgMFRVc=
+github.com/rs/xid v1.5.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg=
github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
github.com/sagikazarmark/locafero v0.4.0 h1:HApY1R9zGo4DBgr7dqsTH/JJxLTTsOt7u6keLGt6kNQ=
github.com/sagikazarmark/locafero v0.4.0/go.mod h1:Pe1W6UlPYUk/+wc/6KFhbORCfqzgYEpgQ3O5fPuL3H4=
@@ -377,7 +369,6 @@ github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ
github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
github.com/sourcegraph/conc v0.3.0 h1:OQTbbt6P72L20UqAkXXuLOj79LfEanQ+YQFNpLA9ySo=
github.com/sourcegraph/conc v0.3.0/go.mod h1:Sdozi7LEKbFPqYX2/J+iBAM6HpqSLTASQIKqDmF7Mt0=
-github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA=
github.com/spf13/afero v1.11.0 h1:WJQKhtpdm3v2IzqG8VMqrr6Rf3UYpEF239Jy9wNepM8=
github.com/spf13/afero v1.11.0/go.mod h1:GH9Y3pIexgf1MTIWtNGyogA5MwRIDXGUr+hbWNoBjkY=
github.com/spf13/cast v1.6.0 h1:GEiTHELF+vaR5dhz3VqZfFSzZjYbgeKDpBxQVS4GYJ0=
@@ -386,20 +377,19 @@ github.com/spf13/cobra v1.8.0 h1:7aJaZx1B85qltLMc546zn58BxxfZdR/W22ej9CFoEf0=
github.com/spf13/cobra v1.8.0/go.mod h1:WXLWApfZ71AjXPya3WOlMsY9yMs7YeiHhFVlvLyhcho=
github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA=
github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
-github.com/spf13/viper v1.18.2 h1:LUXCnvUvSM6FXAsj6nnfc8Q2tp1dIgUfY9Kc8GsSOiQ=
-github.com/spf13/viper v1.18.2/go.mod h1:EKmWIqdnk5lOcmR72yw6hS+8OPYcwD0jteitLMVB+yk=
+github.com/spf13/viper v1.19.0 h1:RWq5SEjt8o25SROyN3z2OrDB9l7RPd3lwTWU8EcEdcI=
+github.com/spf13/viper v1.19.0/go.mod h1:GQUN9bilAbhU/jgc1bKs99f/suXKeUMct8Adx5+Ntkg=
github.com/sqs/goreturns v0.0.0-20181028201513-538ac6014518/go.mod h1:CKI4AZ4XmGV240rTHfO0hfE83S6/a3/Q1siZJ/vXf7A=
-github.com/stoewer/go-strcase v1.2.0/go.mod h1:IBiWB2sKIp3wVVQ3Y035++gc+knqhUQag1KpM8ahLw8=
github.com/stoewer/go-strcase v1.3.0 h1:g0eASXYtp+yvN9fK8sH94oCIk0fau9uV1/ZdJ0AVEzs=
github.com/stoewer/go-strcase v1.3.0/go.mod h1:fAH5hQ5pehh+j3nZfvwdk2RgEgQjAoM8wodgtPmh1xo=
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
+github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA=
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
-github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA=
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
@@ -413,14 +403,12 @@ github.com/subosito/gotenv v1.6.0 h1:9NlTDc1FTs4qu0DDq7AEtTPNw6SVm7uBMsUCUjABIf8
github.com/subosito/gotenv v1.6.0/go.mod h1:Dk4QP5c2W3ibzajGcXpNraDfq2IrhjMIvMSWPKKo0FU=
github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw=
github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc=
-github.com/xeipuuv/gojsonpointer v0.0.0-20180127040702-4e3ac2762d5f/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU=
-github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415/go.mod h1:GwrjFmJcFw6At/Gs6z4yjiIwzuJ1/+UwLxMQDVQXShQ=
-github.com/xeipuuv/gojsonschema v1.2.0/go.mod h1:anYRn/JVcOK2ZgGU+IjEV4nwlhoK5sQluxsYJ78Id3Y=
github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k=
+github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8=
go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
@@ -429,26 +417,26 @@ go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
go.opencensus.io v0.23.0/go.mod h1:XItmlyltB5F7CS4xOC1DcqMoFqwtC6OG2xF7mCv7P7E=
go.opencensus.io v0.24.0 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0=
go.opencensus.io v0.24.0/go.mod h1:vNK8G9p7aAivkbmorf4v+7Hgx+Zs0yY+0fOtgBfjQKo=
-go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI=
go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
go.uber.org/goleak v1.1.11/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ=
-go.uber.org/goleak v1.2.1 h1:NBol2c7O1ZokfZ0LEU9K6Whx/KnwvepVetCUhtKja4A=
-go.uber.org/goleak v1.2.1/go.mod h1:qlT2yGI9QafXHhZZLxlSuNsMw3FFLxBr+tBRlmO1xH4=
+go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto=
+go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE=
go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU=
-go.uber.org/multierr v1.10.0 h1:S0h4aNzvfcFsC3dRF1jLoaov7oRaKqRGC/pUEJ2yvPQ=
-go.uber.org/multierr v1.10.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
+go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
+go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
go.uber.org/zap v1.24.0/go.mod h1:2kMP+WWQ8aoFoedH3T2sq6iJ2yDWpHbP0f6MQbS9Gkg=
-go.uber.org/zap v1.26.0 h1:sI7k6L95XOKS281NhVKOFCUNIvv9e0w4BF8N3u+tCRo=
-go.uber.org/zap v1.26.0/go.mod h1:dtElttAiwGvoJ/vj4IwHBS/gXsEu/pZ50mUIRWuG0so=
+go.uber.org/zap v1.27.0 h1:aJMhYGrd5QSmlpLMr2MftRKl7t8J8PTZPA732ud/XR8=
+go.uber.org/zap v1.27.0/go.mod h1:GB2qFLM7cTU87MWRP2mPIjqfIDnGu+VIO4V/SdhGo2E=
golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
+golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
golang.org/x/crypto v0.21.0 h1:X31++rzVUdKhX5sWmSOFZxx8UW/ldWx55cbf08iNAMA=
golang.org/x/crypto v0.21.0/go.mod h1:0BP7YvVV9gBbVKyeTG0Gyn+gZm94bibOW5BjDEYAOMs=
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
@@ -484,8 +472,9 @@ golang.org/x/mod v0.1.1-0.20191107180719-034126e5016b/go.mod h1:QqPTAvyqsEbceGzB
golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
-golang.org/x/mod v0.13.0 h1:I/DsJXRlw/8l/0c24sM9yb0T4z9liZTduXvdAWYiysY=
-golang.org/x/mod v0.13.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
+golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
+golang.org/x/mod v0.14.0 h1:dGoOF9QVLYng8IHTm7BAyWqCqSheQ5pYWGhzW00YJr0=
+golang.org/x/mod v0.14.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c=
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
@@ -519,13 +508,14 @@ golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81R
golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA=
golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
+golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM=
golang.org/x/net v0.0.0-20210525063256-abc453219eb5/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
-golang.org/x/net v0.0.0-20210805182204-aaa1db679c0d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk=
golang.org/x/net v0.0.0-20220225172249-27dd8689420f/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk=
-golang.org/x/net v0.22.0 h1:9sGLhx7iRIHEiX0oAJ3MRZMUCElJgy7Br1nO+AMN3Tc=
-golang.org/x/net v0.22.0/go.mod h1:JKghWKKOSdJwpW2GEx0Ja7fmaKnMsbu+MWVZTokSYmg=
+golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c=
+golang.org/x/net v0.23.0 h1:7EYJ93RZ9vYSZAIb2x3lnuvqO5zneoD6IvWjuhfxjTs=
+golang.org/x/net v0.23.0/go.mod h1:JKghWKKOSdJwpW2GEx0Ja7fmaKnMsbu+MWVZTokSYmg=
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
@@ -533,8 +523,8 @@ golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4Iltr
golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
golang.org/x/oauth2 v0.0.0-20210514164344-f6687ab2804c/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
golang.org/x/oauth2 v0.0.0-20220223155221-ee480838109b/go.mod h1:DAh4E804XQdzx2j+YRIaUnCqCV2RuMz24cGBJ5QYIrc=
-golang.org/x/oauth2 v0.15.0 h1:s8pnnxNVzjWyrvYdFUQq5llS1PX2zhPXmccZv99h7uQ=
-golang.org/x/oauth2 v0.15.0/go.mod h1:q48ptWNTY5XWf+JNten23lcvHpLJ0ZSxF5ttTHKVCAM=
+golang.org/x/oauth2 v0.18.0 h1:09qnuIAgzdx1XplqJvW6CQqMCtGZykZWcXzPMPUusvI=
+golang.org/x/oauth2 v0.18.0/go.mod h1:Wf7knwG0MPoWIMMBgFlEaSUDaKskp0dCfrlJRJXbBi8=
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
@@ -547,8 +537,9 @@ golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJ
golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20220601150217-0de741cfad7f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
-golang.org/x/sync v0.5.0 h1:60k92dhOjHxJkrqnwsfl8KuaHbn/5dl0lUPUklKo3qE=
-golang.org/x/sync v0.5.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
+golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.6.0 h1:5BMeUDZ7vkXGfEr1x9B4bRcTH4lpkTkpdh0T/J+qjbQ=
+golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
@@ -596,6 +587,7 @@ golang.org/x/sys v0.0.0-20220114195835-da31bd327af9/go.mod h1:oPkhp1MJrh7nUepCBc
golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220708085239-5a0f0661e09d/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
+golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.18.0 h1:DBdB3niSjOA/O0blCZBqDefyWNYveAYMNF1Wum0DYQ4=
golang.org/x/sys v0.18.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
@@ -607,9 +599,9 @@ golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
-golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
+golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ=
golang.org/x/text v0.14.0 h1:ScX5w1eTa3QqT8oi6+ziP7dTV1S2+ALU0bI+0zXKWiQ=
golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
@@ -661,8 +653,9 @@ golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc
golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk=
-golang.org/x/tools v0.14.0 h1:jvNa2pY0M4r62jkRQ6RwEZZyPcymeL9XZMLBbV7U2nc=
-golang.org/x/tools v0.14.0/go.mod h1:uYBEerGOWcJyEORxN+Ek8+TT266gXkNlHdJBwexUsBg=
+golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
+golang.org/x/tools v0.17.0 h1:FvmRgNOcs3kOa+T20R1uhfP9F6HgG2mfxDv1vrx1Htc=
+golang.org/x/tools v0.17.0/go.mod h1:xsh6VxdV005rRVaS6SSAf9oiAqljS7UZUacMZ8Bnsps=
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
@@ -686,16 +679,16 @@ google.golang.org/api v0.25.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0M
google.golang.org/api v0.28.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE=
google.golang.org/api v0.29.0/go.mod h1:Lcubydp8VUV7KeIHD9z2Bys/sm/vGKnG1UHuDBSrHWM=
google.golang.org/api v0.30.0/go.mod h1:QGmEvQ87FHZNiUVJkT14jQNYJ4ZJjdRF23ZXz5138Fc=
-google.golang.org/api v0.153.0 h1:N1AwGhielyKFaUqH07/ZSIQR3uNPcV7NVw0vj+j4iR4=
-google.golang.org/api v0.153.0/go.mod h1:3qNJX5eOmhiWYc67jRA/3GsDw97UFb5ivv7Y2PrriAY=
+google.golang.org/api v0.171.0 h1:w174hnBPqut76FzW5Qaupt7zY8Kql6fiVjgys4f58sU=
+google.golang.org/api v0.171.0/go.mod h1:Hnq5AHm4OTMt2BUVjael2CWZFD6vksJdWCWiUAmjC9o=
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0=
google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
-google.golang.org/appengine v1.6.7 h1:FZR1q0exgwxzPzp/aF+VccGrSfxfPpkBqjIIEq3ru6c=
-google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
+google.golang.org/appengine v1.6.8 h1:IhEN5q69dyKagZPYMSdIjS2HqprW324FRQZJcGqPAsM=
+google.golang.org/appengine v1.6.8/go.mod h1:1jJ3jBArFh5pcgW8gCtRJnepW8FzD1V44FJffLiz/Ds=
google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
@@ -727,13 +720,12 @@ google.golang.org/genproto v0.0.0-20200618031413-b414f8b61790/go.mod h1:jDfRM7Fc
google.golang.org/genproto v0.0.0-20200729003335-053ba62fc06f/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
google.golang.org/genproto v0.0.0-20200804131852-c06518451d9c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
google.golang.org/genproto v0.0.0-20200825200019-8632dd797987/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
-google.golang.org/genproto v0.0.0-20220107163113-42d7afdf6368/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc=
-google.golang.org/genproto v0.0.0-20231106174013-bbf56f31fb17 h1:wpZ8pe2x1Q3f2KyT5f8oP/fa9rHAKgFPr/HZdNuS+PQ=
-google.golang.org/genproto v0.0.0-20231106174013-bbf56f31fb17/go.mod h1:J7XzRzVy1+IPwWHZUzoD0IccYZIrXILAQpc+Qy9CMhY=
-google.golang.org/genproto/googleapis/api v0.0.0-20231106174013-bbf56f31fb17 h1:JpwMPBpFN3uKhdaekDpiNlImDdkUAyiJ6ez/uxGaUSo=
-google.golang.org/genproto/googleapis/api v0.0.0-20231106174013-bbf56f31fb17/go.mod h1:0xJLfVdJqpAPl8tDg1ujOCGzx6LFLttXT5NhllGOXY4=
-google.golang.org/genproto/googleapis/rpc v0.0.0-20231120223509-83a465c0220f h1:ultW7fxlIvee4HYrtnaRPon9HpEgFk5zYpmfMgtKB5I=
-google.golang.org/genproto/googleapis/rpc v0.0.0-20231120223509-83a465c0220f/go.mod h1:L9KNLi232K1/xB6f7AlSX692koaRnKaWSR0stBki0Yc=
+google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9 h1:9+tzLLstTlPTRyJTh+ah5wIMsBW5c4tQwGTN3thOW9Y=
+google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9/go.mod h1:mqHbVIp48Muh7Ywss/AD6I5kNVKZMmAa/QEW58Gxp2s=
+google.golang.org/genproto/googleapis/api v0.0.0-20240311132316-a219d84964c2 h1:rIo7ocm2roD9DcFIX67Ym8icoGCKSARAiPljFhh5suQ=
+google.golang.org/genproto/googleapis/api v0.0.0-20240311132316-a219d84964c2/go.mod h1:O1cOfN1Cy6QEYr7VxtjOyP5AdAuR0aJ/MYZaaof623Y=
+google.golang.org/genproto/googleapis/rpc v0.0.0-20240314234333-6e1732d8331c h1:lfpJ/2rWPa/kJgxyyXM8PrNnfCzcmxJ265mADgwmvLI=
+google.golang.org/genproto/googleapis/rpc v0.0.0-20240314234333-6e1732d8331c/go.mod h1:WtryC6hu0hhx87FDGxWCDptyssuo68sk10vYjF+T9fY=
google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38=
google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM=
@@ -746,12 +738,9 @@ google.golang.org/grpc v1.28.0/go.mod h1:rpkK4SK4GF4Ach/+MFLZUBavHOvF2JJB5uozKKa
google.golang.org/grpc v1.29.1/go.mod h1:itym6AZVZYACWQqET3MqgPpjcuV5QH3BxFS3IjizoKk=
google.golang.org/grpc v1.30.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak=
google.golang.org/grpc v1.31.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak=
-google.golang.org/grpc v1.33.1/go.mod h1:fr5YgcSWrqhRRxogOsw7RzIpsmvOZ6IcH4kBYTpR3n0=
google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc=
-google.golang.org/grpc v1.36.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU=
-google.golang.org/grpc v1.40.0/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34=
-google.golang.org/grpc v1.59.0 h1:Z5Iec2pjwb+LEOqzpB2MR12/eKFhDPhuqW91O+4bwUk=
-google.golang.org/grpc v1.59.0/go.mod h1:aUPDwccQo6OTjy7Hct4AfBPD1GptF4fyUjIkQ9YtF98=
+google.golang.org/grpc v1.62.1 h1:B4n+nfKzOICUXMgyrNd19h/I9oH0L1pizfk1d4zSgTk=
+google.golang.org/grpc v1.62.1/go.mod h1:IWTG0VlJLCh1SkC58F7np9ka9mx/WNkjl4PGJaiq+QE=
google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8=
google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0=
google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM=
@@ -764,7 +753,6 @@ google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGj
google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c=
google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
-google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
google.golang.org/protobuf v1.28.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
google.golang.org/protobuf v1.28.1/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
google.golang.org/protobuf v1.33.0 h1:uNO2rsAINq/JlFpSdYEKIZ0uKD/R9cpdv0T+yoGwGmI=
@@ -792,7 +780,6 @@ gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
-gopkg.in/yaml.v3 v3.0.0-20200615113413-eeeca48fe776/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gopkg.in/yaml.v3 v3.0.0/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
@@ -803,36 +790,36 @@ honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWh
honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg=
honnef.co/go/tools v0.0.1-2020.1.3/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k=
honnef.co/go/tools v0.0.1-2020.1.4/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k=
-k8s.io/api v0.27.7 h1:7yG4D3t/q4utJe2ptlRw9aPuxcSmroTsYxsofkQNl/A=
-k8s.io/api v0.27.7/go.mod h1:ZNExI/Lhrs9YrLgVWx6jjHZdoWCTXfBXuFjt1X6olro=
-k8s.io/apiextensions-apiserver v0.27.7 h1:YqIOwZAUokzxJIjunmUd4zS1v3JhK34EPXn+pP0/bsU=
-k8s.io/apiextensions-apiserver v0.27.7/go.mod h1:x0p+b5a955lfPz9gaDeBy43obM12s+N9dNHK6+dUL+g=
-k8s.io/apimachinery v0.27.7 h1:Gxgtb7Y/Rsu8ymgmUEaiErkxa6RY4oTd8kNUI6SUR58=
-k8s.io/apimachinery v0.27.7/go.mod h1:jBGQgTjkw99ef6q5hv1YurDd3BqKDk9YRxmX0Ozo0i8=
-k8s.io/client-go v0.27.7 h1:+Xgh9OOKv6A3qdD4Dnl/0VOI5EvAv+0s/OseDxVVTwQ=
-k8s.io/client-go v0.27.7/go.mod h1:dZ2kqcalYp5YZ2EV12XIMc77G6PxHWOJp/kclZr4+5Q=
-k8s.io/component-base v0.27.7 h1:kngM58HR9W9Nqpv7e4rpdRyWnKl/ABpUhLAZ+HoliMs=
-k8s.io/component-base v0.27.7/go.mod h1:YGjlCVL1oeKvG3HSciyPHFh+LCjIEqsxz4BDR3cfHRs=
+k8s.io/api v0.28.8 h1:G0/G7yX1puRAcon/+XPLsKXZ9A5L7Ds6oKbDIe027xw=
+k8s.io/api v0.28.8/go.mod h1:rU8f1t9CNUAXlk/1j/wMJ7XnaxkR1g1AlZGQAOOL+sw=
+k8s.io/apiextensions-apiserver v0.28.8 h1:JucS9tcaMMlfFrJ09cgh1Maeb8X2wlnxcfNpplyGHXs=
+k8s.io/apiextensions-apiserver v0.28.8/go.mod h1:IKpLiKmvEYq/ti8sNtB1sM3A3vVV7fILIsvdmZswhoQ=
+k8s.io/apimachinery v0.28.8 h1:hi/nrxHwk4QLV+W/SHve1bypTE59HCDorLY1stBIxKQ=
+k8s.io/apimachinery v0.28.8/go.mod h1:cBnwIM3fXoRo28SqbV/Ihxf/iviw85KyXOrzxvZQ83U=
+k8s.io/client-go v0.28.8 h1:TE59Tjd87WKvS2FPBTfIKLFX0nQJ4SSHsnDo5IHjgOw=
+k8s.io/client-go v0.28.8/go.mod h1:uDVQ/rPzWpWIy40c6lZ4mUwaEvRWGnpoqSO4FM65P3o=
+k8s.io/component-base v0.28.8 h1:N/c5L6Ty5rcrFyhsMYsqRFUOVGrqGQsLfjB0yj6npqM=
+k8s.io/component-base v0.28.8/go.mod h1:9PjQ4nM1Hth6WGe/O+wgLF32eSwf4oPOoN5elmFznJM=
k8s.io/klog/v2 v2.120.1 h1:QXU6cPEOIslTGvZaXvFWiP9VKyeet3sawzTOvdXb4Vw=
k8s.io/klog/v2 v2.120.1/go.mod h1:3Jpz1GvMt720eyJH1ckRHK1EDfpxISzJ7I9OYgaDtPE=
-k8s.io/kube-openapi v0.0.0-20230501164219-8b0f38b5fd1f h1:2kWPakN3i/k81b0gvD5C5FJ2kxm1WrQFanWchyKuqGg=
-k8s.io/kube-openapi v0.0.0-20230501164219-8b0f38b5fd1f/go.mod h1:byini6yhqGC14c3ebc/QwanvYwhuMWF6yz2F8uwW8eg=
-k8s.io/kubectl v0.27.7 h1:HTEDa4s/oWjB3t5ysdW1yKlcNl9bzigcqWBq0LIIe3k=
-k8s.io/kubectl v0.27.7/go.mod h1:Xb1Ubc8uN1i2RvSN1HCgSHTtzgX0woihMk/gW7XbjJU=
+k8s.io/kube-openapi v0.0.0-20230717233707-2695361300d9 h1:LyMgNKD2P8Wn1iAwQU5OhxCKlKJy0sHc+PcDwFB24dQ=
+k8s.io/kube-openapi v0.0.0-20230717233707-2695361300d9/go.mod h1:wZK2AVp1uHCp4VamDVgBP2COHZjqD1T68Rf0CM3YjSM=
+k8s.io/kubectl v0.28.8 h1:bukHhQp4f7QYBFjS3iZTHKecrRpfZsR5lCCKFpbiUQw=
+k8s.io/kubectl v0.28.8/go.mod h1:X3u3VYbeWR1JirXkvtvq6b0kJ002M2ZQSk42M0ayKT8=
k8s.io/utils v0.0.0-20230505201702-9f6742963106 h1:EObNQ3TW2D+WptiYXlApGNLVy0zm/JIBVY9i+M4wpAU=
k8s.io/utils v0.0.0-20230505201702-9f6742963106/go.mod h1:OLgZIPagt7ERELqWJFomSt595RzquPNLL48iOWgYOg0=
-knative.dev/eventing v0.39.3 h1:I3/56fQ3t0t1cAG/6j4f0D+FNo+9avn1Ij2N9jXKSXM=
-knative.dev/eventing v0.39.3/go.mod h1:MlEiEcHALqZnu0OFMuWdJfjBzM7HlSJfk8pQbTcIy4o=
-knative.dev/pkg v0.0.0-20231023151236-29775d7c9e5c h1:xyPoEToTWeBdn6tinhLxXfnhJhTNQt5WzHiTNiFphRw=
-knative.dev/pkg v0.0.0-20231023151236-29775d7c9e5c/go.mod h1:HHRXEd7ZlFpthgE+rwAZ6MUVnuJOAeolnaFSthXloUQ=
+knative.dev/eventing v0.40.3 h1:Fjc5QqHRSuW6sLQ848bZvVAuurQAUbgnLMnwcn44Cc0=
+knative.dev/eventing v0.40.3/go.mod h1:+yUUIyvX9fn9bCSH3012kc8rG7YBbjvvxwy1Kr53dRc=
+knative.dev/pkg v0.0.0-20240116073220-b488e7be5902 h1:H6+JJN23fhwYWCHY1339sY6uhIyoUwDy1a8dN233fdk=
+knative.dev/pkg v0.0.0-20240116073220-b488e7be5902/go.mod h1:NYk8mMYoLkO7CQWnNkti4YGGnvLxN6MIDbUvtgeo0C0=
rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0=
rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA=
-sigs.k8s.io/controller-runtime v0.15.2 h1:9V7b7SDQSJ08IIsJ6CY1CE85Okhp87dyTMNDG0FS7f4=
-sigs.k8s.io/controller-runtime v0.15.2/go.mod h1:7ngYvp1MLT+9GeZ+6lH3LOlcHkp/+tzA/fmHa4iq9kk=
+sigs.k8s.io/controller-runtime v0.16.5 h1:yr1cEJbX08xsTW6XEIzT13KHHmIyX8Umvme2cULvFZw=
+sigs.k8s.io/controller-runtime v0.16.5/go.mod h1:j7bialYoSn142nv9sCOJmQgDXQXxnroFU4VnX/brVJ0=
sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd h1:EDPBXCAspyGV4jQlpZSudPeMmr1bNJefnuqLsRAsHZo=
sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd/go.mod h1:B8JuhiUyNFVKdsE8h686QcCxMaH6HrOAZj4vswFpcB0=
sigs.k8s.io/structured-merge-diff/v4 v4.2.3 h1:PRbqxJClWWYMNV1dhaG4NsibJbArud9kFxnAMREiWFE=
sigs.k8s.io/structured-merge-diff/v4 v4.2.3/go.mod h1:qjx8mGObPmV2aSZepjQjbmb2ihdVs8cGKBraizNC69E=
-sigs.k8s.io/yaml v1.3.0 h1:a2VclLzOGrwOHDiV8EfBGhvjHvP46CtW5j6POvhYGGo=
-sigs.k8s.io/yaml v1.3.0/go.mod h1:GeOyir5tyXNByN85N/dRIT9es5UQNerPYEKK56eTBm8=
+sigs.k8s.io/yaml v1.4.0 h1:Mk1wCc2gy/F0THH0TAp1QYyJNzRm2KCLy3o5ASXVI5E=
+sigs.k8s.io/yaml v1.4.0/go.mod h1:Ejl7/uTz7PSA4eKMyQCUTnhZYNmLIl+5c2lQPGR2BPY=
diff --git a/helm/yaks/crds/crd-instance.yaml b/helm/yaks/crds/crd-instance.yaml
index f08a7bd5..674aa37d 100644
--- a/helm/yaks/crds/crd-instance.yaml
+++ b/helm/yaks/crds/crd-instance.yaml
@@ -20,8 +20,7 @@ metadata:
labels:
app: yaks
annotations:
- controller-gen.kubebuilder.io/version: v0.6.1
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.15.0
name: instances.yaks.citrusframework.org
spec:
group: yaks.citrusframework.org
@@ -54,14 +53,19 @@ spec:
description: Instance is the Schema for the yaks instance.
properties:
apiVersion:
- description: 'APIVersion defines the versioned schema of this representation
- of an object. Servers should convert recognized schemas to the latest
- internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ description: |-
+ APIVersion defines the versioned schema of this representation of an object.
+ Servers should convert recognized schemas to the latest internal value, and
+ may reject unrecognized values.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources
type: string
kind:
- description: 'Kind is a string value representing the REST resource this
- object represents. Servers may infer this from the endpoint the client
- submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ description: |-
+ Kind is a string value representing the REST resource this object represents.
+ Servers may infer this from the endpoint the client submits requests to.
+ Cannot be updated.
+ In CamelCase.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds
type: string
metadata:
type: object
diff --git a/helm/yaks/crds/crd-test.yaml b/helm/yaks/crds/crd-test.yaml
index f96a1822..6c7d2915 100644
--- a/helm/yaks/crds/crd-test.yaml
+++ b/helm/yaks/crds/crd-test.yaml
@@ -20,8 +20,7 @@ metadata:
labels:
app: yaks
annotations:
- controller-gen.kubebuilder.io/version: v0.6.1
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.15.0
name: tests.yaks.citrusframework.org
spec:
group: yaks.citrusframework.org
@@ -66,14 +65,19 @@ spec:
description: Test is the Schema for the tests API.
properties:
apiVersion:
- description: 'APIVersion defines the versioned schema of this representation
- of an object. Servers should convert recognized schemas to the latest
- internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ description: |-
+ APIVersion defines the versioned schema of this representation of an object.
+ Servers should convert recognized schemas to the latest internal value, and
+ may reject unrecognized values.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources
type: string
kind:
- description: 'Kind is a string value representing the REST resource this
- object represents. Servers may infer this from the endpoint the client
- submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ description: |-
+ Kind is a string value representing the REST resource this object represents.
+ Servers may infer this from the endpoint the client submits requests to.
+ Cannot be updated.
+ In CamelCase.
+ More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds
type: string
metadata:
type: object
@@ -120,8 +124,10 @@ spec:
verbose:
type: boolean
type: object
- secret:
- type: string
+ secrets:
+ items:
+ type: string
+ type: array
selenium:
description: SeleniumSpec --.
properties:
diff --git a/java/docs/configuration.adoc b/java/docs/configuration.adoc
index c8d5969b..b2977bca 100644
--- a/java/docs/configuration.adoc
+++ b/java/docs/configuration.adoc
@@ -331,35 +331,37 @@ We can create a secret from that file and label it so that it will be picked up
kubectl create secret generic my-secret --from-file=mysecret.properties
----
-Once the secret is created you can bind it to tests by their name. Given the test `my-test.feature` you can bind the secret to the test
-by adding a label as follows:
+Once the secret is created you can bind it to a test via labels on the secret.
+Given the test named `my-test.feature` you can bind the secret to the test by adding a label as follows:
+.Bind secret to a test via labels
[source,shell script]
----
# Bind secret to the "my-test" test case
kubectl label secret my-secret yaks.citrusframework.org/test=my-test
----
-For multiple secrets and variants of secrets on different environments (e.g. dev, test, staging) you can add a secret id and label that one
-explicitly in addition to the test name.
+When running a test the YAKS operator searches for all secrets labelled with the test name to automatically mount the secret to the test pod container.
+All secret volumes get mounted to the test pod in `/etc/yaks/secrets` folder.
+You can access this folder in the test in order to load the secrets and their content.
-[source,shell script]
-----
-# Bind secret to the named configuration "staging" of the "my-test" test case
-kubectl label secret my-secret yaks.citrusframework.org/test=my-test yaks.citrusframework.org/test.configuration=staging
-----
-
-With that in place you just need to set the secret id in your `yaks-config.yaml` for that test.
+Instead of using the label based binding you can also specify a list of secret names on the test runtime config.
+Each secret gets mounted to the test pod container.
+With that in place you just need to set the secret name in your `yaks-config.yaml` for that test.
.yaks-config.yaml
[source,yaml]
----
config:
runtime:
- secret: staging
+ secrets:
+ - my-secret
----
-You can now write a test and use the secret properties as normal test variables:
+This assumes that there is a secret named `my-secret` available in the test namespace.
+You can now write a test and load the content of the secret (e.g. files).
+Also, property files in the secret get automatically loaded as test variables,
+so you can reference the secret properties with the normal test variable syntax:
.my-test.feature
[source,gherkin]
diff --git a/java/docs/steps-kamelet.adoc b/java/docs/steps-kamelet.adoc
index eedaa57f..a648bb5b 100644
--- a/java/docs/steps-kamelet.adoc
+++ b/java/docs/steps-kamelet.adoc
@@ -10,16 +10,16 @@ YAKS provides steps to manage Kamelets.
[[kamelet-api-version]]
=== API version
-The default Kamelet API version used to create and manage resources is `v1alpha1`. You can overwrite this
+The default Kamelet API version used to create and manage resources is `v1`. You can overwrite this
version with a environment variable set on the YAKS configuration.
.Overwrite Kamelet API version
[source,bash]
----
-YAKS_KAMELET_API_VERSION=v1
+YAKS_KAMELET_API_VERSION=v1alpha1
----
-This sets the Kamelet API version for all operations.
+This sets the Kamelet API version for all operations to `v1alpha1`.
[[kamelet-create]]
=== Create Kamelets
diff --git a/java/pom.xml b/java/pom.xml
index 77e6a6fd..fd9e4854 100644
--- a/java/pom.xml
+++ b/java/pom.xml
@@ -98,9 +98,10 @@
3.25.3
4.2.1
2.25.53
+ 2.3.3
4.2.1
2.12.0
- 7.16.1
+ 7.18.0
3.0.21
2.17.1
4.13.2
@@ -454,6 +455,13 @@
test
+
+
+ org.apache.camel.k
+ camel-k-crds
+ ${camel.k.crds.version}
+
+
com.fasterxml.jackson.core
jackson-core
diff --git a/java/runtime/yaks-runtime-core/src/main/java/org/citrusframework/yaks/config/YaksAutoConfiguration.java b/java/runtime/yaks-runtime-core/src/main/java/org/citrusframework/yaks/config/YaksAutoConfiguration.java
index e24a9b8a..759aa4c5 100644
--- a/java/runtime/yaks-runtime-core/src/main/java/org/citrusframework/yaks/config/YaksAutoConfiguration.java
+++ b/java/runtime/yaks-runtime-core/src/main/java/org/citrusframework/yaks/config/YaksAutoConfiguration.java
@@ -22,6 +22,7 @@
import org.citrusframework.functions.DefaultFunctionLibrary;
import org.citrusframework.functions.FunctionLibrary;
+import org.citrusframework.util.FileUtils;
import org.citrusframework.variable.GlobalVariables;
import org.citrusframework.variable.GlobalVariablesPropertyLoader;
import org.citrusframework.yaks.report.SystemOutTestReporter;
@@ -43,16 +44,16 @@ public GlobalVariables globalVariables() {
public GlobalVariablesPropertyLoader globalVariablesPropertyLoader() {
GlobalVariablesPropertyLoader propertyLoader = new GlobalVariablesPropertyLoader();
- // Load mounted secrets as global variables
+ // Load mounted property files from secrets as global variables
URL mountedSecrets = Thread.currentThread().getContextClassLoader().getResource("secrets");
if (mountedSecrets != null) {
File secretsDir = new File(mountedSecrets.getPath());
if (secretsDir.exists() && secretsDir.isDirectory()) {
File[] propertyFiles = secretsDir.listFiles();
if (propertyFiles != null) {
- Stream.of(propertyFiles).forEach(file -> {
- propertyLoader.getPropertyFiles().add("file:" + file.getPath());
- });
+ Stream.of(propertyFiles)
+ .filter(file -> "properties".equals(FileUtils.getFileExtension(file.getName())))
+ .forEach(file -> propertyLoader.getPropertyFiles().add("file:" + file.getPath()));
}
}
}
diff --git a/java/steps/pom.xml b/java/steps/pom.xml
index 9acb5d12..41ca3dd6 100644
--- a/java/steps/pom.xml
+++ b/java/steps/pom.xml
@@ -21,6 +21,7 @@
org.citrusframework.yaks
yaks-parent
0.20.0-SNAPSHOT
+ ../pom.xml
4.0.0
diff --git a/java/steps/yaks-camel-k/pom.xml b/java/steps/yaks-camel-k/pom.xml
index 0312d871..7536a1ac 100644
--- a/java/steps/yaks-camel-k/pom.xml
+++ b/java/steps/yaks-camel-k/pom.xml
@@ -67,6 +67,11 @@
kubernetes-client
+
+ org.apache.camel.k
+ camel-k-crds
+
+
com.squareup.okhttp3
okhttp
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/CamelKSettings.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/CamelKSettings.java
index 5d441d52..66177a1f 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/CamelKSettings.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/CamelKSettings.java
@@ -60,6 +60,10 @@ public final class CamelKSettings {
private static final String SUPPORT_VARIABLES_IN_SOURCES_ENV = CAMELK_ENV_PREFIX + "SUPPORT_VARIABLES_IN_SOURCES";
private static final String SUPPORT_VARIABLES_IN_SOURCES_DEFAULT = "true";
+ private static final String STOP_ON_ERROR_STATUS_PROPERTY = CAMELK_PROPERTY_PREFIX + "stop.on.error.status";
+ private static final String STOP_ON_ERROR_STATUS_ENV = CAMELK_ENV_PREFIX + "STOP_ON_ERROR_STATUS";
+ private static final String STOP_ON_ERROR_STATUS_DEFAULT = "true";
+
private static final String PRINT_POD_LOGS_PROPERTY = CAMELK_PROPERTY_PREFIX + "print.pod.logs";
private static final String PRINT_POD_LOGS_ENV = CAMELK_ENV_PREFIX + "PRINT_POD_LOGS";
private static final String PRINT_POD_LOGS_DEFAULT = String.valueOf(KubernetesSettings.isPrintPodLogs());
@@ -137,6 +141,16 @@ public static boolean isSupportVariablesInSources() {
System.getenv(SUPPORT_VARIABLES_IN_SOURCES_ENV) != null ? System.getenv(SUPPORT_VARIABLES_IN_SOURCES_ENV) : SUPPORT_VARIABLES_IN_SOURCES_DEFAULT));
}
+ /**
+ * When set to true YAKS will stop the integration verification when integration is in error state.
+ * When verifying the integration status and pod logs the retry mechanism stops when integration status condition is in error state.
+ * @return
+ */
+ public static boolean isStopOnErrorStatus() {
+ return Boolean.parseBoolean(System.getProperty(STOP_ON_ERROR_STATUS_PROPERTY,
+ System.getenv(STOP_ON_ERROR_STATUS_ENV) != null ? System.getenv(STOP_ON_ERROR_STATUS_ENV) : STOP_ON_ERROR_STATUS_DEFAULT));
+ }
+
/**
* When set to true test will print pod logs e.g. while waiting for a pod log message.
* @return
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/CamelKSteps.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/CamelKSteps.java
index afd231c9..cc8438ae 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/CamelKSteps.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/CamelKSteps.java
@@ -35,7 +35,6 @@
import org.citrusframework.context.TestContext;
import org.citrusframework.exceptions.ActionTimeoutException;
import org.citrusframework.exceptions.CitrusRuntimeException;
-import org.citrusframework.spi.Resource;
import org.citrusframework.util.FileUtils;
import org.citrusframework.yaks.camelk.actions.integration.CreateIntegrationAction;
import org.citrusframework.yaks.kubernetes.KubernetesSupport;
@@ -67,8 +66,11 @@ public class CamelKSteps {
private Map properties;
private List buildPropertyFiles;
private Map buildProperties;
+ private List envVarFiles;
+ private Map envVars;
private boolean supportVariablesInSources = CamelKSettings.isSupportVariablesInSources();
+ private boolean stopOnErrorStatus = CamelKSettings.isStopOnErrorStatus();
@Before
public void before(Scenario scenario) {
@@ -80,6 +82,8 @@ public void before(Scenario scenario) {
properties = new LinkedHashMap<>();
buildPropertyFiles = new ArrayList<>();
buildProperties = new LinkedHashMap<>();
+ envVarFiles = new ArrayList<>();
+ envVars = new LinkedHashMap<>();
if (!context.getVariables().containsKey(VariableNames.OPERATOR_NAMESPACE.value())) {
context.setVariable(VariableNames.OPERATOR_NAMESPACE.value(), CamelKSettings.getOperatorNamespace());
@@ -106,6 +110,16 @@ public void enableVariableSupport() {
supportVariablesInSources = true;
}
+ @Given("^Disable stop on error status$")
+ public void disableStopOnErrorStatus() {
+ stopOnErrorStatus = false;
+ }
+
+ @Given("^Enable stop on error status$")
+ public void enableStopOnErrorStatus() {
+ stopOnErrorStatus = true;
+ }
+
@Given("^Camel K resource polling configuration$")
public void configureResourcePolling(Map configuration) {
maxAttempts = Integer.parseInt(configuration.getOrDefault("maxAttempts", maxAttempts).toString());
@@ -156,6 +170,22 @@ public void addBuildProperties(DataTable propertyTable) {
buildProperties.putAll(propertyTable.asMap(String.class, String.class));
}
+ @Given("^Camel K integration environment variable file ([^\\s]+)$")
+ public void addEnVarFile(String filePath) {
+ envVarFiles.add(filePath);
+ }
+
+ @Given("^Camel K integration environment variable ([^\\s]+)=\"([^\"]*)\"$")
+ @Given("^Camel K integration environment variable ([^\\s]+) (?:is|=) \"([^\"]*)\"$")
+ public void addEnVar(String name, String value) {
+ envVars.put(name, value);
+ }
+
+ @Given("^Camel K integration environment variables$")
+ public void addEnvVars(DataTable propertyTable) {
+ envVars.putAll(propertyTable.asMap(String.class, String.class));
+ }
+
@Given("^(?:create|new) Camel K integration ([a-z0-9][a-z0-9-\\.]+[a-z0-9])\\.([a-z0-9-]+) with configuration:?$")
public void createIntegration(String name, String language, Map configuration) {
if (configuration.get("source") == null) {
@@ -192,6 +222,8 @@ public void createIntegration(String name, String language, String source) {
.propertyFiles(propertyFiles)
.buildProperties(buildProperties)
.buildPropertyFiles(buildPropertyFiles)
+ .envVars(envVars)
+ .envVarFiles(envVarFiles)
.supportVariables(supportVariablesInSources)
.source(name + "." + language, source));
@@ -242,6 +274,7 @@ public void integrationShouldPrintMultiline(String name, String message) {
.client(k8sClient)
.verifyIntegration(name)
.printLogs(CamelKSettings.isPrintPodLogs())
+ .stopOnErrorStatus(stopOnErrorStatus)
.maxAttempts(maxAttempts)
.delayBetweenAttempts(delayBetweenAttempts)
.waitForLogMessage(message));
@@ -260,6 +293,7 @@ public void integrationShouldNotPrintMultiline(String name, String message) {
.client(k8sClient)
.verifyIntegration(name)
.printLogs(CamelKSettings.isPrintPodLogs())
+ .stopOnErrorStatus(stopOnErrorStatus)
.maxAttempts(maxAttempts)
.delayBetweenAttempts(delayBetweenAttempts)
.waitForLogMessage(message)));
@@ -274,6 +308,8 @@ private void createIntegration(String name, String language, String source, Map<
.buildProperties(configuration.getOrDefault("build-properties", "").trim())
.buildProperties(buildProperties)
.buildPropertyFiles(buildPropertyFiles)
+ .envVars(envVars)
+ .envVarFiles(envVarFiles)
.properties(configuration.getOrDefault("properties", "").trim())
.properties(properties)
.propertyFiles(propertyFiles)
@@ -283,12 +319,7 @@ private void createIntegration(String name, String language, String source, Map<
String openApiSpec = configuration.getOrDefault("openapi", "");
if (!openApiSpec.isEmpty()) {
- try {
- Resource file = ResourceUtils.resolve(openApiSpec, context);
- create.openApi(FileUtils.getFileName(file.getLocation()), FileUtils.readToString(file));
- } catch (IOException e) {
- throw new CitrusRuntimeException(String.format("Failed to read openapi spec form file path %s", openApiSpec));
- }
+ create.openApi(openApiSpec);
}
runner.run(create);
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/KameletBindingSteps.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/KameletBindingSteps.java
new file mode 100644
index 00000000..a3b465de
--- /dev/null
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/KameletBindingSteps.java
@@ -0,0 +1,233 @@
+/*
+ * Copyright the original author or 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 org.citrusframework.yaks.camelk;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import io.cucumber.java.Before;
+import io.cucumber.java.Scenario;
+import io.cucumber.java.en.Given;
+import io.cucumber.java.en.Then;
+import io.fabric8.kubernetes.client.KubernetesClient;
+import org.apache.camel.v1alpha1.KameletBindingBuilder;
+import org.apache.camel.v1alpha1.kameletbindingspec.SinkBuilder;
+import org.apache.camel.v1alpha1.kameletbindingspec.SourceBuilder;
+import org.apache.camel.v1alpha1.kameletbindingspec.source.Ref;
+import org.apache.camel.v1alpha1.kameletbindingspec.source.RefBuilder;
+import org.citrusframework.Citrus;
+import org.citrusframework.TestCaseRunner;
+import org.citrusframework.annotations.CitrusFramework;
+import org.citrusframework.annotations.CitrusResource;
+import org.citrusframework.context.TestContext;
+import org.citrusframework.spi.Resource;
+import org.citrusframework.yaks.kafka.KafkaSettings;
+import org.citrusframework.yaks.knative.KnativeSettings;
+import org.citrusframework.yaks.kubernetes.KubernetesSupport;
+import org.citrusframework.yaks.util.ResourceUtils;
+
+import static org.citrusframework.container.FinallySequence.Builder.doFinally;
+import static org.citrusframework.yaks.camelk.actions.CamelKActionBuilder.camelk;
+
+public class KameletBindingSteps {
+
+ @CitrusResource
+ private TestCaseRunner runner;
+
+ @CitrusFramework
+ private Citrus citrus;
+
+ @CitrusResource
+ private TestContext context;
+
+ private KubernetesClient k8sClient;
+
+ // Binding endpoints
+ private SourceBuilder source;
+ private SinkBuilder sink;
+
+ private Map sourceProperties;
+ private Map sinkProperties;
+
+ @Before
+ public void before(Scenario scenario) {
+ if (k8sClient == null) {
+ k8sClient = KubernetesSupport.getKubernetesClient(citrus);
+ }
+
+ initializeKameletBindingBuilder();
+ }
+
+ @Given("^KameletBinding source properties$")
+ public void setKameletBindingSourceProperties(Map properties) {
+ this.sourceProperties.putAll(properties);
+ }
+
+ @Given("^KameletBinding sink properties$")
+ public void setKameletBindingSinkProperties(Map properties) {
+ this.sinkProperties.putAll(properties);
+ }
+
+ @Given("^KameletBinding event source Kamelet ([a-z0-9-]+)$")
+ public void setKameletEventSource(String kameletName) {
+ Ref kameletRef = new RefBuilder()
+ .withName(kameletName)
+ .withApiVersion(CamelKSupport.CAMELK_CRD_GROUP + "/" + getKameletApiVersion())
+ .withKind("Kamelet")
+ .withNamespace(getNamespace())
+ .build();
+ source = new SourceBuilder().withRef(kameletRef);
+ }
+
+ @Given("^KameletBinding event sink uri ([^\\s]+)$")
+ public void setEventSinkUri(String uri) {
+ sink = new SinkBuilder().withUri(uri);
+ }
+
+ @Given("^KameletBinding event sink Kafka topic ([^\\s]+)$")
+ public void setEventSinkKafkaTopic(String topic) {
+ org.apache.camel.v1alpha1.kameletbindingspec.sink.Ref sinkRef =
+ new org.apache.camel.v1alpha1.kameletbindingspec.sink.RefBuilder()
+ .withName(topic)
+ .withApiVersion("kafka.strimzi.io/" + KafkaSettings.getApiVersion())
+ .withKind("KafkaTopic")
+ .withNamespace(KafkaSettings.getNamespace())
+ .build();
+ sink = new SinkBuilder().withRef(sinkRef);
+ }
+
+ @Given("^KameletBinding event sink Knative channel ([^\\s]+)$")
+ public void setEventSinkKnativeChannel(String channel) {
+ setEventSinkKnativeChannel(channel, "InMemoryChannel");
+ }
+
+ @Given("^KameletBinding event sink Knative channel ([^\\s]+) of kind ([^\\s]+)$")
+ public void setEventSinkKnativeChannel(String channel, String channelKind) {
+ org.apache.camel.v1alpha1.kameletbindingspec.sink.Ref sinkRef =
+ new org.apache.camel.v1alpha1.kameletbindingspec.sink.RefBuilder()
+ .withName(channel)
+ .withApiVersion("messaging.knative.dev/" + KnativeSettings.getApiVersion())
+ .withKind(channelKind)
+ .withNamespace(KnativeSettings.getNamespace())
+ .build();
+ sink = new SinkBuilder().withRef(sinkRef);
+ }
+
+ @Given("^KameletBinding event sink Knative broker ([^\\s]+)$")
+ public void setEventSinkKnativeBroker(String broker) {
+ org.apache.camel.v1alpha1.kameletbindingspec.sink.Ref sinkRef =
+ new org.apache.camel.v1alpha1.kameletbindingspec.sink.RefBuilder()
+ .withName(broker)
+ .withApiVersion("eventing.knative.dev/" + KnativeSettings.getApiVersion())
+ .withKind("Broker")
+ .withNamespace(KnativeSettings.getNamespace())
+ .build();
+ sink = new SinkBuilder().withRef(sinkRef);
+ }
+
+ @Given("^load KameletBinding ([a-z0-9-]+).yaml$")
+ public void loadKameletBindingFromFile(String fileName) {
+ Resource resource = ResourceUtils.resolve(fileName + ".yaml", context);
+ runner.run(camelk()
+ .client(k8sClient)
+ .createKameletBinding(fileName)
+ .resource(resource));
+
+ if (isAutoRemoveResources()) {
+ runner.then(doFinally()
+ .actions(camelk().client(k8sClient)
+ .deleteKameletBinding(fileName)));
+ }
+ }
+
+ @Given("^(?:create|new) KameletBinding ([a-z0-9-]+)$")
+ public void createNewKameletBinding(String name) {
+ KameletBindingBuilder builder = new KameletBindingBuilder();
+
+ builder.withNewMetadata()
+ .withName(name)
+ .endMetadata();
+
+ source.editOrNewProperties().addToAdditionalProperties(sourceProperties);
+ sink.editOrNewProperties().addToAdditionalProperties(sinkProperties);
+
+ builder.withNewSpec()
+ .withSource(source.build())
+ .withSink(sink.build())
+ .endSpec();
+
+ runner.run(camelk()
+ .client(k8sClient)
+ .createKameletBinding(name)
+ .fromBuilder(builder));
+
+ initializeKameletBindingBuilder();
+
+ if (isAutoRemoveResources()) {
+ runner.then(doFinally()
+ .actions(camelk().client(k8sClient)
+ .deleteKameletBinding(name)));
+ }
+ }
+
+ @Given("^delete KameletBinding ([a-z0-9-]+)$")
+ public void deleteKameletBinding(String name) {
+ runner.run(camelk()
+ .client(k8sClient)
+ .deleteKameletBinding(name));
+ }
+
+ @Given("^KameletBinding ([a-z0-9-]+) is available$")
+ @Then("^KameletBinding ([a-z0-9-]+) should be available$")
+ public void bindingShouldBeAvailable(String name) {
+ runner.run(camelk()
+ .client(k8sClient)
+ .verifyKameletBinding(name)
+ .isAvailable());
+ }
+
+ private void initializeKameletBindingBuilder() {
+ source = null;
+ sink = null;
+ sourceProperties = new HashMap<>();
+ sinkProperties = new HashMap<>();
+ }
+
+ private String getKameletApiVersion() {
+ if (context.getVariables().containsKey(VariableNames.KAMELET_API_VERSION.value())) {
+ return context.getVariable(VariableNames.KAMELET_API_VERSION.value());
+ }
+
+ return KameletSettings.getKameletApiVersion();
+ }
+
+ private String getNamespace() {
+ if (context.getVariables().containsKey(VariableNames.KAMELET_NAMESPACE.value())) {
+ return context.getVariable(VariableNames.KAMELET_NAMESPACE.value());
+ }
+
+ return KameletSettings.getNamespace();
+ }
+
+ private boolean isAutoRemoveResources() {
+ if (context.getVariables().containsKey(VariableNames.AUTO_REMOVE_RESOURCES.value())) {
+ return context.getVariable(VariableNames.AUTO_REMOVE_RESOURCES.value(), Boolean.class);
+ }
+
+ return CamelKSettings.isAutoRemoveResources();
+ }
+}
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/KameletSteps.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/KameletSteps.java
index 1e5ebc75..2883c901 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/KameletSteps.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/KameletSteps.java
@@ -23,7 +23,16 @@
import io.cucumber.java.Scenario;
import io.cucumber.java.en.Given;
import io.cucumber.java.en.Then;
+import io.fabric8.kubernetes.api.model.AnyTypeBuilder;
import io.fabric8.kubernetes.client.KubernetesClient;
+import org.apache.camel.v1.KameletBuilder;
+import org.apache.camel.v1.KameletSpecBuilder;
+import org.apache.camel.v1.kameletspec.DataTypes;
+import org.apache.camel.v1.kameletspec.DefinitionBuilder;
+import org.apache.camel.v1.kameletspec.SourcesBuilder;
+import org.apache.camel.v1.kameletspec.datatypes.TypesBuilder;
+import org.apache.camel.v1.kameletspec.definition.Properties;
+import org.apache.camel.v1.kameletspec.definition.PropertiesBuilder;
import org.citrusframework.Citrus;
import org.citrusframework.TestCaseRunner;
import org.citrusframework.annotations.CitrusFramework;
@@ -31,12 +40,6 @@
import org.citrusframework.context.TestContext;
import org.citrusframework.exceptions.CitrusRuntimeException;
import org.citrusframework.spi.Resource;
-import org.citrusframework.yaks.camelk.model.Kamelet;
-import org.citrusframework.yaks.camelk.model.KameletSpec;
-import org.citrusframework.yaks.camelk.model.Pipe;
-import org.citrusframework.yaks.camelk.model.PipeSpec;
-import org.citrusframework.yaks.kafka.KafkaSettings;
-import org.citrusframework.yaks.knative.KnativeSettings;
import org.citrusframework.yaks.kubernetes.KubernetesSupport;
import org.citrusframework.yaks.util.ResourceUtils;
import org.springframework.util.StringUtils;
@@ -61,15 +64,11 @@ public class KameletSteps {
private String kameletApiVersion = KameletSettings.getKameletApiVersion();
// Kamelet builder
- private Kamelet.Builder kamelet;
- private KameletSpec.Definition definition;
-
- // Pipe endpoints
- private PipeSpec.Endpoint source;
- private PipeSpec.Endpoint sink;
-
- private Map sourceProperties;
- private Map sinkProperties;
+ private KameletBuilder kamelet;
+ private KameletSpecBuilder kameletSpecBuilder;
+ private Map dataTypes;
+ private DefinitionBuilder definition;
+ private String kameletTemplate;
private String namespace = KameletSettings.getNamespace();
@@ -83,17 +82,22 @@ public void before(Scenario scenario) {
}
initializeKameletBuilder();
- initializePipeBuilder();
}
@Given("^Disable auto removal of Kamelet resources$")
public void disableAutoRemove() {
autoRemoveResources = false;
+
+ // update the test variable
+ runner.run(createVariable(VariableNames.AUTO_REMOVE_RESOURCES.value(), "false"));
}
@Given("^Enable auto removal of Kamelet resources$")
public void enableAutoRemove() {
autoRemoveResources = true;
+
+ // update the test variable
+ runner.run(createVariable(VariableNames.AUTO_REMOVE_RESOURCES.value(), "true"));
}
@Given("^Disable variable support in Kamelet sources$")
@@ -124,27 +128,39 @@ public void setNamespace(String namespace) {
@Given("^Kamelet dataType (in|out|error)(?:=| is )\"(.+)\"$")
public void addType(String slot, String format) {
+ DataTypes dt;
+ if (dataTypes.containsKey(slot)) {
+ dt = dataTypes.get(slot);
+ } else {
+ dt = new DataTypes();
+ dataTypes.put(slot, dt);
+ }
+
+ if (dt.getTypes() == null) {
+ dt.setTypes(new HashMap<>());
+ }
+
if (format.contains(":")) {
String[] schemeAndFormat = format.split(":");
- kamelet.addDataType(slot, schemeAndFormat[0], schemeAndFormat[1]);
+ dt.getTypes().put(schemeAndFormat[1], new TypesBuilder().withScheme(schemeAndFormat[0]).withFormat(schemeAndFormat[1]).build());
} else {
- kamelet.addDataType(slot, "camel", format);
+ dt.getTypes().put(format, new TypesBuilder().withScheme("camel").withFormat(format).build());
}
}
@Given("^Kamelet title \"(.+)\"$")
public void setTitle(String title) {
- definition.setTitle(title);
+ definition.withTitle(title);
}
@Given("^Kamelet source ([a-z0-9-]+).([a-z0-9-]+)$")
public void setSource(String name, String language, String content) {
- kamelet.source(name, language, content);
+ kameletSpecBuilder.withSources(new SourcesBuilder().withName(name).withLanguage(language).withContent(content).build());
}
@Given("^Kamelet template")
public void setFlow(String template) {
- kamelet.template(template);
+ kameletTemplate = template;
}
@Given("^Kamelet property definition$")
@@ -165,58 +181,19 @@ public void addPropertyDefinition(String propertyName, Map prope
String required = propertyConfiguration.getOrDefault("required", Boolean.FALSE).toString();
if (Boolean.parseBoolean(required)) {
- definition.getRequired().add(propertyName);
+ definition.addToRequired(propertyName);
}
- definition.getProperties().put(propertyName,
- new KameletSpec.Definition.PropertyConfig(title, type, defaultValue, example));
- }
-
- @Given("^(?:Pipe|KameletBinding) source properties$")
- public void setPipeSourceProperties(Map properties) {
- this.sourceProperties.putAll(properties);
- }
-
- @Given("^(?:Pipe|KameletBinding) sink properties$")
- public void setPipeSinkProperties(Map properties) {
- this.sinkProperties.putAll(properties);
- }
-
- @Given("^bind Kamelet ([a-z0-9-]+) to uri ([^\\s]+)$")
- public void bindKameletToUri(String kameletName, String uri) {
- PipeSpec.Endpoint.ObjectReference sourceRef =
- new PipeSpec.Endpoint.ObjectReference(CamelKSupport.CAMELK_CRD_GROUP + "/" + kameletApiVersion, "Kamelet", namespace, kameletName);
- source = new PipeSpec.Endpoint(sourceRef);
-
- sink = new PipeSpec.Endpoint(uri);
- }
-
- @Given("^bind Kamelet ([a-z0-9-]+) to Kafka topic ([^\\s]+)$")
- public void bindKameletToKafka(String kameletName, String topic) {
- PipeSpec.Endpoint.ObjectReference sourceRef =
- new PipeSpec.Endpoint.ObjectReference(CamelKSupport.CAMELK_CRD_GROUP + "/" + kameletApiVersion, "Kamelet", namespace, kameletName);
- source = new PipeSpec.Endpoint(sourceRef);
-
- PipeSpec.Endpoint.ObjectReference sinkRef =
- new PipeSpec.Endpoint.ObjectReference("KafkaTopic", KafkaSettings.getNamespace(), topic);
- sink = new PipeSpec.Endpoint(sinkRef);
- }
-
- @Given("^bind Kamelet ([a-z0-9-]+) to Knative channel ([^\\s]+)$")
- public void bindKameletToKnativeChannel(String kameletName, String channel) {
- bindKameletToKnativeChannel(kameletName, channel, "InMemoryChannel");
- }
-
- @Given("^bind Kamelet ([a-z0-9-]+) to Knative channel ([^\\s]+) of kind ([^\\s]+)$")
- public void bindKameletToKnativeChannel(String kameletName, String channel, String channelKind) {
- PipeSpec.Endpoint.ObjectReference sourceRef =
- new PipeSpec.Endpoint.ObjectReference(CamelKSupport.CAMELK_CRD_GROUP + "/" + kameletApiVersion, "Kamelet", namespace, kameletName);
- source = new PipeSpec.Endpoint(sourceRef);
+ Properties property = new PropertiesBuilder().withTitle(title).withType(type).build();
+ if (example != null) {
+ property.setExample(new AnyTypeBuilder().withValue(example).build());
+ }
- PipeSpec.Endpoint.ObjectReference sinkRef =
- new PipeSpec.Endpoint.ObjectReference(channelKind, KnativeSettings.getNamespace(), channel);
- sink = new PipeSpec.Endpoint(sinkRef);
- }
+ if (defaultValue != null) {
+ property.set_default(new AnyTypeBuilder().withValue(defaultValue).build());
+ }
+ definition.addToProperties(propertyName, property);
+ }
@Given("^load Kamelet ([a-z0-9-]+).kamelet.yaml$")
public void loadKameletFromFile(String fileName) {
@@ -224,6 +201,7 @@ public void loadKameletFromFile(String fileName) {
runner.run(camelk()
.client(k8sClient)
.createKamelet(fileName)
+ .namespace(namespace)
.apiVersion(kameletApiVersion)
.supportVariables(supportVariablesInSources)
.resource(resource));
@@ -236,36 +214,26 @@ public void loadKameletFromFile(String fileName) {
}
}
- @Given("^load (?:Pipe|KameletBinding) ([a-z0-9-]+).yaml$")
- public void loadPipeFromFile(String fileName) {
- Resource resource = ResourceUtils.resolve(fileName + ".yaml", context);
- runner.run(camelk()
- .client(k8sClient)
- .createPipe(fileName)
- .resource(resource));
-
- if (autoRemoveResources) {
- runner.then(doFinally()
- .actions(camelk().client(k8sClient)
- .deletePipe(fileName)
- .apiVersion(kameletApiVersion)));
- }
- }
-
@Given("^(?:create|new) Kamelet ([a-z0-9-]+)$")
public void createNewKamelet(String name) {
- kamelet.name(name);
+ kamelet.withNewMetadata()
+ .withName(name)
+ .endMetadata();
if (definition.getTitle() == null || definition.getTitle().isEmpty()) {
- definition.setTitle(StringUtils.capitalize(name));
+ definition.withTitle(StringUtils.capitalize(name));
}
- kamelet.definition(definition);
+ kameletSpecBuilder.withDefinition(definition.build());
+ kameletSpecBuilder.withDataTypes(dataTypes);
+
+ kamelet.withSpec(kameletSpecBuilder.build());
runner.run(camelk()
.client(k8sClient)
.createKamelet(name)
.supportVariables(supportVariablesInSources)
+ .template(kameletTemplate)
.fromBuilder(kamelet));
initializeKameletBuilder();
@@ -286,37 +254,10 @@ public void createNewKameletWithFlow(String name, String flow) {
@Given("^(?:create|new) Kamelet ([a-z0-9-]+) with template")
public void createNewKameletWithTemplate(String name, String template) {
- kamelet.template(template);
+ kameletTemplate = template;
createNewKamelet(name);
}
- @Given("^(?:create|new) (?:Pipe|KameletBinding) ([a-z0-9-]+)$")
- public void createNewPipe(String name) {
- Pipe.Builder pipe = new Pipe.Builder();
- pipe.name(name);
-
- source.getProperties().putAll(sourceProperties);
- sink.getProperties().putAll(sinkProperties);
-
- pipe.source(source);
- pipe.sink(sink);
-
- runner.run(camelk()
- .client(k8sClient)
- .createPipe(name)
- .apiVersion(kameletApiVersion)
- .fromBuilder(pipe));
-
- initializePipeBuilder();
-
- if (autoRemoveResources) {
- runner.then(doFinally()
- .actions(camelk().client(k8sClient)
- .deletePipe(name)
- .apiVersion(kameletApiVersion)));
- }
- }
-
@Given("^delete Kamelet ([a-z0-9-]+)$")
public void deleteKamelet(String name) {
runner.run(camelk()
@@ -325,14 +266,6 @@ public void deleteKamelet(String name) {
.apiVersion(kameletApiVersion));
}
- @Given("^delete (?:Pipe|KameletBinding) ([a-z0-9-]+)$")
- public void deletePipe(String name) {
- runner.run(camelk()
- .client(k8sClient)
- .deletePipe(name)
- .apiVersion(kameletApiVersion));
- }
-
@Given("^Kamelet ([a-z0-9-]+) is available$")
@Then("^Kamelet ([a-z0-9-]+) should be available$")
public void kameletShouldBeAvailable(String name) {
@@ -354,25 +287,11 @@ public void kameletShouldBeAvailable(String name, String namespace) {
.isAvailable());
}
- @Given("^(?:Pipe|KameletBinding) ([a-z0-9-]+) is available$")
- @Then("^(?:Pipe|KameletBinding) ([a-z0-9-]+) should be available$")
- public void pipeShouldBeAvailable(String name) {
- runner.run(camelk()
- .client(k8sClient)
- .verifyPipe(name)
- .apiVersion(kameletApiVersion)
- .isAvailable());
- }
-
private void initializeKameletBuilder() {
- kamelet = new Kamelet.Builder();
- definition = new KameletSpec.Definition();
- }
-
- private void initializePipeBuilder() {
- source = null;
- sink = null;
- sourceProperties = new HashMap<>();
- sinkProperties = new HashMap<>();
+ kamelet = new KameletBuilder();
+ definition = new DefinitionBuilder();
+ kameletSpecBuilder = new KameletSpecBuilder();
+ dataTypes = new HashMap<>();
+ kameletTemplate = null;
}
}
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/PipeSteps.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/PipeSteps.java
new file mode 100644
index 00000000..c98a4c07
--- /dev/null
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/PipeSteps.java
@@ -0,0 +1,262 @@
+/*
+ * Copyright the original author or 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 org.citrusframework.yaks.camelk;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import io.cucumber.java.Before;
+import io.cucumber.java.Scenario;
+import io.cucumber.java.en.Given;
+import io.cucumber.java.en.Then;
+import io.fabric8.kubernetes.client.KubernetesClient;
+import org.apache.camel.v1.PipeBuilder;
+import org.apache.camel.v1.pipespec.SinkBuilder;
+import org.apache.camel.v1.pipespec.SourceBuilder;
+import org.apache.camel.v1.pipespec.source.Ref;
+import org.apache.camel.v1.pipespec.source.RefBuilder;
+import org.citrusframework.Citrus;
+import org.citrusframework.TestCaseRunner;
+import org.citrusframework.annotations.CitrusFramework;
+import org.citrusframework.annotations.CitrusResource;
+import org.citrusframework.context.TestContext;
+import org.citrusframework.spi.Resource;
+import org.citrusframework.yaks.kafka.KafkaSettings;
+import org.citrusframework.yaks.knative.KnativeSettings;
+import org.citrusframework.yaks.kubernetes.KubernetesSupport;
+import org.citrusframework.yaks.util.ResourceUtils;
+
+import static org.citrusframework.container.FinallySequence.Builder.doFinally;
+import static org.citrusframework.yaks.camelk.actions.CamelKActionBuilder.camelk;
+
+public class PipeSteps {
+
+ @CitrusResource
+ private TestCaseRunner runner;
+
+ @CitrusFramework
+ private Citrus citrus;
+
+ @CitrusResource
+ private TestContext context;
+
+ private KubernetesClient k8sClient;
+
+ // Pipe endpoints
+ private SourceBuilder source;
+ private SinkBuilder sink;
+
+ private Map sourceProperties;
+ private Map sinkProperties;
+
+ @Before
+ public void before(Scenario scenario) {
+ if (k8sClient == null) {
+ k8sClient = KubernetesSupport.getKubernetesClient(citrus);
+ }
+
+ initializePipeBuilder();
+ }
+
+ @Given("^Pipe source properties$")
+ public void setPipeSourceProperties(Map properties) {
+ this.sourceProperties.putAll(properties);
+ }
+
+ @Given("^Pipe sink properties$")
+ public void setPipeSinkProperties(Map properties) {
+ this.sinkProperties.putAll(properties);
+ }
+
+ @Given("^Pipe event source Kamelet ([a-z0-9-]+)$")
+ public void setKameletEventSource(String kameletName) {
+ Ref kameletRef = new RefBuilder()
+ .withName(kameletName)
+ .withApiVersion(CamelKSupport.CAMELK_CRD_GROUP + "/" + getKameletApiVersion())
+ .withKind("Kamelet")
+ .withNamespace(getNamespace())
+ .build();
+ source = new SourceBuilder().withRef(kameletRef);
+ }
+
+ @Given("^Pipe event sink uri ([^\\s]+)$")
+ public void setEventSinkUri(String uri) {
+ sink = new SinkBuilder().withUri(uri);
+ }
+
+ @Given("^Pipe event sink Kafka topic ([^\\s]+)$")
+ public void setEventSinkKafkaTopic(String topic) {
+ org.apache.camel.v1.pipespec.sink.Ref sinkRef =
+ new org.apache.camel.v1.pipespec.sink.RefBuilder()
+ .withName(topic)
+ .withApiVersion("kafka.strimzi.io/" + KafkaSettings.getApiVersion())
+ .withKind("KafkaTopic")
+ .withNamespace(KafkaSettings.getNamespace())
+ .build();
+ sink = new SinkBuilder().withRef(sinkRef);
+ }
+
+ @Given("^Pipe event sink Knative channel ([^\\s]+)$")
+ public void setEventSinkKnativeChannel(String channel) {
+ setEventSinkKnativeChannel(channel, "InMemoryChannel");
+ }
+
+ @Given("^Pipe event sink Knative channel ([^\\s]+) of kind ([^\\s]+)$")
+ public void setEventSinkKnativeChannel(String channel, String channelKind) {
+ org.apache.camel.v1.pipespec.sink.Ref sinkRef =
+ new org.apache.camel.v1.pipespec.sink.RefBuilder()
+ .withName(channel)
+ .withApiVersion("messaging.knative.dev/" + KnativeSettings.getApiVersion())
+ .withKind(channelKind)
+ .withNamespace(KnativeSettings.getNamespace())
+ .build();
+ sink = new SinkBuilder().withRef(sinkRef);
+ }
+
+ @Given("^Pipe event sink Knative broker ([^\\s]+)$")
+ public void setEventSinkKnativeBroker(String broker) {
+ org.apache.camel.v1.pipespec.sink.Ref sinkRef =
+ new org.apache.camel.v1.pipespec.sink.RefBuilder()
+ .withName(broker)
+ .withApiVersion("eventing.knative.dev/" + KnativeSettings.getApiVersion())
+ .withKind("Broker")
+ .withNamespace(KnativeSettings.getNamespace())
+ .build();
+ sink = new SinkBuilder().withRef(sinkRef);
+ }
+
+ @Given("^bind Kamelet ([a-z0-9-]+) to uri ([^\\s]+)$")
+ public void bindKameletToUri(String kameletName, String uri) {
+ setKameletEventSource(kameletName);
+ setEventSinkUri(uri);
+ }
+
+ @Given("^bind Kamelet ([a-z0-9-]+) to Kafka topic ([^\\s]+)$")
+ public void bindKameletToKafka(String kameletName, String topic) {
+ setKameletEventSource(kameletName);
+ setEventSinkKafkaTopic(topic);
+ }
+
+ @Given("^bind Kamelet ([a-z0-9-]+) to Knative channel ([^\\s]+)$")
+ public void bindKameletToKnativeChannel(String kameletName, String channel) {
+ bindKameletToKnativeChannel(kameletName, channel, "InMemoryChannel");
+ }
+
+ @Given("^bind Kamelet ([a-z0-9-]+) to Knative channel ([^\\s]+) of kind ([^\\s]+)$")
+ public void bindKameletToKnativeChannel(String kameletName, String channel, String channelKind) {
+ setKameletEventSource(kameletName);
+ setEventSinkKnativeChannel(channel, channelKind);
+ }
+
+ @Given("^bind Kamelet ([a-z0-9-]+) to Knative broker ([^\\s]+)$")
+ public void bindKameletToKnativeBroker(String kameletName, String broker) {
+ setKameletEventSource(kameletName);
+ setEventSinkKnativeBroker(broker);
+ }
+
+ @Given("^load Pipe ([a-z0-9-]+).yaml$")
+ public void loadPipeFromFile(String fileName) {
+ Resource resource = ResourceUtils.resolve(fileName + ".yaml", context);
+ runner.run(camelk()
+ .client(k8sClient)
+ .createPipe(fileName)
+ .resource(resource));
+
+ if (isAutoRemoveResources()) {
+ runner.then(doFinally()
+ .actions(camelk().client(k8sClient)
+ .deletePipe(fileName)));
+ }
+ }
+
+ @Given("^(?:create|new) Pipe ([a-z0-9-]+)$")
+ public void createNewPipe(String name) {
+ PipeBuilder pipe = new PipeBuilder();
+
+ pipe.withNewMetadata()
+ .withName(name)
+ .endMetadata();
+
+ source.editOrNewProperties().addToAdditionalProperties(sourceProperties);
+ sink.editOrNewProperties().addToAdditionalProperties(sinkProperties);
+
+ pipe.withNewSpec()
+ .withSource(source.build())
+ .withSink(sink.build())
+ .endSpec();
+
+ runner.run(camelk()
+ .client(k8sClient)
+ .createPipe(name)
+ .fromBuilder(pipe));
+
+ initializePipeBuilder();
+
+ if (isAutoRemoveResources()) {
+ runner.then(doFinally()
+ .actions(camelk().client(k8sClient)
+ .deletePipe(name)));
+ }
+ }
+
+ @Given("^delete Pipe ([a-z0-9-]+)$")
+ public void deletePipe(String name) {
+ runner.run(camelk()
+ .client(k8sClient)
+ .deletePipe(name));
+ }
+
+ @Given("^Pipe ([a-z0-9-]+) is available$")
+ @Then("^Pipe ([a-z0-9-]+) should be available$")
+ public void pipeShouldBeAvailable(String name) {
+ runner.run(camelk()
+ .client(k8sClient)
+ .verifyPipe(name)
+ .isAvailable());
+ }
+
+ private void initializePipeBuilder() {
+ source = null;
+ sink = null;
+ sourceProperties = new HashMap<>();
+ sinkProperties = new HashMap<>();
+ }
+
+ private String getKameletApiVersion() {
+ if (context.getVariables().containsKey(VariableNames.KAMELET_API_VERSION.value())) {
+ return context.getVariable(VariableNames.KAMELET_API_VERSION.value());
+ }
+
+ return KameletSettings.getKameletApiVersion();
+ }
+
+ private String getNamespace() {
+ if (context.getVariables().containsKey(VariableNames.KAMELET_NAMESPACE.value())) {
+ return context.getVariable(VariableNames.KAMELET_NAMESPACE.value());
+ }
+
+ return KameletSettings.getNamespace();
+ }
+
+ private boolean isAutoRemoveResources() {
+ if (context.getVariables().containsKey(VariableNames.AUTO_REMOVE_RESOURCES.value())) {
+ return context.getVariable(VariableNames.AUTO_REMOVE_RESOURCES.value(), Boolean.class);
+ }
+
+ return CamelKSettings.isAutoRemoveResources();
+ }
+}
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/VariableNames.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/VariableNames.java
index ae07df55..6f465f29 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/VariableNames.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/VariableNames.java
@@ -26,7 +26,8 @@ public enum VariableNames {
KAMELET_API_VERSION("KAMELET_API_VERSION"),
KAMELET_NAMESPACE("KAMELET_NAMESPACE"),
- OPERATOR_NAMESPACE("CAMELK_OPERATOR_NAMESPACE");
+ OPERATOR_NAMESPACE("CAMELK_OPERATOR_NAMESPACE"),
+ AUTO_REMOVE_RESOURCES("CAMELK_AUTO_REMOVE_RESOURCES");
private final String variableName;
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/CamelKActionBuilder.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/CamelKActionBuilder.java
index 1dfd3e14..75beb9b3 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/CamelKActionBuilder.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/CamelKActionBuilder.java
@@ -16,16 +16,19 @@
package org.citrusframework.yaks.camelk.actions;
-import org.citrusframework.TestActionBuilder;
import io.fabric8.kubernetes.client.KubernetesClient;
+import org.citrusframework.TestActionBuilder;
import org.citrusframework.yaks.camelk.actions.integration.CreateIntegrationAction;
import org.citrusframework.yaks.camelk.actions.integration.DeleteIntegrationAction;
import org.citrusframework.yaks.camelk.actions.integration.VerifyIntegrationAction;
import org.citrusframework.yaks.camelk.actions.kamelet.CreateKameletAction;
+import org.citrusframework.yaks.camelk.actions.kamelet.CreateKameletBindingAction;
import org.citrusframework.yaks.camelk.actions.kamelet.CreatePipeAction;
import org.citrusframework.yaks.camelk.actions.kamelet.DeleteKameletAction;
+import org.citrusframework.yaks.camelk.actions.kamelet.DeleteKameletBindingAction;
import org.citrusframework.yaks.camelk.actions.kamelet.DeletePipeAction;
import org.citrusframework.yaks.camelk.actions.kamelet.VerifyKameletAction;
+import org.citrusframework.yaks.camelk.actions.kamelet.VerifyKameletBindingAction;
import org.citrusframework.yaks.camelk.actions.kamelet.VerifyPipeAction;
import org.springframework.util.Assert;
@@ -127,6 +130,30 @@ public DeletePipeAction.Builder deletePipe(String pipeName) {
return builder;
}
+ /**
+ * Create binding CRD in current namespace.
+ * @param bindingName the name of the binding.
+ */
+ public CreateKameletBindingAction.Builder createKameletBinding(String bindingName) {
+ CreateKameletBindingAction.Builder builder = new CreateKameletBindingAction.Builder()
+ .client(kubernetesClient)
+ .binding(bindingName);
+ this.delegate = builder;
+ return builder;
+ }
+
+ /**
+ * Delete binding CRD from current namespace.
+ * @param bindingName the name of the binding.
+ */
+ public DeleteKameletBindingAction.Builder deleteKameletBinding(String bindingName) {
+ DeleteKameletBindingAction.Builder builder = new DeleteKameletBindingAction.Builder()
+ .client(kubernetesClient)
+ .binding(bindingName);
+ this.delegate = builder;
+ return builder;
+ }
+
/**
* Verify that given integration is running.
* @param integrationName the name of the Camel K integration.
@@ -163,6 +190,18 @@ public VerifyPipeAction.Builder verifyPipe(String pipeName) {
return builder;
}
+ /**
+ * Verify that given binding CRD is available in current namespace.
+ * @param bindingName the name of the binding.
+ */
+ public VerifyKameletBindingAction.Builder verifyKameletBinding(String bindingName) {
+ VerifyKameletBindingAction.Builder builder = new VerifyKameletBindingAction.Builder()
+ .client(kubernetesClient)
+ .isAvailable(bindingName);
+ this.delegate = builder;
+ return builder;
+ }
+
@Override
public CamelKAction build() {
Assert.notNull(delegate, "Missing delegate action to build");
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/CreateIntegrationAction.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/CreateIntegrationAction.java
index 7eec264b..16cb9719 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/CreateIntegrationAction.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/CreateIntegrationAction.java
@@ -17,34 +17,34 @@
package org.citrusframework.yaks.camelk.actions.integration;
import java.io.IOException;
-import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.StandardOpenOption;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
-import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
-import java.util.stream.Collectors;
+import io.fabric8.kubernetes.client.KubernetesClient;
import io.fabric8.kubernetes.client.dsl.Updatable;
+import org.apache.camel.v1.Integration;
+import org.apache.camel.v1.IntegrationBuilder;
+import org.apache.camel.v1.IntegrationSpecBuilder;
+import org.apache.camel.v1.integrationspec.ConfigurationBuilder;
+import org.apache.camel.v1.integrationspec.SourcesBuilder;
+import org.apache.camel.v1.integrationspec.Traits;
import org.citrusframework.context.TestContext;
import org.citrusframework.exceptions.CitrusRuntimeException;
-import org.citrusframework.util.FileUtils;
import org.citrusframework.variable.VariableUtils;
-import io.fabric8.kubernetes.client.KubernetesClient;
import org.citrusframework.yaks.YaksSettings;
import org.citrusframework.yaks.camelk.actions.AbstractCamelKAction;
import org.citrusframework.yaks.camelk.jbang.CamelJBangSettings;
import org.citrusframework.yaks.camelk.jbang.ProcessAndOutput;
-import org.citrusframework.yaks.camelk.model.Integration;
import org.citrusframework.yaks.camelk.model.IntegrationList;
-import org.citrusframework.yaks.camelk.model.IntegrationSpec;
import org.citrusframework.yaks.kubernetes.KubernetesSupport;
import org.citrusframework.yaks.util.ResourceUtils;
import org.slf4j.Logger;
@@ -69,10 +69,12 @@ public class CreateIntegrationAction extends AbstractCamelKAction {
private final List dependencies;
private final List buildProperties;
private final List buildPropertyFiles;
+ private final List envVars;
+ private final List envVarFiles;
private final List properties;
private final List propertyFiles;
private final List traits;
- private final Map openApis;
+ private final List openApis;
private final boolean supportVariables;
/**
@@ -87,6 +89,8 @@ public CreateIntegrationAction(Builder builder) {
this.dependencies = builder.dependencies;
this.buildProperties = builder.buildProperties;
this.buildPropertyFiles = builder.buildPropertyFiles;
+ this.envVars = builder.envVars;
+ this.envVarFiles = builder.envVarFiles;
this.properties = builder.properties;
this.propertyFiles = builder.propertyFiles;
this.traits = builder.traits;
@@ -107,21 +111,31 @@ public void doExecute(TestContext context) {
resolvedSource = source;
}
- final Integration.Builder integrationBuilder = new Integration.Builder()
- .name(name)
- .source(context.replaceDynamicContentInString(fileName), resolvedSource);
+ final IntegrationBuilder integrationBuilder = new IntegrationBuilder()
+ .withNewMetadata()
+ .withName(name)
+ .endMetadata();
+
+ IntegrationSpecBuilder specBuilder = new IntegrationSpecBuilder();
+ specBuilder.addToSources(new SourcesBuilder().withName(context.replaceDynamicContentInString(fileName)).withContent(resolvedSource).build());
List resolvedDependencies = resolveDependencies(resolvedSource, context.resolveDynamicValuesInList(dependencies));
if (!resolvedDependencies.isEmpty()) {
- integrationBuilder.dependencies(resolvedDependencies);
+ specBuilder.addAllToDependencies(resolvedDependencies);
}
- addPropertyConfigurationSpec(integrationBuilder, context);
- addBuildPropertyConfigurationSpec(integrationBuilder, resolvedSource, context);
- addRuntimeConfigurationSpec(integrationBuilder, resolvedSource, context);
- addTraitSpec(integrationBuilder, resolvedSource, context);
- addOpenApiSpec(integrationBuilder, resolvedSource, context);
+ Map> traitConfigMap = new HashMap<>();
+ addPropertyConfigurationSpec(specBuilder, context);
+ addRuntimeConfigurationSpec(specBuilder, resolvedSource, context);
+ addBuildPropertyConfigurationSpec(traitConfigMap, resolvedSource, context);
+ addEnvVarConfigurationSpec(traitConfigMap, resolvedSource, context);
+ addOpenApiSpec(traitConfigMap, resolvedSource, context);
+ addTraitSpec(traitConfigMap, resolvedSource, context);
+
+ specBuilder.withTraits(KubernetesSupport.json().convertValue(traitConfigMap, Traits.class));
- final Integration integration = integrationBuilder.build();
+ final Integration integration = integrationBuilder
+ .withSpec(specBuilder.build())
+ .build();
if (YaksSettings.isLocal(clusterType(context))) {
createLocalIntegration(integration, integration.getMetadata().getName(), context);
} else {
@@ -139,7 +153,7 @@ public void doExecute(TestContext context) {
*/
private static void createIntegration(KubernetesClient k8sClient, String namespace, Integration integration) {
if (LOG.isDebugEnabled()) {
- LOG.debug(KubernetesSupport.yaml().dumpAsMap(integration));
+ LOG.debug(KubernetesSupport.yaml(new IntegrationValuePropertyMapper()).dumpAsMap(integration));
}
k8sClient.resources(Integration.class, IntegrationList.class)
@@ -156,7 +170,7 @@ private static void createIntegration(KubernetesClient k8sClient, String namespa
*/
private static void createLocalIntegration(Integration integration, String name, TestContext context) {
try {
- String integrationYaml = KubernetesSupport.yaml().dumpAsMap(integration);
+ String integrationYaml = KubernetesSupport.yaml(new IntegrationValuePropertyMapper()).dumpAsMap(integration);
if (LOG.isDebugEnabled()) {
LOG.debug(integrationYaml);
@@ -168,7 +182,7 @@ private static void createLocalIntegration(Integration integration, String name,
Path workDir = CamelJBangSettings.getWorkDir();
Files.createDirectories(workDir);
Path file = workDir.resolve(String.format("i-%s.yaml", name));
- Files.write(file, integrationYaml.getBytes(StandardCharsets.UTF_8),
+ Files.writeString(file, integrationYaml,
StandardOpenOption.WRITE,
StandardOpenOption.CREATE,
StandardOpenOption.TRUNCATE_EXISTING);
@@ -199,39 +213,35 @@ private static void createLocalIntegration(Integration integration, String name,
private static String[] camelRunArgs(Integration integration) {
List args = new ArrayList<>();
- if (integration.getSpec().getResources() != null) {
- List openApiResources = integration.getSpec().getResources()
- .stream()
- .filter(r -> "openapi".equals(r.getType()))
- .collect(Collectors.toList());
-
- for (IntegrationSpec.Resource resource : openApiResources) {
+ if (integration.getSpec().getTraits().getOpenapi() != null) {
+ for (String openApi : integration.getSpec().getTraits().getOpenapi().getConfigmaps()) {
args.add("--open-api");
- args.add(resource.getName());
+ if (openApi.startsWith("configmap:")) {
+ args.add(openApi.substring("configmap:".length()));
+ } else {
+ args.add(openApi);
+ }
}
}
return args.toArray(String[]::new);
}
- private void addOpenApiSpec(Integration.Builder integrationBuilder, String source, TestContext context) {
+ private void addOpenApiSpec(Map> traitConfigMap, String source, TestContext context) {
+ String traitName = "openapi.configmaps";
Pattern pattern = getModelinePattern("open-api");
Matcher depMatcher = pattern.matcher(source);
while (depMatcher.find()) {
String openApiSpecFile = depMatcher.group(1);
- try {
- integrationBuilder.openApi(openApiSpecFile, FileUtils.readToString(FileUtils.getFileResource(openApiSpecFile, context)));
- } catch (IOException e) {
- throw new CitrusRuntimeException(String.format("Failed to load OpenAPI spec from file '%s'", openApiSpecFile), e);
- }
+ addTraitSpec("%s=%s".formatted(traitName, context.replaceDynamicContentInString(openApiSpecFile)), traitConfigMap);
}
- openApis.forEach((k, v) -> integrationBuilder.openApi(k, context.replaceDynamicContentInString(v)));
+ for (String openApi : openApis) {
+ addTraitSpec("%s=%s".formatted(traitName, context.replaceDynamicContentInString(openApi)), traitConfigMap);
+ }
}
- private void addTraitSpec(Integration.Builder integrationBuilder, String source, TestContext context) {
- final Map traitConfigMap = new HashMap<>();
-
+ private void addTraitSpec(Map> traitConfigMap, String source, TestContext context) {
if (traits != null && !traits.isEmpty()) {
for (String t : context.resolveDynamicValuesInList(traits)) {
addTraitSpec(t, traitConfigMap);
@@ -243,36 +253,34 @@ private void addTraitSpec(Integration.Builder integrationBuilder, String source,
while (depMatcher.find()) {
addTraitSpec(depMatcher.group(1), traitConfigMap);
}
-
- if (!traitConfigMap.isEmpty()) {
- integrationBuilder.traits(traitConfigMap);
- }
}
- private void addTraitSpec(String traitExpression, Map configMap) {
+ private void addTraitSpec(String traitExpression, Map> traitConfigMap) {
//traitName.key=value
final String[] trait = traitExpression.split("\\.",2);
final String[] traitConfig = trait[1].split("=", 2);
final String traitKey = traitConfig[0];
final Object traitValue = resolveTraitValue(traitKey, traitConfig[1].trim());
- if (configMap.containsKey(trait[0])) {
- IntegrationSpec.TraitConfig config = configMap.get(trait[0]);
+ if (traitConfigMap.containsKey(trait[0])) {
+ Map config = traitConfigMap.get(trait[0]);
- if (config.getConfiguration().containsKey(traitKey)) {
- Object existingValue = config.getConfiguration().get(traitKey);
+ if (config.containsKey(traitKey)) {
+ Object existingValue = config.get(traitKey);
if (existingValue instanceof List) {
List values = (List) existingValue;
values.add(traitValue.toString());
} else {
- config.add(traitKey, Arrays.asList(existingValue.toString(), traitValue));
+ config.put(traitKey, Arrays.asList(existingValue.toString(), traitValue));
}
} else {
- config.add(traitKey, initializeTraitValue(traitValue));
+ config.put(traitKey, initializeTraitValue(traitValue));
}
} else {
- configMap.put(trait[0], new IntegrationSpec.TraitConfig(traitKey, initializeTraitValue(traitValue)));
+ Map config = new HashMap<>();
+ config.put(traitKey, initializeTraitValue(traitValue));
+ traitConfigMap.put(trait[0], config);
}
}
@@ -306,18 +314,21 @@ private Object resolveTraitValue(String traitKey, String value) {
return Boolean.valueOf(value);
}
- return value;
+ try {
+ return Integer.parseInt(value);
+ } catch (NumberFormatException e) {
+ return value;
+ }
}
- private void addPropertyConfigurationSpec(Integration.Builder integrationBuilder, TestContext context) {
- final List configurationList = new ArrayList<>();
+ private void addPropertyConfigurationSpec(IntegrationSpecBuilder specBuilder, TestContext context) {
if (properties != null && !properties.isEmpty()) {
for (String p : context.resolveDynamicValuesInList(properties)){
//key=value
if (isValidPropertyFormat(p)) {
final String[] property = p.split("=",2);
- configurationList.add(
- new IntegrationSpec.Configuration("property", createPropertySpec(property[0], property[1], context)));
+ specBuilder.addToConfiguration(
+ new ConfigurationBuilder().withType("property").withValue(createPropertySpec(property[0], property[1], context)).build());
} else {
throw new IllegalArgumentException("Property " + p + " does not match format key=value");
}
@@ -329,22 +340,17 @@ private void addPropertyConfigurationSpec(Integration.Builder integrationBuilder
try {
Properties props = new Properties();
props.load(ResourceUtils.resolve(pf, context).getInputStream());
- props.forEach((key, value) -> configurationList.add(
- new IntegrationSpec.Configuration("property", createPropertySpec(key.toString(), value.toString(), context))));
+ props.forEach((key, value) -> specBuilder.addToConfiguration(
+ new ConfigurationBuilder().withType("property").withValue(createPropertySpec(key.toString(), value.toString(), context)).build()));
} catch (IOException e) {
throw new CitrusRuntimeException("Failed to load property file", e);
}
}
}
-
- if (!configurationList.isEmpty()) {
- integrationBuilder.configuration(configurationList);
- }
}
- private void addBuildPropertyConfigurationSpec(Integration.Builder integrationBuilder, String source, TestContext context) {
+ private void addBuildPropertyConfigurationSpec(Map> traitConfigMap, String source, TestContext context) {
final String traitName = "builder.properties";
- final Map traitConfigMap = new HashMap<>();
if (buildProperties != null && !buildProperties.isEmpty()) {
for (String p : context.resolveDynamicValuesInList(buildProperties)){
@@ -377,29 +383,55 @@ private void addBuildPropertyConfigurationSpec(Integration.Builder integrationBu
while (depMatcher.find()) {
addTraitSpec(String.format("%s=%s", traitName, depMatcher.group(1)), traitConfigMap);
}
+ }
- if (!traitConfigMap.isEmpty()) {
- integrationBuilder.traits(traitConfigMap);
+ private void addEnvVarConfigurationSpec(Map> traitConfigMap, String source, TestContext context) {
+ final String traitName = "environment.vars";
+
+ if (envVars != null && !envVars.isEmpty()) {
+ for (String v : context.resolveDynamicValuesInList(envVars)){
+ //key=value
+ if (isValidPropertyFormat(v)) {
+ final String[] property = v.split("=", 2);
+ addTraitSpec(String.format("%s=%s", traitName, createPropertySpec(property[0], property[1], context)), traitConfigMap);
+ } else {
+ throw new IllegalArgumentException("EnvVar " + v + " does not match format key=value");
+ }
+ }
}
- }
- private void addRuntimeConfigurationSpec(Integration.Builder integrationBuilder, String source, TestContext context) {
- final List configurationList = new ArrayList<>();
+ if (envVarFiles != null && !envVarFiles.isEmpty()) {
+ for (String vf : envVarFiles){
+ try {
+ Properties props = new Properties();
+ props.load(ResourceUtils.resolve(vf, context).getInputStream());
+ props.forEach((key, value) -> addTraitSpec(String.format("%s=%s",
+ traitName,
+ createPropertySpec(key.toString(), value.toString(), context)), traitConfigMap));
+ } catch (IOException e) {
+ throw new CitrusRuntimeException("Failed to load env var file", e);
+ }
+ }
+ }
+
+ Pattern pattern = getModelinePattern("env");
+ Matcher depMatcher = pattern.matcher(source);
+ while (depMatcher.find()) {
+ addTraitSpec(String.format("%s=%s", traitName, depMatcher.group(1)), traitConfigMap);
+ }
+ }
+ private void addRuntimeConfigurationSpec(IntegrationSpecBuilder specBuilder, String source, TestContext context) {
Pattern pattern = getModelinePattern("config");
Matcher depMatcher = pattern.matcher(source);
while (depMatcher.find()) {
String[] config = depMatcher.group(1).split(":", 2);
if (config.length == 2) {
- configurationList.add(new IntegrationSpec.Configuration(config[0], config[1]));
+ specBuilder.addToConfiguration(new ConfigurationBuilder().withType(config[0]).withValue(context.replaceDynamicContentInString(config[1])).build());
} else {
- configurationList.add(new IntegrationSpec.Configuration("property", depMatcher.group(1)));
+ specBuilder.addToConfiguration(new ConfigurationBuilder().withType("property").withValue(context.replaceDynamicContentInString(depMatcher.group(1))).build());
}
}
-
- if (!configurationList.isEmpty()) {
- integrationBuilder.configuration(configurationList);
- }
}
private String createPropertySpec(String key, String value, TestContext context) {
@@ -472,10 +504,12 @@ public static final class Builder extends AbstractCamelKAction.Builder dependencies = new ArrayList<>();
private final List buildProperties = new ArrayList<>();
private final List buildPropertyFiles = new ArrayList<>();
+ private final List envVars = new ArrayList<>();
+ private final List envVarFiles = new ArrayList<>();
private final List properties = new ArrayList<>();
private final List propertyFiles = new ArrayList<>();
private final List traits = new ArrayList<>();
- private final Map openApis = new LinkedHashMap<>();
+ private final List openApis = new ArrayList<>();
private boolean supportVariables = true;
public Builder integration(String integrationName) {
@@ -509,8 +543,8 @@ public Builder source(String fileName, String source) {
return this;
}
- public Builder openApi(String fileName, String content) {
- this.openApis.put(fileName, content);
+ public Builder openApi(String configMap) {
+ this.openApis.add(configMap);
return this;
}
@@ -561,7 +595,7 @@ public Builder buildPropertyFiles(List propertyFiles) {
}
public Builder buildProperties(String properties) {
- if (properties != null && properties.length() > 0) {
+ if (properties != null && !properties.isEmpty()) {
buildProperties(Arrays.asList(properties.split(",")));
}
@@ -578,6 +612,34 @@ public Builder buildProperties(Map properties) {
return this;
}
+ public Builder envVarFile(String envVarFile) {
+ this.envVarFiles.add(envVarFile);
+ return this;
+ }
+
+ public Builder envVarFiles(List envVarFiles) {
+ this.envVarFiles.addAll(envVarFiles);
+ return this;
+ }
+
+ public Builder envVars(String vars) {
+ if (vars != null && !vars.isEmpty()) {
+ envVars(Arrays.asList(vars.split(",")));
+ }
+
+ return this;
+ }
+
+ public Builder envVars(List vars) {
+ this.envVars.addAll(vars);
+ return this;
+ }
+
+ public Builder envVars(Map vars) {
+ vars.forEach(this::envVar);
+ return this;
+ }
+
public Builder dependencies(List dependencies) {
this.dependencies.addAll(dependencies);
return this;
@@ -598,6 +660,11 @@ public Builder buildProperty(String name, String value) {
return this;
}
+ public Builder envVar(String name, String value) {
+ this.envVars.add(name + "=" + value);
+ return this;
+ }
+
public Builder traits(String traits) {
if (traits != null && traits.length() > 0) {
traits(Arrays.asList(traits.split(",")));
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/DeleteIntegrationAction.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/DeleteIntegrationAction.java
index 9b72b360..65ed6775 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/DeleteIntegrationAction.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/DeleteIntegrationAction.java
@@ -16,12 +16,12 @@
package org.citrusframework.yaks.camelk.actions.integration;
+import io.fabric8.kubernetes.client.KubernetesClient;
+import org.apache.camel.v1.Integration;
import org.citrusframework.context.TestContext;
import org.citrusframework.exceptions.CitrusRuntimeException;
-import io.fabric8.kubernetes.client.KubernetesClient;
import org.citrusframework.yaks.YaksSettings;
import org.citrusframework.yaks.camelk.actions.AbstractCamelKAction;
-import org.citrusframework.yaks.camelk.model.Integration;
import org.citrusframework.yaks.camelk.model.IntegrationList;
import static org.citrusframework.yaks.camelk.jbang.CamelJBang.camel;
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/IntegrationValuePropertyMapper.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/IntegrationValuePropertyMapper.java
new file mode 100644
index 00000000..38ee8ca0
--- /dev/null
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/IntegrationValuePropertyMapper.java
@@ -0,0 +1,44 @@
+/*
+ * Copyright the original author or 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 org.citrusframework.yaks.camelk.actions.integration;
+
+import java.util.Collections;
+
+import org.citrusframework.yaks.kubernetes.KubernetesSupport;
+import org.yaml.snakeyaml.introspector.Property;
+
+/**
+ * Helper to properly handle additional properties on Integration additional properties.
+ */
+class IntegrationValuePropertyMapper implements KubernetesSupport.PropertyValueMapper {
+ @Override
+ public Object map(Property property, Object propertyValue) {
+ if (propertyValue == null) {
+ return null;
+ }
+
+ if (property.getName().equals("additionalProperties")) {
+ return Collections.emptyMap();
+ }
+
+ if (propertyValue instanceof org.apache.camel.v1.integrationspec.Flows flowsProps) {
+ return flowsProps.getAdditionalProperties();
+ }
+
+ return propertyValue;
+ }
+}
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/VerifyIntegrationAction.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/VerifyIntegrationAction.java
index bd9f6bef..92118d3e 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/VerifyIntegrationAction.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/integration/VerifyIntegrationAction.java
@@ -18,16 +18,21 @@
import java.util.Map;
-import org.citrusframework.context.TestContext;
-import org.citrusframework.exceptions.ActionTimeoutException;
-import org.citrusframework.exceptions.CitrusRuntimeException;
import io.fabric8.kubernetes.api.model.Pod;
+import io.fabric8.kubernetes.api.model.PodCondition;
import io.fabric8.kubernetes.api.model.PodList;
import io.fabric8.kubernetes.client.dsl.PodResource;
+import org.apache.camel.v1.Integration;
+import org.apache.camel.v1.IntegrationStatus;
+import org.apache.camel.v1.integrationstatus.Conditions;
+import org.citrusframework.context.TestContext;
+import org.citrusframework.exceptions.ActionTimeoutException;
+import org.citrusframework.exceptions.CitrusRuntimeException;
import org.citrusframework.yaks.YaksSettings;
import org.citrusframework.yaks.camelk.CamelKSettings;
import org.citrusframework.yaks.camelk.actions.AbstractCamelKAction;
import org.citrusframework.yaks.camelk.jbang.ProcessAndOutput;
+import org.citrusframework.yaks.camelk.model.IntegrationList;
import org.citrusframework.yaks.kubernetes.KubernetesSupport;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
@@ -54,6 +59,8 @@ public class VerifyIntegrationAction extends AbstractCamelKAction {
private final String phase;
private final boolean printLogs;
+ private final boolean stopOnErrorStatus;
+
/**
* Constructor using given builder.
* @param builder
@@ -66,6 +73,7 @@ public VerifyIntegrationAction(Builder builder) {
this.maxAttempts = builder.maxAttempts;
this.delayBetweenAttempts = builder.delayBetweenAttempts;
this.printLogs = builder.printLogs;
+ this.stopOnErrorStatus = builder.stopOnErrorStatus;
}
@Override
@@ -139,6 +147,11 @@ private Long verifyLocalIntegrationStatus(String integration, String phase, Test
if ((phase.equals("Stopped") && properties.isEmpty()) || (!properties.isEmpty() && properties.get("STATUS").equals(phase))) {
LOG.info(String.format("Verified integration '%s' state '%s' - All values OK!", integration, phase));
return pid;
+ } else if (phase.equals("Error")) {
+ LOG.info(String.format("Integration '%s' is in state 'Error'", integration));
+ if (stopOnErrorStatus) {
+ throw new CitrusRuntimeException(String.format("Failed to verify integration '%s' - is in state 'Error'", integration));
+ }
}
}
@@ -244,10 +257,23 @@ private Pod verifyIntegrationPod(String name, String phase, String namespace) {
INTEGRATION_STATUS_LOG.info(String.format("Waiting for integration '%s' to be in state '%s'", name, phase));
for (int i = 0; i < maxAttempts; i++) {
- Pod pod = getIntegrationPod(name, phase, namespace);
- if (pod != null) {
- LOG.info(String.format("Verified integration pod '%s' state '%s' - All values OK!", name, phase));
- return pod;
+ Integration integration = getIntegration(name, namespace);
+ if (integration != null && integration.getStatus() != null) {
+ if ("Error".equals(integration.getStatus().getPhase())) {
+ String readyConditionError = getReadyConditionErrorDetails(integration.getStatus());
+ if (stopOnErrorStatus) {
+ INTEGRATION_STATUS_LOG.info(String.format("Integration '%s' is in state 'Error' - %s", name, readyConditionError));
+ throw new CitrusRuntimeException(String.format("Failed to verify integration '%s' - is in state 'Error' - %s", name, readyConditionError));
+ } else {
+ INTEGRATION_STATUS_LOG.info(String.format("Integration '%s' is in state 'Error' - %s. Will keep checking ...", name, readyConditionError));
+ }
+ } else {
+ Pod pod = getIntegrationPod(name, phase, namespace);
+ if (pod != null) {
+ LOG.info(String.format("Verified integration pod '%s' state '%s' - All values OK!", name, phase));
+ return pod;
+ }
+ }
}
LOG.info(String.format("Waiting for integration '%s' to be in state '%s'- retry in %s ms", name, phase, delayBetweenAttempts));
@@ -263,6 +289,13 @@ private Pod verifyIntegrationPod(String name, String phase, String namespace) {
"is not in state '%s' after %d attempts", name, phase, maxAttempts)));
}
+ private Integration getIntegration(String name, String namespace) {
+ return getKubernetesClient().resources(Integration.class, IntegrationList.class)
+ .inNamespace(namespace)
+ .withName(name)
+ .get();
+ }
+
/**
* Retrieve pod given state.
* @param integration
@@ -285,7 +318,18 @@ private Pod getIntegrationPod(final String integration, final String phase, fina
boolean verified = KubernetesSupport.verifyPodStatus(pod, phase);
if (!verified) {
- INTEGRATION_STATUS_LOG.info(String.format("Integration '%s' not yet in state '%s'. Will keep checking ...", integration, phase));
+ if (pod != null && pod.getStatus() != null &&
+ "Error".equals(pod.getStatus().getPhase())) {
+ String readyConditionError = getReadyConditionErrorDetails(pod);
+ if (stopOnErrorStatus) {
+ INTEGRATION_STATUS_LOG.info(String.format("Integration '%s' is in state 'Error' - %s", integration, readyConditionError));
+ throw new CitrusRuntimeException(String.format("Failed to verify integration '%s' - is in state 'Error' - %s", integration, readyConditionError));
+ } else {
+ INTEGRATION_STATUS_LOG.info(String.format("Integration '%s' is in state 'Error' - %s. Will keep checking ...", integration, readyConditionError));
+ }
+ } else {
+ INTEGRATION_STATUS_LOG.info(String.format("Integration '%s' not yet in state '%s'. Will keep checking ...", integration, phase));
+ }
}
return verified;
@@ -294,6 +338,26 @@ private Pod getIntegrationPod(final String integration, final String phase, fina
.orElse(null);
}
+ private String getReadyConditionErrorDetails(IntegrationStatus status) {
+ for (Conditions condition : status.getConditions()) {
+ if ("Ready".equals(condition.getType()) && "False".equalsIgnoreCase(condition.getStatus())) {
+ return "%s: %s".formatted(condition.getReason(), condition.getMessage());
+ }
+ }
+
+ return "Unknown error";
+ }
+
+ private String getReadyConditionErrorDetails(Pod pod) {
+ for (PodCondition condition : pod.getStatus().getConditions()) {
+ if ("Ready".equals(condition.getType()) && "False".equalsIgnoreCase(condition.getStatus())) {
+ return "%s: %s".formatted(condition.getReason(), condition.getMessage());
+ }
+ }
+
+ return "Unknown error";
+ }
+
/**
* Action builder.
*/
@@ -308,6 +372,8 @@ public static final class Builder extends AbstractCamelKAction.Builder dependencies;
- private final Map dataTypes;
+ private final Map dataTypes;
private final Resource resource;
private final boolean supportVariables;
@@ -89,7 +100,13 @@ private void createKamelet(TestContext context) {
resolvedSource = FileUtils.readToString(resource);
}
- kamelet = KubernetesSupport.yaml().loadAs(resolvedSource, Kamelet.class);
+ if (IsJsonPredicate.getInstance().test(resolvedSource)) {
+ kamelet = KubernetesSupport.json().readValue(resolvedSource, Kamelet.class);
+ } else {
+ // need to make a detour over Json to support additional properties set on Pipe
+ Map raw = KubernetesSupport.yaml().load(resolvedSource);
+ kamelet = KubernetesSupport.json().convertValue(raw, Kamelet.class);
+ }
} catch (IOException e) {
throw new CitrusRuntimeException(String.format("Failed to load Kamelet from resource %s", name + ".kamelet.yaml"), e);
}
@@ -105,27 +122,36 @@ private void createKamelet(TestContext context) {
definition.setProperties(context.resolveDynamicValuesInMap(definition.getProperties()));
definition.setRequired(context.resolveDynamicValuesInList(definition.getRequired()));
- final Kamelet.Builder builder = new Kamelet.Builder()
- .name(context.replaceDynamicContentInString(name))
- .definition(definition);
+ final KameletBuilder builder = new KameletBuilder()
+ .withNewMetadata()
+ .withName(context.replaceDynamicContentInString(name))
+ .endMetadata();
+
+ KameletSpecBuilder specBuilder = new KameletSpecBuilder()
+ .withDefinition(definition);
if (template != null) {
- builder.template(context.replaceDynamicContentInString(template));
+ specBuilder.withTemplate(new TemplateBuilder()
+ .withAdditionalProperties(KubernetesSupport.yaml(new KameletValuePropertyMapper()).load(context.replaceDynamicContentInString(template)))
+ .build());
}
if (source != null) {
- builder.source(source.getName(), context.replaceDynamicContentInString(source.getContent()));
+ specBuilder.withSources(new SourcesBuilder()
+ .withName(source.getName())
+ .withContent(context.replaceDynamicContentInString(source.getContent()))
+ .build());
}
if (dependencies != null && !dependencies.isEmpty()) {
- builder.dependencies(context.resolveDynamicValuesInList(dependencies));
+ specBuilder.withDependencies(context.resolveDynamicValuesInList(dependencies));
}
if (dataTypes != null && !dataTypes.isEmpty()) {
- builder.dataTypes(context.resolveDynamicValuesInMap(dataTypes));
+ specBuilder.withDataTypes(dataTypes);
}
- kamelet = builder.build();
+ kamelet = builder.withSpec(specBuilder.build()).build();
}
if (!kamelet.getMetadata().getLabels().containsKey(KameletSettings.KAMELET_TYPE_LABEL)) {
@@ -150,12 +176,7 @@ private void createKamelet(TestContext context) {
}
if (getApiVersion(context).equals(CamelKSettings.V1ALPHA1)) {
- KameletV1Alpha1 kameletV1Alpha1;
- if (kamelet instanceof KameletV1Alpha1) {
- kameletV1Alpha1 = (KameletV1Alpha1) kamelet;
- } else {
- kameletV1Alpha1 = new KameletV1Alpha1.Builder().from(kamelet).build();
- }
+ KameletV1Alpha1 kameletV1Alpha1 = KameletV1Alpha1.from(kamelet);
getKubernetesClient().resources(KameletV1Alpha1.class, KameletV1Alpha1List.class)
.inNamespace(kameletNamespace(context))
@@ -178,10 +199,10 @@ public static final class Builder extends AbstractKameletAction.Builder dependencies = new ArrayList<>();
- private KameletSpec.Definition definition = new KameletSpec.Definition();
- private final Map dataTypes = new HashMap<>();
+ private Definition definition = new Definition();
+ private final Map dataTypes = new HashMap<>();
private Resource resource;
@@ -204,12 +225,18 @@ public Builder title(String title) {
}
public Builder source(String name, String language, String content) {
- this.source = new KameletSpec.Source(name + "." + language, content);
+ this.source = new SourcesBuilder()
+ .withName(name + "." + language)
+ .withContent(content)
+ .build();
return this;
}
public Builder source(String name, String content) {
- this.source = new KameletSpec.Source(name, content);
+ this.source = new SourcesBuilder()
+ .withName(name)
+ .withContent(content)
+ .build();
return this;
}
@@ -239,12 +266,12 @@ public Builder dependency(String dependency) {
return this;
}
- public Builder definition(KameletSpec.Definition definition) {
+ public Builder definition(Definition definition) {
this.definition = definition;
return this;
}
- public Builder addProperty(String name, KameletSpec.Definition.PropertyConfig propertyConfig) {
+ public Builder addProperty(String name, Properties propertyConfig) {
this.definition.getProperties().put(name, propertyConfig);
return this;
}
@@ -263,30 +290,36 @@ public Builder errorType(String scheme, String format) {
public Builder addDataType(String slot, String scheme, String format) {
if (dataTypes.containsKey(slot)) {
- this.dataTypes.get(slot).getTypes().put(format, new KameletSpec.DataTypeSpec(scheme, format));
+ this.dataTypes.get(slot).getTypes().put(format, new TypesBuilder().withScheme(scheme).withFormat(format).build());
} else {
- this.dataTypes.put(slot, new KameletSpec.DataTypesSpec(format, new KameletSpec.DataTypeSpec(scheme, format)));
+ Map dataTypes = new HashMap<>();
+ dataTypes.put(format, new TypesBuilder().withScheme(scheme).withFormat(format).build());
+ this.dataTypes.put(slot, new DataTypesBuilder().withTypes(dataTypes).build());
}
return this;
}
- public Builder fromBuilder(Kamelet.Builder builder) {
+ public Builder fromBuilder(KameletBuilder builder) {
Kamelet kamelet = builder.build();
name = kamelet.getMetadata().getName();
definition = kamelet.getSpec().getDefinition();
- dependencies.addAll(kamelet.getSpec().getDependencies());
- dataTypes.putAll(kamelet.getSpec().getDataTypes());
+
+ if (kamelet.getSpec().getDependencies() != null) {
+ dependencies.addAll(kamelet.getSpec().getDependencies());
+ }
+
+ if (kamelet.getSpec().getDataTypes() != null) {
+ dataTypes.putAll(kamelet.getSpec().getDataTypes());
+ }
if (kamelet.getSpec().getSources() != null && !kamelet.getSpec().getSources().isEmpty()) {
source = kamelet.getSpec().getSources().get(0);
}
if (kamelet.getSpec().getTemplate() != null) {
- template = KubernetesSupport.yaml().dumpAsMap(kamelet.getSpec().getTemplate());
- } else if (kamelet.getSpec().getFlow() != null) {
- template = KubernetesSupport.yaml().dumpAsMap(kamelet.getSpec().getFlow());
+ template = KubernetesSupport.yaml(new KameletValuePropertyMapper()).dumpAsMap(kamelet.getSpec().getTemplate());
}
return this;
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/CreateKameletBindingAction.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/CreateKameletBindingAction.java
new file mode 100644
index 00000000..450c7e9b
--- /dev/null
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/CreateKameletBindingAction.java
@@ -0,0 +1,285 @@
+/*
+ * Copyright the original author or 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 org.citrusframework.yaks.camelk.actions.kamelet;
+
+import java.io.IOException;
+import java.nio.file.Files;
+import java.nio.file.Path;
+import java.nio.file.StandardOpenOption;
+import java.util.Map;
+
+import io.fabric8.kubernetes.client.KubernetesClient;
+import io.fabric8.kubernetes.client.dsl.Updatable;
+import org.apache.camel.v1alpha1.KameletBinding;
+import org.apache.camel.v1alpha1.KameletBindingBuilder;
+import org.apache.camel.v1alpha1.KameletBindingSpecBuilder;
+import org.apache.camel.v1alpha1.kameletbindingspec.Integration;
+import org.apache.camel.v1alpha1.kameletbindingspec.Sink;
+import org.apache.camel.v1alpha1.kameletbindingspec.SinkBuilder;
+import org.apache.camel.v1alpha1.kameletbindingspec.Source;
+import org.apache.camel.v1alpha1.kameletbindingspec.SourceBuilder;
+import org.apache.camel.v1alpha1.kameletbindingspec.source.Ref;
+import org.citrusframework.context.TestContext;
+import org.citrusframework.exceptions.CitrusRuntimeException;
+import org.citrusframework.spi.Resource;
+import org.citrusframework.util.FileUtils;
+import org.citrusframework.util.IsJsonPredicate;
+import org.citrusframework.yaks.YaksSettings;
+import org.citrusframework.yaks.camelk.CamelKSettings;
+import org.citrusframework.yaks.camelk.jbang.CamelJBangSettings;
+import org.citrusframework.yaks.camelk.jbang.ProcessAndOutput;
+import org.citrusframework.yaks.camelk.model.v1alpha1.KameletBindingList;
+import org.citrusframework.yaks.kubernetes.KubernetesSupport;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import static org.citrusframework.yaks.camelk.jbang.CamelJBang.camel;
+
+/**
+ * Test action creates new Camel K binding with given name and source code. Uses given Kubernetes client to
+ * create a custom resource of type binding.
+ *
+ * @author Christoph Deppisch
+ */
+public class CreateKameletBindingAction extends AbstractKameletAction {
+
+ /** Logger */
+ private static final Logger LOG = LoggerFactory.getLogger(CreateKameletBindingAction.class);
+
+ private final String bindingName;
+ private final Integration integration;
+ private final Source source;
+ private final Sink sink;
+ private final Resource resource;
+
+ /**
+ * Constructor using given builder.
+ * @param builder
+ */
+ public CreateKameletBindingAction(Builder builder) {
+ super("create-binding", builder);
+ this.bindingName = builder.bindingName;
+ this.integration = builder.integration;
+ this.source = builder.source;
+ this.sink = builder.sink;
+ this.resource = builder.resource;
+ }
+
+ @Override
+ public void doExecute(TestContext context) {
+ final KameletBinding binding;
+
+ String bindingName = context.replaceDynamicContentInString(this.bindingName);
+ LOG.info(String.format("Creating Camel K binding '%s'", bindingName));
+
+ if (resource != null) {
+ try {
+ String yamlOrJson = context.replaceDynamicContentInString(FileUtils.readToString(resource));
+ if (IsJsonPredicate.getInstance().test(yamlOrJson)) {
+ binding = KubernetesSupport.json().readValue(yamlOrJson, KameletBinding.class);
+ } else {
+ // need to make a detour over Json to support additional properties set on Pipe
+ Map raw = KubernetesSupport.yaml().load(yamlOrJson);
+ binding = KubernetesSupport.json().convertValue(raw, KameletBinding.class);
+ }
+ } catch (IOException e) {
+ throw new CitrusRuntimeException(String.format("Failed to load binding from resource %s", bindingName + ".yaml"), e);
+ }
+ } else {
+ final KameletBindingBuilder builder = new KameletBindingBuilder()
+ .withNewMetadata()
+ .withName(bindingName)
+ .endMetadata();
+
+ KameletBindingSpecBuilder specBuilder = new KameletBindingSpecBuilder();
+ if (integration != null) {
+ specBuilder.withIntegration(integration);
+ }
+
+ if (source != null) {
+ if (source.getProperties() != null && source.getProperties().getAdditionalProperties() != null) {
+ context.resolveDynamicValuesInMap(source.getProperties().getAdditionalProperties());
+ }
+ specBuilder.withSource(source);
+ }
+
+ if (sink != null) {
+ if (sink.getUri() != null) {
+ sink.setUri(context.replaceDynamicContentInString(sink.getUri()));
+ }
+
+ if (sink.getProperties() != null && sink.getProperties().getAdditionalProperties() != null) {
+ context.resolveDynamicValuesInMap(sink.getProperties().getAdditionalProperties());
+ }
+ specBuilder.withSink(sink);
+ }
+
+ binding = builder.withSpec(specBuilder.build()).build();
+ }
+
+ if (YaksSettings.isLocal(clusterType(context))) {
+ createLocal(KubernetesSupport.yaml(new KameletBindingValuePropertyMapper()).dumpAsMap(binding), bindingName, context);
+ } else {
+ createKameletBinding(getKubernetesClient(), namespace(context), binding, context);
+ }
+
+ LOG.info(String.format("Successfully created binding '%s'", binding.getMetadata().getName()));
+ }
+
+ /**
+ * Creates the Kamelet binding as a custom resource in given namespace.
+ * @param k8sClient
+ * @param namespace
+ * @param binding
+ * @param context
+ */
+ private void createKameletBinding(KubernetesClient k8sClient, String namespace, KameletBinding binding, TestContext context) {
+ if (LOG.isDebugEnabled()) {
+ LOG.debug(KubernetesSupport.yaml(new KameletBindingValuePropertyMapper()).dumpAsMap(binding));
+ }
+
+ k8sClient.resources(KameletBinding.class, KameletBindingList.class)
+ .inNamespace(namespace)
+ .resource(binding)
+ .createOr(Updatable::update);
+ }
+
+ /**
+ * Creates the binding with local JBang runtime.
+ * @param yaml
+ * @param name
+ * @param context
+ */
+ private void createLocal(String yaml, String name, TestContext context) {
+ try {
+ if (LOG.isDebugEnabled()) {
+ LOG.debug(yaml);
+ }
+
+ Path workDir = CamelJBangSettings.getWorkDir();
+ Files.createDirectories(workDir);
+ Path file = workDir.resolve(String.format("i-%s.yaml", name));
+ Files.writeString(file, yaml,
+ StandardOpenOption.WRITE,
+ StandardOpenOption.CREATE,
+ StandardOpenOption.TRUNCATE_EXISTING);
+ ProcessAndOutput pao = camel().run(name, file);
+
+ if (!pao.getProcess().isAlive()) {
+ if (LOG.isDebugEnabled()) {
+ LOG.debug(pao.getOutput());
+ }
+
+ throw new CitrusRuntimeException(String.format("Failed to create binding - exit code %s", pao.getProcess().exitValue()));
+ }
+
+ Long pid = pao.getCamelProcessId();
+ context.setVariable(name + ":pid", pid);
+ context.setVariable(name + ":process:" + pid, pao);
+ } catch (IOException e) {
+ throw new CitrusRuntimeException("Failed to create binding file", e);
+ }
+ }
+
+ /**
+ * Action builder.
+ */
+ public static final class Builder extends AbstractKameletAction.Builder {
+
+ private String bindingName;
+ private Integration integration;
+ private Source source;
+ private Sink sink;
+ private Resource resource;
+
+ public Builder binding(String bindingName) {
+ apiVersion(CamelKSettings.V1ALPHA1);
+ this.bindingName = bindingName;
+ return this;
+ }
+
+ public Builder integration(Integration integration) {
+ this.integration = integration;
+ return this;
+ }
+
+ public Builder source(Source source) {
+ this.source = source;
+ return this;
+ }
+
+ public Builder source(String uri) {
+ return source(new SourceBuilder().withUri(uri).build());
+ }
+
+ public Builder source(Ref ref, String properties) {
+ Map props = null;
+ if (properties != null && !properties.isEmpty()) {
+ props = KubernetesSupport.yaml().load(properties);
+ }
+
+ return source(new SourceBuilder().withRef(ref)
+ .withNewProperties()
+ .addToAdditionalProperties(props)
+ .endProperties().build());
+ }
+
+ public Builder sink(Sink sink) {
+ this.sink = sink;
+ return this;
+ }
+
+ public Builder sink(String uri) {
+ return sink(new SinkBuilder().withUri(uri).build());
+ }
+
+ public Builder sink(org.apache.camel.v1alpha1.kameletbindingspec.sink.Ref ref, String properties) {
+ Map props = null;
+ if (properties != null && !properties.isEmpty()) {
+ props = KubernetesSupport.yaml().load(properties);
+ }
+
+ return sink(new SinkBuilder().withRef(ref)
+ .withNewProperties()
+ .addToAdditionalProperties(props)
+ .endProperties()
+ .build());
+ }
+
+ public Builder fromBuilder(KameletBindingBuilder builder) {
+ KameletBinding binding = builder.build();
+
+ bindingName = binding.getMetadata().getName();
+ integration = binding.getSpec().getIntegration();
+ source = binding.getSpec().getSource();
+ sink = binding.getSpec().getSink();
+
+ return this;
+ }
+
+ public Builder resource(Resource resource) {
+ this.resource = resource;
+ return this;
+ }
+
+ @Override
+ public CreateKameletBindingAction build() {
+ return new CreateKameletBindingAction(this);
+ }
+ }
+
+}
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/CreatePipeAction.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/CreatePipeAction.java
index 67351ebb..a868845b 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/CreatePipeAction.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/CreatePipeAction.java
@@ -17,7 +17,6 @@
package org.citrusframework.yaks.camelk.actions.kamelet;
import java.io.IOException;
-import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.StandardOpenOption;
@@ -25,20 +24,24 @@
import io.fabric8.kubernetes.client.KubernetesClient;
import io.fabric8.kubernetes.client.dsl.Updatable;
+import org.apache.camel.v1.Pipe;
+import org.apache.camel.v1.PipeBuilder;
+import org.apache.camel.v1.PipeSpecBuilder;
+import org.apache.camel.v1.pipespec.Integration;
+import org.apache.camel.v1.pipespec.Sink;
+import org.apache.camel.v1.pipespec.SinkBuilder;
+import org.apache.camel.v1.pipespec.Source;
+import org.apache.camel.v1.pipespec.SourceBuilder;
+import org.apache.camel.v1.pipespec.source.Ref;
import org.citrusframework.context.TestContext;
import org.citrusframework.exceptions.CitrusRuntimeException;
import org.citrusframework.spi.Resource;
import org.citrusframework.util.FileUtils;
+import org.citrusframework.util.IsJsonPredicate;
import org.citrusframework.yaks.YaksSettings;
-import org.citrusframework.yaks.camelk.CamelKSettings;
import org.citrusframework.yaks.camelk.jbang.CamelJBangSettings;
import org.citrusframework.yaks.camelk.jbang.ProcessAndOutput;
-import org.citrusframework.yaks.camelk.model.IntegrationSpec;
-import org.citrusframework.yaks.camelk.model.Pipe;
import org.citrusframework.yaks.camelk.model.PipeList;
-import org.citrusframework.yaks.camelk.model.PipeSpec;
-import org.citrusframework.yaks.camelk.model.v1alpha1.KameletBinding;
-import org.citrusframework.yaks.camelk.model.v1alpha1.KameletBindingList;
import org.citrusframework.yaks.kubernetes.KubernetesSupport;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
@@ -57,9 +60,9 @@ public class CreatePipeAction extends AbstractKameletAction {
private static final Logger LOG = LoggerFactory.getLogger(CreatePipeAction.class);
private final String pipeName;
- private final IntegrationSpec integration;
- private final PipeSpec.Endpoint source;
- private final PipeSpec.Endpoint sink;
+ private final Integration integration;
+ private final Source source;
+ private final Sink sink;
private final Resource resource;
/**
@@ -84,27 +87,33 @@ public void doExecute(TestContext context) {
if (resource != null) {
try {
- if (getApiVersion(context).equals(CamelKSettings.V1ALPHA1)) {
- pipe = KubernetesSupport.yaml().loadAs(
- context.replaceDynamicContentInString(FileUtils.readToString(resource)), KameletBinding.class);
+ String yamlOrJson = context.replaceDynamicContentInString(FileUtils.readToString(resource));
+ if (IsJsonPredicate.getInstance().test(yamlOrJson)) {
+ pipe = KubernetesSupport.json().readValue(yamlOrJson, Pipe.class);
} else {
- pipe = KubernetesSupport.yaml().loadAs(
- context.replaceDynamicContentInString(FileUtils.readToString(resource)), Pipe.class);
+ // need to make a detour over Json to support additional properties set on Pipe
+ Map raw = KubernetesSupport.yaml().load(yamlOrJson);
+ pipe = KubernetesSupport.json().convertValue(raw, Pipe.class);
}
} catch (IOException e) {
throw new CitrusRuntimeException(String.format("Failed to load pipe from resource %s", pipeName + ".yaml"), e);
}
} else {
- final Pipe.Builder builder = new Pipe.Builder()
- .name(pipeName);
+ final PipeBuilder builder = new PipeBuilder()
+ .withNewMetadata()
+ .withName(pipeName)
+ .endMetadata();
+ PipeSpecBuilder specBuilder = new PipeSpecBuilder();
if (integration != null) {
- builder.integration(integration);
+ specBuilder.withIntegration(integration);
}
if (source != null) {
- source.setProperties(context.resolveDynamicValuesInMap(source.getProperties()));
- builder.source(source);
+ if (source.getProperties() != null && source.getProperties().getAdditionalProperties() != null) {
+ context.resolveDynamicValuesInMap(source.getProperties().getAdditionalProperties());
+ }
+ specBuilder.withSource(source);
}
if (sink != null) {
@@ -112,15 +121,17 @@ public void doExecute(TestContext context) {
sink.setUri(context.replaceDynamicContentInString(sink.getUri()));
}
- sink.setProperties(context.resolveDynamicValuesInMap(sink.getProperties()));
- builder.sink(sink);
+ if (sink.getProperties() != null && sink.getProperties().getAdditionalProperties() != null) {
+ context.resolveDynamicValuesInMap(sink.getProperties().getAdditionalProperties());
+ }
+ specBuilder.withSink(sink);
}
- pipe = builder.build();
+ pipe = builder.withSpec(specBuilder.build()).build();
}
if (YaksSettings.isLocal(clusterType(context))) {
- createLocalPipe(pipe, pipeName, context);
+ createLocal(KubernetesSupport.yaml(new PipeValuePropertyMapper()).dumpAsMap(pipe), pipeName, context);
} else {
createPipe(getKubernetesClient(), namespace(context), pipe, context);
}
@@ -137,60 +148,31 @@ public void doExecute(TestContext context) {
*/
private void createPipe(KubernetesClient k8sClient, String namespace, Pipe pipe, TestContext context) {
if (LOG.isDebugEnabled()) {
- LOG.debug(KubernetesSupport.yaml().dumpAsMap(pipe));
+ LOG.debug(KubernetesSupport.yaml(new PipeValuePropertyMapper()).dumpAsMap(pipe));
}
- if (getApiVersion(context).equals(CamelKSettings.V1ALPHA1)) {
- KameletBinding kb;
- if (pipe instanceof KameletBinding) {
- kb = (KameletBinding) pipe;
- } else {
- kb = new KameletBinding.Builder().from(pipe).build();
- }
-
- k8sClient.resources(KameletBinding.class, KameletBindingList.class)
- .inNamespace(namespace)
- .resource(kb)
- .createOr(Updatable::update);
- } else {
- k8sClient.resources(Pipe.class, PipeList.class)
- .inNamespace(namespace)
- .resource(pipe)
- .createOr(Updatable::update);
- }
+ k8sClient.resources(Pipe.class, PipeList.class)
+ .inNamespace(namespace)
+ .resource(pipe)
+ .createOr(Updatable::update);
}
/**
* Creates the pipe with local JBang runtime.
- * @param pipe
+ * @param yaml
* @param name
* @param context
*/
- private void createLocalPipe(Pipe pipe, String name, TestContext context) {
+ private void createLocal(String yaml, String name, TestContext context) {
try {
- String pipeYaml;
-
- if (getApiVersion(context).equals(CamelKSettings.V1ALPHA1)) {
- KameletBinding kb;
- if (pipe instanceof KameletBinding) {
- kb = (KameletBinding) pipe;
- } else {
- kb = new KameletBinding.Builder().from(pipe).build();
- }
-
- pipeYaml = KubernetesSupport.yaml().dumpAsMap(kb);
- } else {
- pipeYaml = KubernetesSupport.yaml().dumpAsMap(pipe);
- }
-
if (LOG.isDebugEnabled()) {
- LOG.debug(pipeYaml);
+ LOG.debug(yaml);
}
Path workDir = CamelJBangSettings.getWorkDir();
Files.createDirectories(workDir);
Path file = workDir.resolve(String.format("i-%s.yaml", name));
- Files.write(file, pipeYaml.getBytes(StandardCharsets.UTF_8),
+ Files.writeString(file, yaml,
StandardOpenOption.WRITE,
StandardOpenOption.CREATE,
StandardOpenOption.TRUNCATE_EXISTING);
@@ -218,64 +200,65 @@ private void createLocalPipe(Pipe pipe, String name, TestContext context) {
public static final class Builder extends AbstractKameletAction.Builder {
private String pipeName;
- private IntegrationSpec integration;
- private PipeSpec.Endpoint source;
- private PipeSpec.Endpoint sink;
+ private Integration integration;
+ private Source source;
+ private Sink sink;
private Resource resource;
- public Builder binding(String pipeName) {
- apiVersion(CamelKSettings.V1ALPHA1);
- this.pipeName = pipeName;
- return this;
- }
-
public Builder pipe(String pipeName) {
this.pipeName = pipeName;
return this;
}
- public Builder integration(IntegrationSpec integration) {
+ public Builder integration(Integration integration) {
this.integration = integration;
return this;
}
- public Builder source(PipeSpec.Endpoint source) {
+ public Builder source(Source source) {
this.source = source;
return this;
}
public Builder source(String uri) {
- return source(new PipeSpec.Endpoint(uri));
+ return source(new SourceBuilder().withUri(uri).build());
}
- public Builder source(PipeSpec.Endpoint.ObjectReference ref, String properties) {
+ public Builder source(Ref ref, String properties) {
Map props = null;
if (properties != null && !properties.isEmpty()) {
props = KubernetesSupport.yaml().load(properties);
}
- return source(new PipeSpec.Endpoint(ref, props));
+ return source(new SourceBuilder().withRef(ref)
+ .withNewProperties()
+ .addToAdditionalProperties(props)
+ .endProperties().build());
}
- public Builder sink(PipeSpec.Endpoint sink) {
+ public Builder sink(Sink sink) {
this.sink = sink;
return this;
}
public Builder sink(String uri) {
- return sink(new PipeSpec.Endpoint(uri));
+ return sink(new SinkBuilder().withUri(uri).build());
}
- public Builder sink(PipeSpec.Endpoint.ObjectReference ref, String properties) {
+ public Builder sink(org.apache.camel.v1.pipespec.sink.Ref ref, String properties) {
Map props = null;
if (properties != null && !properties.isEmpty()) {
props = KubernetesSupport.yaml().load(properties);
}
- return sink(new PipeSpec.Endpoint(ref, props));
+ return sink(new SinkBuilder().withRef(ref)
+ .withNewProperties()
+ .addToAdditionalProperties(props)
+ .endProperties()
+ .build());
}
- public Builder fromBuilder(Pipe.Builder builder) {
+ public Builder fromBuilder(PipeBuilder builder) {
Pipe pipe = builder.build();
pipeName = pipe.getMetadata().getName();
@@ -296,4 +279,5 @@ public CreatePipeAction build() {
return new CreatePipeAction(this);
}
}
+
}
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/DeleteKameletAction.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/DeleteKameletAction.java
index 544eba7b..eb6451ac 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/DeleteKameletAction.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/DeleteKameletAction.java
@@ -16,9 +16,9 @@
package org.citrusframework.yaks.camelk.actions.kamelet;
+import org.apache.camel.v1.Kamelet;
import org.citrusframework.context.TestContext;
import org.citrusframework.yaks.camelk.CamelKSettings;
-import org.citrusframework.yaks.camelk.model.Kamelet;
import org.citrusframework.yaks.camelk.model.KameletList;
import org.citrusframework.yaks.camelk.model.v1alpha1.KameletV1Alpha1;
import org.citrusframework.yaks.camelk.model.v1alpha1.KameletV1Alpha1List;
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/DeleteKameletBindingAction.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/DeleteKameletBindingAction.java
new file mode 100644
index 00000000..bae2df62
--- /dev/null
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/DeleteKameletBindingAction.java
@@ -0,0 +1,101 @@
+/*
+ * Copyright the original author or 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 org.citrusframework.yaks.camelk.actions.kamelet;
+
+import io.fabric8.kubernetes.client.KubernetesClient;
+import org.apache.camel.v1alpha1.KameletBinding;
+import org.citrusframework.context.TestContext;
+import org.citrusframework.exceptions.CitrusRuntimeException;
+import org.citrusframework.yaks.YaksSettings;
+import org.citrusframework.yaks.camelk.CamelKSettings;
+import org.citrusframework.yaks.camelk.model.v1alpha1.KameletBindingList;
+
+import static org.citrusframework.yaks.camelk.jbang.CamelJBang.camel;
+
+/**
+ * @author Christoph Deppisch
+ */
+public class DeleteKameletBindingAction extends AbstractKameletAction {
+
+ private final String bindingName;
+
+ public DeleteKameletBindingAction(Builder builder) {
+ super("delete-binding", builder);
+
+ this.bindingName = builder.bindingName;
+ }
+
+ @Override
+ public void doExecute(TestContext context) {
+ String binding = context.replaceDynamicContentInString(bindingName);
+
+ LOG.info(String.format("Deleting binding '%s'", binding));
+
+ if (YaksSettings.isLocal(clusterType(context))) {
+ deleteLocalBinding(binding, context);
+ } else {
+ deleteBinding(getKubernetesClient(), namespace(context), binding, context);
+ }
+
+ LOG.info(String.format("Successfully deleted binding '%s'", binding));
+ }
+
+ private void deleteBinding(KubernetesClient k8sClient, String namespace, String name, TestContext context) {
+ k8sClient.resources(KameletBinding.class, KameletBindingList.class)
+ .inNamespace(namespace)
+ .withName(name)
+ .delete();
+ }
+
+ /**
+ * Deletes the Camel K integration from local JBang runtime.
+ * @param name
+ * @param context
+ */
+ private static void deleteLocalBinding(String name, TestContext context) {
+ Long pid;
+ if (context.getVariables().containsKey(name + ":pid")) {
+ pid = context.getVariable(name + ":pid", Long.class);
+ } else {
+ pid = camel().getAll().stream()
+ .filter(props -> name.equals(props.get("NAME")) && !props.getOrDefault("PID", "").isBlank())
+ .map(props -> Long.valueOf(props.get("PID"))).findFirst()
+ .orElseThrow(() -> new CitrusRuntimeException(String.format("Unable to retrieve binding process id %s:pid", name)));
+ }
+
+ camel().stop(pid);
+ }
+
+ /**
+ * Action builder.
+ */
+ public static class Builder extends AbstractKameletAction.Builder {
+
+ private String bindingName;
+
+ public Builder binding(String name) {
+ apiVersion(CamelKSettings.V1ALPHA1);
+ this.bindingName = name;
+ return this;
+ }
+
+ @Override
+ public DeleteKameletBindingAction build() {
+ return new DeleteKameletBindingAction(this);
+ }
+ }
+}
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/DeletePipeAction.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/DeletePipeAction.java
index 0f38fe97..0a247a21 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/DeletePipeAction.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/DeletePipeAction.java
@@ -16,14 +16,11 @@
package org.citrusframework.yaks.camelk.actions.kamelet;
+import io.fabric8.kubernetes.client.KubernetesClient;
+import org.apache.camel.v1.Pipe;
import org.citrusframework.context.TestContext;
import org.citrusframework.exceptions.CitrusRuntimeException;
-import io.fabric8.kubernetes.client.KubernetesClient;
import org.citrusframework.yaks.YaksSettings;
-import org.citrusframework.yaks.camelk.CamelKSettings;
-import org.citrusframework.yaks.camelk.model.v1alpha1.KameletBinding;
-import org.citrusframework.yaks.camelk.model.v1alpha1.KameletBindingList;
-import org.citrusframework.yaks.camelk.model.Pipe;
import org.citrusframework.yaks.camelk.model.PipeList;
import static org.citrusframework.yaks.camelk.jbang.CamelJBang.camel;
@@ -48,26 +45,19 @@ public void doExecute(TestContext context) {
LOG.info(String.format("Deleting pipe '%s'", pipe));
if (YaksSettings.isLocal(clusterType(context))) {
- deleteLocalBinding(pipe, context);
+ deleteLocalPipe(pipe, context);
} else {
- deleteBinding(getKubernetesClient(), namespace(context), pipe, context);
+ deletePipe(getKubernetesClient(), namespace(context), pipe, context);
}
LOG.info(String.format("Successfully deleted pipe '%s'", pipe));
}
- private void deleteBinding(KubernetesClient k8sClient, String namespace, String name, TestContext context) {
- if (getApiVersion(context).equals(CamelKSettings.V1ALPHA1)) {
- k8sClient.resources(KameletBinding.class, KameletBindingList.class)
- .inNamespace(namespace)
- .withName(name)
- .delete();
- } else {
- k8sClient.resources(Pipe.class, PipeList.class)
+ private void deletePipe(KubernetesClient k8sClient, String namespace, String name, TestContext context) {
+ k8sClient.resources(Pipe.class, PipeList.class)
.inNamespace(namespace)
.withName(name)
.delete();
- }
}
/**
@@ -75,7 +65,7 @@ private void deleteBinding(KubernetesClient k8sClient, String namespace, String
* @param name
* @param context
*/
- private static void deleteLocalBinding(String name, TestContext context) {
+ private static void deleteLocalPipe(String name, TestContext context) {
Long pid;
if (context.getVariables().containsKey(name + ":pid")) {
pid = context.getVariable(name + ":pid", Long.class);
@@ -96,12 +86,6 @@ public static class Builder extends AbstractKameletAction.Builder {
LOG.info(String.format("Verify Kamlet '%s' exists in namespace '%s'", name, namespace));
- Kamelet kamelet;
+ HasMetadata kamelet;
if (getApiVersion(context).equals(CamelKSettings.V1ALPHA1)) {
kamelet = getKubernetesClient().resources(KameletV1Alpha1.class, KameletV1Alpha1List.class)
.inNamespace(namespace)
@@ -91,7 +92,7 @@ private boolean findKamelet(String name, TestContext context, String ... namespa
LOG.debug(String.format("Kamelet '%s' is not present in namespace '%s'", name, namespace));
} else {
LOG.debug(String.format("Found Kamelet in namespace '%s'", namespace));
- LOG.debug(KubernetesSupport.yaml().dumpAsMap(kamelet));
+ LOG.debug(KubernetesSupport.yaml(new KameletValuePropertyMapper()).dumpAsMap(kamelet));
}
}
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/VerifyKameletBindingAction.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/VerifyKameletBindingAction.java
new file mode 100644
index 00000000..da412cf9
--- /dev/null
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/VerifyKameletBindingAction.java
@@ -0,0 +1,153 @@
+/*
+ * Copyright the original author or 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 org.citrusframework.yaks.camelk.actions.kamelet;
+
+import java.util.Map;
+
+import org.apache.camel.v1alpha1.KameletBinding;
+import org.citrusframework.context.TestContext;
+import org.citrusframework.exceptions.ValidationException;
+import org.citrusframework.yaks.YaksSettings;
+import org.citrusframework.yaks.camelk.CamelKSettings;
+import org.citrusframework.yaks.camelk.model.v1alpha1.KameletBindingList;
+import org.citrusframework.yaks.kubernetes.KubernetesSupport;
+
+import static org.citrusframework.yaks.camelk.jbang.CamelJBang.camel;
+
+/**
+ * Test action verifies Camel K binding is present in given namespace.
+ *
+ * @author Christoph Deppisch
+ */
+public class VerifyKameletBindingAction extends AbstractKameletAction {
+
+ private final String bindingName;
+
+ private final int maxAttempts;
+ private final long delayBetweenAttempts;
+
+ /**
+ * Constructor using given builder.
+ * @param builder
+ */
+ public VerifyKameletBindingAction(Builder builder) {
+ super("verify-binding", builder);
+ this.bindingName = builder.bindingName;
+ this.maxAttempts = builder.maxAttempts;
+ this.delayBetweenAttempts = builder.delayBetweenAttempts;
+ }
+
+ @Override
+ public void doExecute(TestContext context) {
+ String name = context.replaceDynamicContentInString(this.bindingName);
+
+ LOG.info(String.format("Verify binding '%s'", name));
+
+ if (YaksSettings.isLocal(clusterType(context))) {
+ verifyLocalKameletBinding(name, context);
+ } else {
+ verifyKameletBinding(namespace(context), name, context);
+ }
+
+ LOG.info(String.format("Successfully verified binding '%s' - All values OK!", name));
+ }
+
+ private void verifyLocalKameletBinding(String name, TestContext context) {
+ Long pid = context.getVariable(name + ":pid", Long.class);
+
+ for (int i = 0; i < maxAttempts; i++) {
+ Map properties = camel().get(pid);
+ if ((!properties.isEmpty() && properties.get("STATUS").equals("Running"))) {
+ LOG.info(String.format("Verified binding '%s' state 'Running' - All values OK!", name));
+ return;
+ }
+
+ LOG.info(String.format("Waiting for binding '%s' to be in state 'Running'- retry in %s ms", name, delayBetweenAttempts));
+ try {
+ Thread.sleep(delayBetweenAttempts);
+ } catch (InterruptedException e) {
+ LOG.warn("Interrupted while waiting for binding", e);
+ }
+ }
+
+ throw new ValidationException(String.format("Failed to retrieve binding '%s' in state 'Running'", name));
+ }
+
+ private void verifyKameletBinding(String namespace, String name, TestContext context) {
+ KameletBinding binding = null;
+ for (int i = 0; i < maxAttempts; i++) {
+ binding = getKubernetesClient().resources(KameletBinding.class, KameletBindingList.class)
+ .inNamespace(namespace)
+ .withName(name)
+ .get();
+
+ if (binding == null) {
+ LOG.info(String.format("Waiting for binding '%s' - retry in %s ms", name, delayBetweenAttempts));
+ try {
+ Thread.sleep(delayBetweenAttempts);
+ } catch (InterruptedException e) {
+ LOG.warn("Interrupted while waiting for binding", e);
+ }
+ } else {
+ break;
+ }
+ }
+
+ if (binding == null) {
+ throw new ValidationException(String.format("Failed to retrieve binding '%s' in namespace '%s'", name, namespace));
+ }
+
+ if (LOG.isDebugEnabled()) {
+ LOG.debug(KubernetesSupport.yaml(new KameletBindingValuePropertyMapper()).dumpAsMap(binding));
+ }
+ }
+
+ /**
+ * Action builder.
+ */
+ public static final class Builder extends AbstractKameletAction.Builder {
+
+ private String bindingName;
+
+ private int maxAttempts = CamelKSettings.getMaxAttempts();
+ private long delayBetweenAttempts = CamelKSettings.getDelayBetweenAttempts();
+
+ public Builder isAvailable() {
+ return this;
+ }
+
+ public Builder isAvailable(String name) {
+ this.bindingName = name;
+ return this;
+ }
+
+ public Builder maxAttempts(int maxAttempts) {
+ this.maxAttempts = maxAttempts;
+ return this;
+ }
+
+ public Builder delayBetweenAttempts(long delayBetweenAttempts) {
+ this.delayBetweenAttempts = delayBetweenAttempts;
+ return this;
+ }
+
+ @Override
+ public VerifyKameletBindingAction build() {
+ return new VerifyKameletBindingAction(this);
+ }
+ }
+}
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/VerifyPipeAction.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/VerifyPipeAction.java
index 7df3db96..013c1ca3 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/VerifyPipeAction.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/actions/kamelet/VerifyPipeAction.java
@@ -18,20 +18,18 @@
import java.util.Map;
+import org.apache.camel.v1.Pipe;
import org.citrusframework.context.TestContext;
import org.citrusframework.exceptions.ValidationException;
import org.citrusframework.yaks.YaksSettings;
import org.citrusframework.yaks.camelk.CamelKSettings;
-import org.citrusframework.yaks.camelk.model.v1alpha1.KameletBinding;
-import org.citrusframework.yaks.camelk.model.v1alpha1.KameletBindingList;
-import org.citrusframework.yaks.camelk.model.Pipe;
import org.citrusframework.yaks.camelk.model.PipeList;
import org.citrusframework.yaks.kubernetes.KubernetesSupport;
import static org.citrusframework.yaks.camelk.jbang.CamelJBang.camel;
/**
- * Test action verifies Camel K binding is present in given namespace.
+ * Test action verifies Camel K pipe is present in given namespace.
*
* @author Christoph Deppisch
*/
@@ -92,17 +90,10 @@ private void verifyLocalPipe(String name, TestContext context) {
private void verifyPipe(String namespace, String name, TestContext context) {
Pipe pipe = null;
for (int i = 0; i < maxAttempts; i++) {
- if (getApiVersion(context).equals(CamelKSettings.V1ALPHA1)) {
- pipe = getKubernetesClient().resources(KameletBinding.class, KameletBindingList.class)
- .inNamespace(namespace)
- .withName(name)
- .get();
- } else {
- pipe = getKubernetesClient().resources(Pipe.class, PipeList.class)
- .inNamespace(namespace)
- .withName(name)
- .get();
- }
+ pipe = getKubernetesClient().resources(Pipe.class, PipeList.class)
+ .inNamespace(namespace)
+ .withName(name)
+ .get();
if (pipe == null) {
LOG.info(String.format("Waiting for pipe '%s' - retry in %s ms", name, delayBetweenAttempts));
@@ -121,7 +112,7 @@ private void verifyPipe(String namespace, String name, TestContext context) {
}
if (LOG.isDebugEnabled()) {
- LOG.debug(KubernetesSupport.yaml().dumpAsMap(pipe));
+ LOG.debug(KubernetesSupport.yaml(new PipeValuePropertyMapper()).dumpAsMap(pipe));
}
}
@@ -135,7 +126,6 @@ public static final class Builder extends AbstractKameletAction.Builder it.contains("--logging-color"))) {
+ // disable logging colors when writing logs to file
+ runArgs.add("--logging-color=false");
+ }
+
return executeAsync(camel(runArgs.toArray(String[]::new)), outputFile);
} else {
return executeAsync(camel(runArgs.toArray(String[]::new)));
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/jbang/ProcessAndOutput.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/jbang/ProcessAndOutput.java
index 3df20de7..473ac3b6 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/jbang/ProcessAndOutput.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/jbang/ProcessAndOutput.java
@@ -86,7 +86,7 @@ private void readAllAndClose() {
builder.append(line).append(System.lineSeparator());
}
- if (builder.length() > 0) {
+ if (!builder.isEmpty()) {
output += builder;
}
} catch (IOException e) {
@@ -124,7 +124,7 @@ private void readChunk() {
throw new CitrusRuntimeException("Failed to get JBang process output", e);
}
- if (builder.length() > 0) {
+ if (!builder.isEmpty()) {
if (output == null) {
output = builder.toString();
} else {
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/model/Integration.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/model/Integration.java
deleted file mode 100644
index 61c1d393..00000000
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/model/Integration.java
+++ /dev/null
@@ -1,144 +0,0 @@
-/*
- * Copyright the original author or 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 org.citrusframework.yaks.camelk.model;
-
-import java.util.ArrayList;
-import java.util.Collections;
-import java.util.LinkedHashMap;
-import java.util.List;
-import java.util.Map;
-
-import com.fasterxml.jackson.annotation.JsonInclude;
-import io.fabric8.kubernetes.api.model.Namespaced;
-import io.fabric8.kubernetes.client.CustomResource;
-import io.fabric8.kubernetes.model.annotation.Group;
-import io.fabric8.kubernetes.model.annotation.Version;
-import org.citrusframework.yaks.camelk.CamelKSettings;
-import org.citrusframework.yaks.camelk.CamelKSupport;
-
-@JsonInclude(JsonInclude.Include.NON_NULL)
-@Group(CamelKSupport.CAMELK_CRD_GROUP)
-@Version(CamelKSettings.API_VERSION_DEFAULT)
-public class Integration extends CustomResource implements Namespaced {
-
- public Integration() {
- super();
- this.spec = new IntegrationSpec();
- this.status = null;
- }
-
- @Override
- public String getApiVersion() {
- return CamelKSupport.CAMELK_CRD_GROUP + "/" + CamelKSettings.getApiVersion();
- }
-
- /**
- * Fluent builder
- */
- public static class Builder {
- private final Map traits = new LinkedHashMap<>();
- private final List resources = new ArrayList<>();
- private final List dependencies = new ArrayList<>();
- private final List configuration = new ArrayList<>();
- private String source;
- private String fileName;
- private String name;
-
- public Builder name(String name) {
- this.name = name;
-
- if (fileName == null) {
- this.fileName = name;
- }
-
- return this;
- }
-
- public Builder source(String source) {
- this.source = source;
- return this;
- }
-
- public Builder source(String fileName, String source) {
- this.source = source;
- this.fileName = fileName;
- return this;
- }
-
- public Builder openApi(String fileName, String content) {
- this.resources.add(new IntegrationSpec.Resource("openapi", null, fileName, content));
- return this;
- }
-
- public Builder dependencies(List dependencies) {
- this.dependencies.addAll(dependencies);
- return this;
- }
-
- public Builder traits(Map traits) {
- this.traits.putAll(traits);
- return this;
- }
-
- public Builder trait(String name, IntegrationSpec.TraitConfig config) {
- this.traits.put(name ,config);
- return this;
- }
-
- public Builder configuration(List configuration) {
- this.configuration.addAll(configuration);
- return this;
- }
-
- public Integration build() {
- Integration i = new Integration();
- i.getMetadata().setName(sanitizeIntegrationName(name));
- i.getSpec().setSources(Collections.singletonList(new IntegrationSpec.Source(fileName, source)));
-
- if (!dependencies.isEmpty()) {
- i.getSpec().setDependencies(dependencies);
- }
-
- if (!traits.isEmpty()) {
- i.getSpec().setTraits(traits);
- }
-
- if (!configuration.isEmpty()) {
- i.getSpec().setConfiguration(configuration);
- }
-
- if (!resources.isEmpty()) {
- i.getSpec().setResources(resources);
- }
-
- return i;
- }
-
- private String sanitizeIntegrationName(String name) {
- String sanitized;
-
- if (name.contains(".")) {
- sanitized = name.substring(0, name.indexOf("."));
- } else {
- sanitized = name;
- }
-
- sanitized = sanitized.replaceAll("([a-z])([A-Z]+)", "$1-$2").toLowerCase();
- return sanitized.replaceAll("[^a-z0-9-]", "");
- }
- }
-}
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/model/IntegrationList.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/model/IntegrationList.java
index 4b58dc2a..b4167a39 100644
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/model/IntegrationList.java
+++ b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/model/IntegrationList.java
@@ -17,6 +17,7 @@
package org.citrusframework.yaks.camelk.model;
import io.fabric8.kubernetes.api.model.DefaultKubernetesResourceList;
+import org.apache.camel.v1.Integration;
/**
* @author Christoph Deppisch
diff --git a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/model/IntegrationSpec.java b/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/model/IntegrationSpec.java
deleted file mode 100644
index c6c505c0..00000000
--- a/java/steps/yaks-camel-k/src/main/java/org/citrusframework/yaks/camelk/model/IntegrationSpec.java
+++ /dev/null
@@ -1,413 +0,0 @@
-/*
- * Copyright the original author or 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 org.citrusframework.yaks.camelk.model;
-
-import java.util.HashMap;
-import java.util.List;
-import java.util.Map;
-
-import com.fasterxml.jackson.annotation.JsonInclude;
-import com.fasterxml.jackson.annotation.JsonProperty;
-import com.fasterxml.jackson.annotation.JsonPropertyOrder;
-import com.fasterxml.jackson.databind.JsonDeserializer;
-import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
-import io.fabric8.kubernetes.api.model.KubernetesResource;
-
-@JsonDeserialize(using = JsonDeserializer.None.class)
-@JsonInclude(JsonInclude.Include.NON_NULL)
-@JsonPropertyOrder({"replicas", "flows", "sources", "resources", "kit", "dependencies", "profile", "traits",
- "configuration", "repositories", "serviceAccountName"})
-public class IntegrationSpec implements KubernetesResource {
-
- @JsonProperty("replicas")
- private Integer replicas;
- @JsonProperty("flows")
- @JsonInclude(JsonInclude.Include.NON_EMPTY)
- private List