Skip to content

Commit

Permalink
(#5522) Enhance environment trait to include values from secrets/conf…
Browse files Browse the repository at this point in the history
…igmaps
  • Loading branch information
tdiesler committed Jun 17, 2024
1 parent ca5239b commit d66a444
Show file tree
Hide file tree
Showing 5 changed files with 157 additions and 7 deletions.
53 changes: 51 additions & 2 deletions e2e/advanced/environment_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,56 @@ import (
"github.com/apache/camel-k/v2/pkg/util/defaults"
)

func TestEnvironmentTrait(t *testing.T) {
func TestEnvironmentTraitVars(t *testing.T) {
t.Parallel()

WithNewTestNamespace(t, func(ctx context.Context, g *WithT, ns string) {
operatorID := "camel-k-trait-environment"
g.Expect(CopyCamelCatalog(t, ctx, ns, operatorID)).To(Succeed())
g.Expect(CopyIntegrationKits(t, ctx, ns, operatorID)).To(Succeed())
g.Expect(KamelInstallWithID(t, ctx, operatorID, ns)).To(Succeed())

g.Eventually(SelectedPlatformPhase(t, ctx, ns, operatorID), TestTimeoutMedium).Should(Equal(v1.IntegrationPlatformPhaseReady))

// Create configmap
var cmData = make(map[string]string)
cmData["my-cm-key"] = "hello configmap"
g.Expect(CreatePlainTextConfigmap(t, ctx, ns, "my-cm", cmData)).Should(Succeed())

// Create secret
var secData = make(map[string]string)
secData["my-sec-key"] = "very top secret"
g.Expect(CreatePlainTextSecret(t, ctx, ns, "my-sec", secData)).Should(Succeed())

t.Run("Run simple env-var", func(t *testing.T) {
name := RandomizedSuffixName("envvar")
g.Expect(KamelRunWithID(t, ctx, operatorID, ns, "--name", name, "-t", "environment.vars=MY_ENV_VAR='hello world'", "files/envvar.yaml").Execute()).To(Succeed())
g.Eventually(IntegrationPodPhase(t, ctx, ns, name), TestTimeoutLong).Should(Equal(corev1.PodRunning))
g.Eventually(IntegrationConditionStatus(t, ctx, ns, name, v1.IntegrationConditionReady), TestTimeoutShort).Should(Equal(corev1.ConditionTrue))
g.Eventually(IntegrationLogs(t, ctx, ns, name), TestTimeoutShort).Should(ContainSubstring("hello world"))
})

t.Run("Run env-var from configmap", func(t *testing.T) {
name := RandomizedSuffixName("envvar-configmap")
g.Expect(KamelRunWithID(t, ctx, operatorID, ns, "--name", name, "-t", "environment.vars=MY_ENV_VAR=configmap:my-cm/my-cm-key", "files/envvar.yaml").Execute()).To(Succeed())
g.Eventually(IntegrationPodPhase(t, ctx, ns, name), TestTimeoutLong).Should(Equal(corev1.PodRunning))
g.Eventually(IntegrationConditionStatus(t, ctx, ns, name, v1.IntegrationConditionReady), TestTimeoutShort).Should(Equal(corev1.ConditionTrue))
g.Eventually(IntegrationLogs(t, ctx, ns, name), TestTimeoutShort).Should(ContainSubstring("hello configmap"))
})

t.Run("Run env-var from secret", func(t *testing.T) {
name := RandomizedSuffixName("envvar-secret")
g.Expect(KamelRunWithID(t, ctx, operatorID, ns, "--name", name, "-t", "environment.vars=MY_ENV_VAR=secret:my-sec/my-sec-key", "files/envvar.yaml").Execute()).To(Succeed())
g.Eventually(IntegrationPodPhase(t, ctx, ns, name), TestTimeoutLong).Should(Equal(corev1.PodRunning))
g.Eventually(IntegrationConditionStatus(t, ctx, ns, name, v1.IntegrationConditionReady), TestTimeoutShort).Should(Equal(corev1.ConditionTrue))
g.Eventually(IntegrationLogs(t, ctx, ns, name), TestTimeoutShort).Should(ContainSubstring("very top secret"))
})

g.Expect(Kamel(t, ctx, "delete", "--all", "-n", ns).Execute()).To(Succeed())
})
}

func TestEnvironmentTraitHttpProxy(t *testing.T) {
t.Parallel()

WithNewTestNamespace(t, func(ctx context.Context, g *WithT, ns string) {
Expand Down Expand Up @@ -67,7 +116,7 @@ func TestEnvironmentTrait(t *testing.T) {
}

// Install Camel K with the HTTP proxy environment variable
operatorID := "camel-k-trait-environment"
operatorID := "camel-k-trait-environment-http"
g.Expect(CopyCamelCatalog(t, ctx, ns, operatorID)).To(Succeed())
g.Expect(CopyIntegrationKits(t, ctx, ns, operatorID)).To(Succeed())
g.Expect(KamelInstallWithID(t, ctx, operatorID, ns, "--operator-env-vars", fmt.Sprintf("HTTP_PROXY=%s", httpProxy), "--operator-env-vars", "NO_PROXY="+strings.Join(noProxy, ","))).To(Succeed())
Expand Down
25 changes: 25 additions & 0 deletions e2e/advanced/files/envvar.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
# camel-k: language=yaml

# ---------------------------------------------------------------------------
# Licensed to the Apache Software Foundation (ASF) under one or more
# contributor license agreements. See the NOTICE file distributed with
# this work for additional information regarding copyright ownership.
# The ASF licenses this file to You under the Apache License, Version 2.0
# (the "License"); you may not use this file except in compliance with
# the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
# ---------------------------------------------------------------------------

- from:
uri: "timer:tick"
steps:
- setBody:
simple: "${env:MY_ENV_VAR}"
- to: "log:info"
1 change: 1 addition & 0 deletions pkg/apis/camel/v1/trait/environment.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ type EnvironmentTrait struct {
HTTPProxy *bool `property:"http-proxy" json:"httpProxy,omitempty"`
// A list of environment variables to be added to the integration container.
// The syntax is KEY=VALUE, e.g., `MY_VAR="my value"`.
// The value may also be a reference to a configmap or secret, e.g. `MY_VAR=configmap:my-cm/my-cm-key`
// These take precedence over the previously defined environment variables.
Vars []string `property:"vars" json:"vars,omitempty"`
}
81 changes: 76 additions & 5 deletions pkg/trait/environment.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,9 @@ limitations under the License.
package trait

import (
corev1 "k8s.io/api/core/v1"
"os"
"strings"

"k8s.io/utils/pointer"

Expand Down Expand Up @@ -46,7 +48,7 @@ const (
envVarCamelKRuntimeVersion = "CAMEL_K_RUNTIME_VERSION"
envVarMountPathConfigMaps = "CAMEL_K_MOUNT_PATH_CONFIGMAPS"

// Disabling gosec linter as it may triggers:
// Disabling gosec linter as it may trigger:
//
// pkg/trait/environment.go:41: G101: Potential hardcoded credentials (gosec)
// envVarMountPathSecrets = "CAMEL_K_MOUNT_PATH_SECRETS"
Expand Down Expand Up @@ -99,12 +101,81 @@ func (t *environmentTrait) Apply(e *Environment) error {
}
}

if t.Vars != nil {
for _, env := range t.Vars {
k, v := property.SplitPropertyFileEntry(env)
for _, env := range t.Vars {
k, v := property.SplitPropertyFileEntry(env)
switch {
case strings.HasPrefix(v, "configmap:"):
path := strings.TrimPrefix(v, "configmap:")
setValFromConfigMapKeySelector(&e.EnvVars, k, path)
case strings.HasPrefix(v, "secret:"):
path := strings.TrimPrefix(v, "secret:")
setValFromSecretKeySelector(&e.EnvVars, k, path)
default:
envvar.SetVal(&e.EnvVars, k, v)
}
}

return nil
}

func setValFromConfigMapKeySelector(vars *[]corev1.EnvVar, envName string, path string) {
refName, srcKey := splitValueToNameAndKey(path)
if envVar := envvar.Get(*vars, envName); envVar != nil {
envVar.Value = ""
envVar.ValueFrom = &corev1.EnvVarSource{
ConfigMapKeyRef: &corev1.ConfigMapKeySelector{
LocalObjectReference: corev1.LocalObjectReference{
Name: refName,
},
Key: srcKey,
},
}
} else {
*vars = append(*vars, corev1.EnvVar{
Name: envName,
ValueFrom: &corev1.EnvVarSource{
ConfigMapKeyRef: &corev1.ConfigMapKeySelector{
LocalObjectReference: corev1.LocalObjectReference{
Name: refName,
},
Key: srcKey,
},
},
})
}
}

func setValFromSecretKeySelector(vars *[]corev1.EnvVar, envName string, path string) {
refName, srcKey := splitValueToNameAndKey(path)
if envVar := envvar.Get(*vars, envName); envVar != nil {
envVar.Value = ""
envVar.ValueFrom = &corev1.EnvVarSource{
SecretKeyRef: &corev1.SecretKeySelector{
LocalObjectReference: corev1.LocalObjectReference{
Name: refName,
},
Key: srcKey,
},
}
} else {
*vars = append(*vars, corev1.EnvVar{
Name: envName,
ValueFrom: &corev1.EnvVarSource{
SecretKeyRef: &corev1.SecretKeySelector{
LocalObjectReference: corev1.LocalObjectReference{
Name: refName,
},
Key: srcKey,
},
},
})
}
}

func splitValueToNameAndKey(path string) (string, string) {
toks := strings.SplitN(path, "/", 2)
name, key := toks[0], ""
if len(toks) > 1 {
key = toks[1]
}
return name, key
}
4 changes: 4 additions & 0 deletions script/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -397,9 +397,13 @@ endif
./script/bundle_kamelets.sh $(KAMELET_CATALOG_REPO) $(KAMELET_CATALOG_REPO_TAG)

build-compile-integration-tests:
ifndef NOTEST
@echo "####### Compiling integration tests..."
export CAMEL_K_E2E_JUST_COMPILE="true"; \
go test -run nope -tags="integration" ./e2e/...
else
@echo "####### Skipping e2e test compile..."
endif

clean:
# disable gomodules when executing go clean:
Expand Down

0 comments on commit d66a444

Please sign in to comment.