io.dekorate.testing.kubernetes.KubernetesExtension Maven / Gradle / Ivy
/**
* Copyright 2018 The original 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 io.dekorate.testing.kubernetes;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;
import org.junit.jupiter.api.extension.AfterAllCallback;
import org.junit.jupiter.api.extension.BeforeAllCallback;
import org.junit.jupiter.api.extension.ConditionEvaluationResult;
import org.junit.jupiter.api.extension.ExecutionCondition;
import org.junit.jupiter.api.extension.ExtensionContext;
import io.dekorate.Logger;
import io.dekorate.LoggerFactory;
import io.dekorate.deps.kubernetes.api.model.HasMetadata;
import io.dekorate.deps.kubernetes.api.model.KubernetesList;
import io.dekorate.deps.kubernetes.api.model.Pod;
import io.dekorate.deps.kubernetes.api.model.ReplicationController;
import io.dekorate.deps.kubernetes.api.model.apps.Deployment;
import io.dekorate.deps.kubernetes.api.model.apps.ReplicaSet;
import io.dekorate.deps.kubernetes.client.KubernetesClient;
import io.dekorate.deps.kubernetes.client.VersionInfo;
import io.dekorate.deps.kubernetes.client.internal.readiness.Readiness;
import io.dekorate.deps.openshift.api.model.DeploymentConfig;
import io.dekorate.hook.OrderedHook;
import io.dekorate.kubernetes.config.KubernetesConfig;
import io.dekorate.kubernetes.hook.DockerBuildHook;
import io.dekorate.kubernetes.hook.DockerPushHook;
import io.dekorate.testing.WithEvents;
import io.dekorate.testing.WithKubernetesClient;
import io.dekorate.testing.WithPod;
import io.dekorate.testing.WithProject;
import io.dekorate.testing.config.KubernetesIntegrationTestConfig;
import io.dekorate.utils.Strings;
public class KubernetesExtension implements ExecutionCondition, BeforeAllCallback, AfterAllCallback,
WithKubernetesIntegrationTestConfig, WithPod, WithKubernetesClient, WithKubernetesResources, WithEvents, WithProject, WithKubernetesConfig {
private final Logger LOGGER = LoggerFactory.getLogger();
@Override
public ConditionEvaluationResult evaluateExecutionCondition(ExtensionContext context) {
try {
VersionInfo version = getKubernetesClient(context).getVersion();
String message = "Found version:" + version;
LOGGER.info(message);
return ConditionEvaluationResult.enabled(message);
} catch (Throwable t) {
String reason = "Could not communicate with KubernetesExtension API server.";
LOGGER.error(reason);
return ConditionEvaluationResult.disabled(reason);
}
}
@Override
public void beforeAll(ExtensionContext context) throws Exception {
KubernetesIntegrationTestConfig config = getKubernetesIntegrationTestConfig(context);
KubernetesClient client = getKubernetesClient(context);
KubernetesList list = getKubernetesResources(context);
if (hasKubernetesConfig()) {
KubernetesConfig kubernetesConfig = getKubernetesConfig();
//
// We use the isAutoPushEnabled flag of the @KubernetesApplication annotation and not @KubernetesIntegrationTest.
// The reason is that the @KubernetesApplication.isAutoPushEnabled() affects the generated manifests (adds the registry).
// and thus the tests MUST follow.
if (kubernetesConfig.isAutoPushEnabled()) {
DockerBuildHook buildHook = new DockerBuildHook(getProject(), kubernetesConfig);
DockerPushHook pushHook = new DockerPushHook(getProject(), kubernetesConfig);
OrderedHook hook = OrderedHook.create(buildHook, pushHook);
hook.run();
} else if (kubernetesConfig.isAutoBuildEnabled()) {
DockerBuildHook build = new DockerBuildHook(getProject(), kubernetesConfig);
build.run();
} else if (config.isBuildEnabled()) {
DockerBuildHook build = new DockerBuildHook(getProject(), kubernetesConfig);
build.run();
}
}
if (config.isDeployEnabled()) {
list.getItems().stream()
.forEach(i -> {
client.resourceList(i).createOrReplace();
LOGGER.info("Created: " + i.getKind() + " name:" + i.getMetadata().getName() + ".");
});
List waitables = list.getItems().stream().filter(i->
i instanceof Deployment ||
i instanceof DeploymentConfig ||
i instanceof Pod ||
i instanceof ReplicaSet ||
i instanceof ReplicationController).collect(Collectors.toList());
long started = System.currentTimeMillis();
LOGGER.info("Waiting until ready ("+config.getReadinessTimeout()+ " ms)...");
waitUntilCondition(context, waitables, i -> Readiness.isReady(i), config.getReadinessTimeout(), TimeUnit.MILLISECONDS);
long ended = System.currentTimeMillis();
LOGGER.info("Waited: " + (ended-started)+ " ms.");
//Display the item status
waitables.stream().map(r->client.resource(r).fromServer().get())
.forEach(i -> {
if (!Readiness.isReady(i)) {
LOGGER.warning(i.getKind() + ":" + i.getMetadata().getName() + " not ready!");
getEvents(context, i).getItems().stream().forEach(e -> {
if (Strings.isNotNullOrEmpty(e.getMessage())) {
LOGGER.warning(e.getMessage());
}
});
}
});
}
}
@Override
public void postProcessTestInstance(Object testInstance, ExtensionContext context) throws Exception {
Arrays.stream( testInstance.getClass().getDeclaredFields())
.forEach(f -> {
injectKubernetesClient(context, testInstance, f);
injectKubernetesResources(context, testInstance, f);
injectPod(context, testInstance, f);
});
}
@Override
public void afterAll(ExtensionContext context) {
getKubernetesResources(context).getItems().stream().forEach(r -> {
LOGGER.info("Deleting: " + r.getKind() + " name:" +r.getMetadata().getName()+ " status:"+ getKubernetesClient(context).resource(r).cascading(true).delete());
});
}
/**
* Returns the configured name.
* @return The name.
*/
public String getName() {
return getKubernetesConfig().getName();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy