Skip to content

Commit

Permalink
Allow use of prefix on envFrom
Browse files Browse the repository at this point in the history
  • Loading branch information
mcruzdev committed Apr 12, 2024
1 parent bb6da8e commit 364d6ac
Show file tree
Hide file tree
Showing 19 changed files with 405 additions and 38 deletions.
2 changes: 1 addition & 1 deletion bom/application/pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -164,7 +164,7 @@
<kotlin.coroutine.version>1.8.0</kotlin.coroutine.version>
<azure.toolkit-lib.version>0.27.0</azure.toolkit-lib.version>
<kotlin-serialization.version>1.6.2</kotlin-serialization.version>
<dekorate.version>4.1.2</dekorate.version> <!-- Please check with Java Operator SDK team before updating -->
<dekorate.version>4.1.3</dekorate.version> <!-- Please check with Java Operator SDK team before updating -->
<maven-invoker.version>3.2.0</maven-invoker.version>
<awaitility.version>4.2.1</awaitility.version>
<jboss-logmanager.version>3.0.4.Final</jboss-logmanager.version>
Expand Down
34 changes: 34 additions & 0 deletions docs/src/main/asciidoc/deploying-to-kubernetes.adoc
Original file line number Diff line number Diff line change
Expand Up @@ -467,6 +467,23 @@ This would generate the following in the `env` section of your container:
optional: false
----

It's also possible to add a prefix when you are generating env from Secret, the following configuration creates environment variable from Secret with key `foo` adding a prefix `BAR`:

Check warning on line 470 in docs/src/main/asciidoc/deploying-to-kubernetes.adoc

View workflow job for this annotation

GitHub Actions / Linting with Vale

[vale] reported by reviewdog 🐶 [Quarkus.Spelling] Use correct American English spelling. Did you really mean 'env'? Raw Output: {"message": "[Quarkus.Spelling] Use correct American English spelling. Did you really mean 'env'?", "location": {"path": "docs/src/main/asciidoc/deploying-to-kubernetes.adoc", "range": {"start": {"line": 470, "column": 60}}}, "severity": "WARNING"}

[source,properties]
----
quarkus.kubernetes.env.secrets=foo[BAR]
----

This would generate the following in the `env` section of your container:
[source,yaml]
----
- env:
envFrom:
- secretRef:
name: foo
prefix: BAR
----

===== Environment variables from ConfigMap

To add all key/value pairs from `ConfigMap` as environment variables just apply the following configuration, separating each

Check warning on line 489 in docs/src/main/asciidoc/deploying-to-kubernetes.adoc

View workflow job for this annotation

GitHub Actions / Linting with Vale

[vale] reported by reviewdog 🐶 [Quarkus.TermsSuggestions] Depending on the context, consider using 'because' or 'while' rather than 'as'. Raw Output: {"message": "[Quarkus.TermsSuggestions] Depending on the context, consider using 'because' or 'while' rather than 'as'.", "location": {"path": "docs/src/main/asciidoc/deploying-to-kubernetes.adoc", "range": {"start": {"line": 489, "column": 34}}}, "severity": "INFO"}
Expand Down Expand Up @@ -512,6 +529,23 @@ This would generate the following in the `env` section of your container:
optional: false
----

It's also possible to add a prefix when you are generating env from ConfigMap, the following configuration creates environment variable from ConfigMap with key `foo` adding a prefix `BAR`:

Check warning on line 532 in docs/src/main/asciidoc/deploying-to-kubernetes.adoc

View workflow job for this annotation

GitHub Actions / Linting with Vale

[vale] reported by reviewdog 🐶 [Quarkus.Spelling] Use correct American English spelling. Did you really mean 'env'? Raw Output: {"message": "[Quarkus.Spelling] Use correct American English spelling. Did you really mean 'env'?", "location": {"path": "docs/src/main/asciidoc/deploying-to-kubernetes.adoc", "range": {"start": {"line": 532, "column": 60}}}, "severity": "WARNING"}

[source,properties]
----
quarkus.kubernetes.env.configmaps=foo[BAR]
----

This would generate the following in the `env` section of your container:
[source,yaml]
----
- env:
envFrom:
- configMapRef:
name: foo
prefix: BAR
----

===== Environment variables from fields

It's also possible to use the value from another field to add a new environment variable by specifying the path of the field to be used as a source, as follows:

Check warning on line 551 in docs/src/main/asciidoc/deploying-to-kubernetes.adoc

View workflow job for this annotation

GitHub Actions / Linting with Vale

[vale] reported by reviewdog 🐶 [Quarkus.TermsSuggestions] Depending on the context, consider using 'because' or 'while' rather than 'as'. Raw Output: {"message": "[Quarkus.TermsSuggestions] Depending on the context, consider using 'because' or 'while' rather than 'as'.", "location": {"path": "docs/src/main/asciidoc/deploying-to-kubernetes.adoc", "range": {"start": {"line": 551, "column": 137}}}, "severity": "INFO"}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -54,42 +54,45 @@ public boolean mightConflictWith(EnvType type) {
private final EnvType type;
private final String target;
private final boolean oldStyle;
private final String prefix;

public static KubernetesEnvBuildItem createFromField(String name, String targetField, String target,
boolean... oldStyle) {
return create(name, null, null, null, targetField, target, isOldStyle(oldStyle));
return create(name, null, null, null, targetField, target, null, isOldStyle(oldStyle));
}

public static KubernetesEnvBuildItem createFromConfigMap(String configMapName, String target, boolean... oldStyle) {
return create(configMapName, null, null, configMapName, null, target, isOldStyle(oldStyle));
public static KubernetesEnvBuildItem createFromConfigMap(String configMapName, String target, String prefix,
boolean... oldStyle) {
return create(configMapName, null, null, configMapName, null, target, prefix, isOldStyle(oldStyle));
}

public static KubernetesEnvBuildItem createFromSecret(String secretName, String target, boolean... oldStyle) {
return create(secretName, null, secretName, null, null, target, isOldStyle(oldStyle));
public static KubernetesEnvBuildItem createFromSecret(String secretName, String target, String prefix,
boolean... oldStyle) {
return create(secretName, null, secretName, null, null, target, prefix, isOldStyle(oldStyle));
}

public static KubernetesEnvBuildItem createSimpleVar(String name, String value, String target,
boolean... oldStyle) {
return create(name, value, null, null, null, target, isOldStyle(oldStyle));
return create(name, value, null, null, null, target, null, isOldStyle(oldStyle));
}

public static KubernetesEnvBuildItem createFromConfigMapKey(String varName, String key, String configmap, String target,
boolean... oldStyle) {
return create(varName, key, null, configmap, null, target, isOldStyle(oldStyle));
String prefix, boolean... oldStyle) {
return create(varName, key, null, configmap, null, target, prefix, isOldStyle(oldStyle));
}

public static KubernetesEnvBuildItem createFromSecretKey(String varName, String key, String secret, String target,
boolean... oldStyle) {
return create(varName, key, secret, null, null, target, isOldStyle(oldStyle));
String prefix, boolean... oldStyle) {
return create(varName, key, secret, null, null, target, prefix, isOldStyle(oldStyle));
}

public static KubernetesEnvBuildItem createFromResourceKey(String varName, String key, String secret,
String configmap, String target, boolean... oldStyle) {
return create(varName, key, secret, configmap, null, target, isOldStyle(oldStyle));
return create(varName, key, secret, configmap, null, target, null, isOldStyle(oldStyle));
}

public static KubernetesEnvBuildItem create(String name, String value, String secret, String configmap, String field,
String target, boolean... oldStyle) throws IllegalArgumentException {
String target, String prefix, boolean... oldStyle) throws IllegalArgumentException {
final boolean secretPresent = secret != null;
final boolean configmapPresent = configmap != null;
final boolean valuePresent = value != null;
Expand Down Expand Up @@ -134,22 +137,23 @@ public static KubernetesEnvBuildItem create(String name, String value, String se
} else {
type = EnvType.var;
}
return new KubernetesEnvBuildItem(name, value, configmap, secret, field, type, target, isOldStyle(oldStyle));
return new KubernetesEnvBuildItem(name, value, configmap, secret, field, type, target, prefix, isOldStyle(oldStyle));
}

private static boolean isOldStyle(boolean[] oldStyle) {
return oldStyle.length >= 1 && oldStyle[0];
}

KubernetesEnvBuildItem(String name, String value, String configmap, String secret, String field, EnvType type,
String target, boolean oldStyle) {
String target, String prefix, boolean oldStyle) {
this.name = name;
this.value = value;
this.configmap = configmap;
this.secret = secret;
this.field = field;
this.type = type;
this.target = target;
this.prefix = prefix;
this.oldStyle = oldStyle;
}

Expand Down Expand Up @@ -185,9 +189,13 @@ public String getTarget() {
return target;
}

public String getPrefix() {
return prefix;
}

public KubernetesEnvBuildItem newWithTarget(String newTarget) {
return new KubernetesEnvBuildItem(this.name, this.value, this.configmap, this.secret, this.field, this.type, newTarget,
this.oldStyle);
this.prefix, this.oldStyle);
}

public String toString() {
Expand Down Expand Up @@ -228,6 +236,8 @@ public boolean equals(Object o) {
return false;
if (field != null ? !field.equals(that.field) : that.field != null)
return false;
if (prefix != null ? !prefix.equals(that.prefix) : that.prefix != null)
return false;
return type == that.type;
}

Expand All @@ -239,6 +249,7 @@ public int hashCode() {
result = 31 * result + (secret != null ? secret.hashCode() : 0);
result = 31 * result + (field != null ? field.hashCode() : 0);
result = 31 * result + type.hashCode();
result = 31 * result + (prefix != null ? prefix.hashCode() : 0);
return result;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -29,10 +29,11 @@ public class KubernetesEnvBuildItemTest {
private static final String TARGET = "target";
private static final String VALUE = "value";
private static final String NAME = "name";
private static final String PREFIX = "prefix";

@Test
public void testCreateSimpleVarFromEnvConfig() {
final KubernetesEnvBuildItem item = create(NAME, VALUE, null, null, null, TARGET);
final KubernetesEnvBuildItem item = create(NAME, VALUE, null, null, null, TARGET, null);
assertEquals(var, item.getType());
assertEquals(NAME, item.getName());
assertEquals(VALUE, item.getValue());
Expand All @@ -44,12 +45,36 @@ public void testCreateSimpleVarFromEnvConfig() {

@Test
public void testCreateLoadFromConfigMapFromEnvConfig() {
final KubernetesEnvBuildItem item = create(NAME, null, null, VALUE, null, TARGET);
final KubernetesEnvBuildItem item = create(NAME, null, null, VALUE, null, TARGET, null);
assertEquals(configmap, item.getType());
assertEquals(VALUE, item.getName());
assertNull(item.getValue());
assertEquals(VALUE, item.getConfigMap());
assertNull(item.getSecret());
assertNull(item.getField());
}

@Test
public void testCreateConfigMapWithPrefix() {
final KubernetesEnvBuildItem item = create(NAME, null, null, VALUE, null, TARGET, PREFIX);
assertEquals(configmap, item.getType());
assertEquals(VALUE, item.getName());
assertNull(item.getValue());
assertEquals(VALUE, item.getConfigMap());
assertNull(item.getSecret());
assertNull(item.getField());
assertEquals(PREFIX, item.getPrefix());
}

@Test
public void testCreateSecretWithPrefix() {
final KubernetesEnvBuildItem item = create(NAME, null, VALUE, null, null, TARGET, PREFIX);
assertEquals(configmap, item.getType());
assertEquals(VALUE, item.getName());
assertNull(item.getValue());
assertEquals(VALUE, item.getSecret());
assertNull(item.getSecret());
assertNull(item.getField());
assertEquals(PREFIX, item.getPrefix());
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -144,7 +144,7 @@ public String getTargetPlatformName() {
public Collection<Env> convertToEnvs() {
return convertToBuildItems().stream()
.map(kebi -> new Env(EnvConverter.convertName(kebi.getName()), kebi.getValue(), kebi.getSecret(),
kebi.getConfigMap(), kebi.getField(), null))
kebi.getConfigMap(), kebi.getField(), null, kebi.getPrefix()))
.collect(Collectors.toList());
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -108,6 +108,7 @@ public static List<DecoratorBuildItem> createDecorators(String clusterKind,
.withSecret(e.getSecret())
.withConfigmap(e.getConfigMap())
.withField(e.getField())
.withPrefix(e.getPrefix())
.build())));
});

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,17 @@
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import io.dekorate.kubernetes.config.Env;
import io.dekorate.kubernetes.config.EnvBuilder;

public class EnvConverter {
static Pattern EXTRACT_PREFIX_PATTERN = Pattern.compile("\\[([a-zA-Z0-9-]+)]$");
static Pattern EXTRACT_NAME_PATTERN = Pattern.compile("^([a-zA-Z0-9-]+)\\[");

public static Env convert(Map.Entry<String, EnvConfig> e) {
return convert(e.getValue()).withName(convertName(e.getKey())).build();
}
Expand All @@ -25,9 +31,13 @@ private static EnvBuilder convert(EnvConfig env) {

public static List<Env> convert(EnvVarsConfig e) {
List<Env> envs = new LinkedList<>();
e.secrets.ifPresent(sl -> sl.forEach(s -> envs.add(new EnvBuilder().withName(convertName(s)).withSecret(s).build())));
e.secrets.ifPresent(sl -> sl
.forEach(s -> envs.add(new EnvBuilder().withName(convertName(extractName(s))).withSecret(s)
.withPrefix(extractPrefix(s).orElse(null)).build())));
e.configmaps
.ifPresent(cl -> cl.forEach(c -> envs.add(new EnvBuilder().withName(convertName(c)).withConfigmap(c).build())));
.ifPresent(cl -> cl.forEach(c -> envs
.add(new EnvBuilder().withName(convertName(extractName(c))).withConfigmap(c)
.withPrefix(extractPrefix(c).orElse(null)).build())));
e.vars.forEach((k, v) -> envs.add(new EnvBuilder().withName(convertName(k)).withValue(v.orElse("")).build()));
e.fields.forEach((k, v) -> {
// env vars from fields need to have their name set in addition to their field field :)
Expand All @@ -43,4 +53,20 @@ public static List<Env> convert(EnvVarsConfig e) {
public static String convertName(String name) {
return name != null ? name.toUpperCase().replace('-', '_').replace('.', '_').replace('/', '_') : null;
}

public static String extractName(String input) {
Matcher matcher = EXTRACT_NAME_PATTERN.matcher(input);
if (matcher.find()) {
return matcher.group(1);
}
return input;
}

public static Optional<String> extractPrefix(String input) {
Matcher matcher = EXTRACT_PREFIX_PATTERN.matcher(input);
if (matcher.find()) {
return Optional.of(matcher.group(1));
}
return Optional.empty();
}
}
Original file line number Diff line number Diff line change
@@ -1,7 +1,11 @@
package io.quarkus.kubernetes.deployment;

import static io.quarkus.kubernetes.deployment.EnvConverter.extractName;
import static io.quarkus.kubernetes.deployment.EnvConverter.extractPrefix;

import java.util.Collection;
import java.util.Map;
import java.util.Optional;

import io.quarkus.kubernetes.spi.KubernetesEnvBuildItem;

Expand Down Expand Up @@ -43,20 +47,24 @@ default Collection<KubernetesEnvBuildItem> convertToBuildItems() {
final String target = getTargetPlatformName();
getEnvVars().forEach((key, envConfig) -> {
validator.process(key, envConfig.value, envConfig.secret, envConfig.configmap, envConfig.field, target,
true);
Optional.empty(), true);
});

// override old-style with newer versions if present
final EnvVarsConfig c = getEnv();
c.vars.forEach((k, v) -> validator.process(KubernetesEnvBuildItem.createSimpleVar(k, v.orElse(""), target)));
c.fields.forEach((k, v) -> validator.process(KubernetesEnvBuildItem.createFromField(k, v, target)));
c.configmaps
.ifPresent(cl -> cl.forEach(cm -> validator.process(KubernetesEnvBuildItem.createFromConfigMap(cm, target))));
c.secrets.ifPresent(sl -> sl.forEach(s -> validator.process(KubernetesEnvBuildItem.createFromSecret(s, target))));
.ifPresent(
cl -> cl.forEach(cm -> validator.process(KubernetesEnvBuildItem.createFromConfigMap(extractName(cm),
target, extractPrefix(cm).orElse(null)))));
c.secrets.ifPresent(sl -> sl.forEach(s -> validator.process(KubernetesEnvBuildItem.createFromSecret(extractName(s),
target, extractPrefix(s).orElse(null)))));
c.mapping.forEach(
(varName, config) -> validator.process(KubernetesEnvBuildItem.createFromResourceKey(varName, config.withKey,
config.fromSecret.orElse(null), config.fromConfigmap.orElse(null), target)));

return validator.getBuildItems();
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -18,10 +18,10 @@ public class EnvVarValidator {
private final Set<String> errors = new HashSet<>();

void process(String name, Optional<String> value, Optional<String> secret, Optional<String> configmap,
Optional<String> field, String target, boolean... oldStyle) {
Optional<String> field, String target, Optional<String> prefix, boolean... oldStyle) {
try {
final KubernetesEnvBuildItem kebi = KubernetesEnvBuildItem.create(name, value.orElse(null),
secret.orElse(null), configmap.orElse(null), field.orElse(null), target, oldStyle);
secret.orElse(null), configmap.orElse(null), field.orElse(null), target, prefix.orElse(null), oldStyle);
process(kebi);
} catch (IllegalArgumentException e) {
errors.add(e.getMessage());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -232,6 +232,7 @@ public List<DecoratorBuildItem> createDecorators(ApplicationInfoBuildItem applic
new AddEnvVarDecorator(ApplicationContainerDecorator.ANY, name,
new EnvBuilder().withName(EnvConverter.convertName(e.getName())).withValue(e.getValue())
.withSecret(e.getSecret()).withConfigmap(e.getConfigMap()).withField(e.getField())
.withPrefix(e.getPrefix())
.build())));
});

Expand Down
Loading

0 comments on commit 364d6ac

Please sign in to comment.