Skip to content

Commit

Permalink
fix: ensure manually defined CRDs are considered before generated ones (
Browse files Browse the repository at this point in the history
#2662)

* fix: ensure manually defined CRDs are considered before generated ones

Fixes #2658

Signed-off-by: Chris Laprun <[email protected]>

* fix: output more details in logs on errors / failures

Signed-off-by: Chris Laprun <[email protected]>

* wip: add more logging

Signed-off-by: Chris Laprun <[email protected]>

* fix: add missing file

Signed-off-by: Chris Laprun <[email protected]>

* fix: make sure generated CRDs are applied

Signed-off-by: Chris Laprun <[email protected]>

---------

Signed-off-by: Chris Laprun <[email protected]>
  • Loading branch information
metacosm authored Jan 10, 2025
1 parent d7c72b4 commit 67993de
Show file tree
Hide file tree
Showing 9 changed files with 201 additions and 68 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -2,18 +2,18 @@

import java.io.ByteArrayInputStream;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.time.Duration;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.stream.Stream;

import org.junit.jupiter.api.extension.ExtensionContext;
Expand Down Expand Up @@ -47,7 +47,7 @@ public class LocallyRunOperatorExtension extends AbstractOperatorExtension {
private final List<LocalPortForward> localPortForwards;
private final List<Class<? extends CustomResource>> additionalCustomResourceDefinitions;
private final Map<Reconciler, RegisteredController> registeredControllers;
private final List<String> additionalCrds;
private final Map<String, String> crdMappings;

private LocallyRunOperatorExtension(
List<ReconcilerSpec> reconcilers,
Expand Down Expand Up @@ -82,7 +82,24 @@ private LocallyRunOperatorExtension(
: overrider -> overrider.withKubernetesClient(kubernetesClient);
this.operator = new Operator(configurationServiceOverrider);
this.registeredControllers = new HashMap<>();
this.additionalCrds = additionalCrds;
crdMappings = getAdditionalCRDsFromFiles(additionalCrds, getKubernetesClient());
}

static Map<String, String> getAdditionalCRDsFromFiles(Iterable<String> additionalCrds,
KubernetesClient client) {
Map<String, String> crdMappings = new HashMap<>();
additionalCrds.forEach(p -> {
try (InputStream is = new FileInputStream(p)) {
client.load(is).items().stream()
// only consider CRDs to avoid applying random resources to the cluster
.filter(CustomResourceDefinition.class::isInstance)
.map(CustomResourceDefinition.class::cast)
.forEach(crd -> crdMappings.put(crd.getMetadata().getName(), p));
} catch (Exception e) {
throw new RuntimeException("Couldn't load CRD at " + p, e);
}
});
return crdMappings;
}

/**
Expand Down Expand Up @@ -112,25 +129,18 @@ public static void applyCrd(Class<? extends HasMetadata> resourceClass, Kubernet
public static void applyCrd(String resourceTypeName, KubernetesClient client) {
String path = "/META-INF/fabric8/" + resourceTypeName + "-v1.yml";
try (InputStream is = LocallyRunOperatorExtension.class.getResourceAsStream(path)) {
applyCrd(is, path, client);
} catch (IllegalStateException e) {
// rethrow directly
throw e;
if (is == null) {
throw new IllegalStateException("Cannot find CRD at " + path);
}
var crdString = new String(is.readAllBytes(), StandardCharsets.UTF_8);
applyCrd(crdString, path, client);
} catch (IOException e) {
throw new IllegalStateException("Cannot apply CRD yaml: " + path, e);
}
}

public static void applyCrd(CustomResourceDefinition crd, KubernetesClient client) {
client.resource(crd).serverSideApply();
}

private static void applyCrd(InputStream is, String path, KubernetesClient client) {
private static void applyCrd(String crdString, String path, KubernetesClient client) {
try {
if (is == null) {
throw new IllegalStateException("Cannot find CRD at " + path);
}
var crdString = new String(is.readAllBytes(), StandardCharsets.UTF_8);
LOGGER.debug("Applying CRD: {}", crdString);
final var crd = client.load(new ByteArrayInputStream(crdString.getBytes()));
crd.serverSideApply();
Expand All @@ -144,14 +154,42 @@ private static void applyCrd(InputStream is, String path, KubernetesClient clien
}
}

public static List<CustomResourceDefinition> parseCrds(String path, KubernetesClient client) {
try (InputStream is = new FileInputStream(path)) {
return client.load(new ByteArrayInputStream(is.readAllBytes()))
.items().stream().map(i -> (CustomResourceDefinition) i).collect(Collectors.toList());
} catch (FileNotFoundException e) {
throw new RuntimeException(e);
} catch (IOException e) {
throw new RuntimeException(e);
/**
* Applies the CRD associated with the specified custom resource, first checking if a CRD has been
* manually specified using {@link Builder#withAdditionalCRD}, otherwise assuming that its CRD
* should be found in the standard location as explained in
* {@link LocallyRunOperatorExtension#applyCrd(String, KubernetesClient)}
*
* @param crClass the custom resource class for which we want to apply the CRD
*/
public void applyCrd(Class<? extends CustomResource> crClass) {
applyCrd(ReconcilerUtils.getResourceTypeName(crClass));
}

/**
* Applies the CRD associated with the specified resource type name, first checking if a CRD has
* been manually specified using {@link Builder#withAdditionalCRD}, otherwise assuming that its
* CRD should be found in the standard location as explained in
* {@link LocallyRunOperatorExtension#applyCrd(String, KubernetesClient)}
*
* @param resourceTypeName the resource type name associated with the CRD to be applied,
* typically, given a resource type, its name would be obtained using
* {@link ReconcilerUtils#getResourceTypeName(Class)}
*/
public void applyCrd(String resourceTypeName) {
// first attempt to use a manually defined CRD
final var pathAsString = crdMappings.get(resourceTypeName);
if (pathAsString != null) {
final var path = Path.of(pathAsString);
try {
applyCrd(Files.readString(path), pathAsString, getKubernetesClient());
} catch (IOException e) {
throw new IllegalStateException("Cannot open CRD file at " + path.toAbsolutePath(), e);
}
crdMappings.remove(resourceTypeName);
} else {
// if no manually defined CRD matches the resource type, apply the generated one
applyCrd(resourceTypeName, getKubernetesClient());
}
}

Expand All @@ -160,7 +198,7 @@ private Stream<Reconciler> reconcilers() {
}

public List<Reconciler> getReconcilers() {
return reconcilers().collect(Collectors.toUnmodifiableList());
return reconcilers().toList();
}

public Reconciler getFirstReconciler() {
Expand Down Expand Up @@ -207,7 +245,6 @@ protected void before(ExtensionContext context) {
}

additionalCustomResourceDefinitions.forEach(this::applyCrd);
Map<String, CustomResourceDefinition> unappliedCRDs = getAdditionalCRDsFromFiles();
for (var ref : reconcilers) {
final var config = operator.getConfigurationService().getConfigurationFor(ref.reconciler);
final var oconfig = override(config);
Expand All @@ -227,49 +264,28 @@ protected void before(ExtensionContext context) {
final var resourceTypeName = ReconcilerUtils.getResourceTypeName(resourceClass);
// only try to apply a CRD for the reconciler if it is associated to a CR
if (CustomResource.class.isAssignableFrom(resourceClass)) {
if (unappliedCRDs.get(resourceTypeName) != null) {
applyCrd(resourceTypeName);
unappliedCRDs.remove(resourceTypeName);
} else {
applyCrd(resourceClass);
}
applyCrd(resourceTypeName);
}

// apply yet unapplied CRDs
var registeredController = this.operator.register(ref.reconciler, oconfig.build());
registeredControllers.put(ref.reconciler, registeredController);
}
unappliedCRDs.keySet().forEach(this::applyCrd);
crdMappings.forEach((crdName, path) -> {
final String crdString;
try {
crdString = Files.readString(Path.of(path));
} catch (IOException e) {
throw new IllegalArgumentException("Couldn't read CRD located at " + path, e);
}
applyCrd(crdString, path, getKubernetesClient());
});
crdMappings.clear();

LOGGER.debug("Starting the operator locally");
this.operator.start();
}

private Map<String, CustomResourceDefinition> getAdditionalCRDsFromFiles() {
Map<String, CustomResourceDefinition> crdMappings = new HashMap<>();
additionalCrds.forEach(p -> {
var crds = parseCrds(p, getKubernetesClient());
crds.forEach(c -> crdMappings.put(c.getMetadata().getName(), c));
});
return crdMappings;
}

/**
* Applies the CRD associated with the specified custom resource, first checking if a CRD has been
* manually specified using {@link Builder#withAdditionalCRD(String)}, otherwise assuming that its
* CRD should be found in the standard location as explained in
* {@link LocallyRunOperatorExtension#applyCrd(String, KubernetesClient)}
*
* @param crClass the custom resource class for which we want to apply the CRD
*/
public void applyCrd(Class<? extends CustomResource> crClass) {
applyCrd(ReconcilerUtils.getResourceTypeName(crClass));
}

public void applyCrd(String resourceTypeName) {
applyCrd(resourceTypeName, getKubernetesClient());
}

@Override
protected void after(ExtensionContext context) {
super.after(context);
Expand All @@ -295,7 +311,6 @@ public static class Builder extends AbstractBuilder<Builder> {
private final List<ReconcilerSpec> reconcilers;
private final List<PortForwardSpec> portForwards;
private final List<Class<? extends CustomResource>> additionalCustomResourceDefinitions;
private final Map<String, String> crdMappings;
private final List<String> additionalCRDs = new ArrayList<>();
private KubernetesClient kubernetesClient;

Expand All @@ -304,7 +319,6 @@ protected Builder() {
this.reconcilers = new ArrayList<>();
this.portForwards = new ArrayList<>();
this.additionalCustomResourceDefinitions = new ArrayList<>();
this.crdMappings = new HashMap<>();
}

public Builder withReconciler(
Expand Down Expand Up @@ -359,8 +373,10 @@ public Builder withAdditionalCustomResourceDefinition(
return this;
}

public Builder withAdditionalCRD(String path) {
additionalCRDs.add(path);
public Builder withAdditionalCRD(String... paths) {
if (paths != null) {
additionalCRDs.addAll(List.of(paths));
}
return this;
}

Expand Down
21 changes: 21 additions & 0 deletions operator-framework-junit5/src/test/crd/test.crd
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
name: externals.crd.example
spec:
group: crd.example
names:
kind: External
singular: external
plural: externals
scope: Namespaced
versions:
- name: v1
schema:
openAPIV3Schema:
properties:
foo:
type: "string"
type: "object"
served: true
storage: true
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
package io.javaoperatorsdk.operator.junit;

import java.nio.file.Path;
import java.util.List;

import org.junit.jupiter.api.Test;

import io.fabric8.kubernetes.client.KubernetesClientBuilder;

import static org.junit.jupiter.api.Assertions.*;

class LocallyRunOperatorExtensionTest {

@Test
void getAdditionalCRDsFromFiles() {
System.out.println(Path.of("").toAbsolutePath());
System.out.println(Path.of("src/test/crd/test.crd").toAbsolutePath());
final var crds = LocallyRunOperatorExtension.getAdditionalCRDsFromFiles(
List.of("src/test/resources/crd/test.crd", "src/test/crd/test.crd"),
new KubernetesClientBuilder().build());
assertNotNull(crds);
assertEquals(2, crds.size());
assertEquals("src/test/crd/test.crd", crds.get("externals.crd.example"));
assertEquals("src/test/resources/crd/test.crd", crds.get("tests.crd.example"));
}
}
19 changes: 19 additions & 0 deletions operator-framework-junit5/src/test/resources/crd/test.crd
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
name: tests.crd.example
spec:
group: crd.example
names:
kind: Test
singular: test
plural: tests
scope: Namespaced
versions:
- name: v1
schema:
openAPIV3Schema:
properties:
type: "object"
served: true
storage: true
16 changes: 16 additions & 0 deletions operator-framework-junit5/src/test/resources/log4j2.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
<?xml version="1.0" encoding="UTF-8"?>
<Configuration name="TestConfig" status="WARN">
<Appenders>
<Console name="Console" target="SYSTEM_OUT">
<PatternLayout pattern="%d %threadId %-30c{1.} [%-5level] %msg%n%throwable"/>
</Console>
</Appenders>
<Loggers>
<Logger level="debug" name="io.javaoperatorsdk.operator" additivity="false">
<AppenderRef ref="Console"/>
</Logger>
<Root level="info">
<AppenderRef ref="Console"/>
</Root>
</Loggers>
</Configuration>
19 changes: 19 additions & 0 deletions operator-framework/src/test/crd/test.crd
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
name: externals.crd.example
spec:
group: crd.example
names:
kind: External
singular: external
plural: externals
scope: Namespaced
versions:
- name: v1
schema:
openAPIV3Schema:
properties:
type: "object"
served: true
storage: true
Original file line number Diff line number Diff line change
Expand Up @@ -28,16 +28,22 @@ public class CRDMappingInTestExtensionIT {
LocallyRunOperatorExtension operator =
LocallyRunOperatorExtension.builder()
.withReconciler(new TestReconciler())
.withAdditionalCRD("src/test/resources/crd/test.crd")
.withAdditionalCRD("src/test/resources/crd/test.crd", "src/test/crd/test.crd")
.build();

@Test
void correctlyAppliesManuallySpecifiedCRD() {
operator.applyCrd(TestCR.class);

final var crdClient = client.apiextensions().v1().customResourceDefinitions();
await().pollDelay(Duration.ofMillis(150))
.untilAsserted(() -> assertThat(crdClient.withName("tests.crd.example").get()).isNotNull());
.untilAsserted(() -> {
final var actual = crdClient.withName("tests.crd.example").get();
assertThat(actual).isNotNull();
assertThat(actual.getSpec().getVersions().get(0).getSchema().getOpenAPIV3Schema()
.getProperties().containsKey("foo")).isTrue();
});
await().pollDelay(Duration.ofMillis(150))
.untilAsserted(
() -> assertThat(crdClient.withName("externals.crd.example").get()).isNotNull());
}

@Group("crd.example")
Expand Down
4 changes: 3 additions & 1 deletion operator-framework/src/test/resources/crd/test.crd
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,8 @@ spec:
schema:
openAPIV3Schema:
properties:
foo:
type: "string"
type: "object"
served: true
storage: true
storage: true
Loading

0 comments on commit 67993de

Please sign in to comment.