io.dekorate.testing.WithKubernetesClient 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;
import static io.dekorate.testing.Testing.DEKORATE_STORE;
import static java.util.Arrays.stream;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.concurrent.TimeUnit;
import java.util.function.Predicate;
import java.util.stream.Collectors;
import org.junit.jupiter.api.extension.ExtensionContext;
import org.junit.jupiter.api.extension.TestInstancePostProcessor;
import io.dekorate.DekorateException;
import io.fabric8.kubernetes.api.model.HasMetadata;
import io.fabric8.kubernetes.client.BaseClient;
import io.fabric8.kubernetes.client.DefaultKubernetesClient;
import io.fabric8.kubernetes.client.KubernetesClient;
import io.fabric8.kubernetes.client.KubernetesClientException;
/**
* Mixin for storing the kubernetes client into the context.
* It also provides functionality for injecting the client.
*/
public interface WithKubernetesClient extends TestInstancePostProcessor {
String KUBERNETES_CLIENT = "KUBERNETES_CLIENT";
default void postProcessTestInstance(Object testInstance, ExtensionContext context) throws Exception {
stream(testInstance.getClass().getDeclaredFields())
.forEach(f -> injectKubernetesClient(context, testInstance, f));
}
/**
* Inject an instance of {@link KubernetesClient} to the specified {@link Field}.
*
* @param context The execution context.
* @param testInstance The target test instance.
* @param field The field to inject.
*/
default void injectKubernetesClient(ExtensionContext context, Object testInstance, Field field) {
if (!field.getType().isAssignableFrom(KubernetesClient.class)) {
return;
}
//This is to make sure we don't write on fields by accident.
if (!stream(field.getDeclaredAnnotations()).filter(a -> a.annotationType().getSimpleName().equalsIgnoreCase("Inject"))
.findAny().isPresent()) {
return;
}
field.setAccessible(true);
try {
field.set(testInstance, getKubernetesClient(context));
} catch (IllegalAccessException e) {
throw DekorateException.launderThrowable(e);
}
}
/**
* Gets or creates an instance of {@link KubernetesClient} from the {@link ExtensionContext}.
*
* @param context The context.
* @return An instance of the client.
*/
default KubernetesClient getKubernetesClient(ExtensionContext context) {
Object client = context.getStore(DEKORATE_STORE).get(KUBERNETES_CLIENT);
if (client instanceof KubernetesClient) {
return (KubernetesClient) client;
}
client = new DefaultKubernetesClient();
context.getStore(DEKORATE_STORE).put(KUBERNETES_CLIENT, client);
return (KubernetesClient) client;
}
default void closeKubernetesClient(ExtensionContext context) {
Object client = context.getStore(DEKORATE_STORE).remove(KUBERNETES_CLIENT);
if (client instanceof KubernetesClient) {
((BaseClient) client).close();
}
}
/**
* Wait until the specified resources satisfy the specified predicate.
* Workaround for https://github.com/fabric8io/kubernetes-client/issues/1607.
*
* @param context The context.
* @param items The items.
* @param condition The condition.
* @param amount The amount of time to wait.
* @param timeUnit The time unit of the amount of time to wait.
* @return true if condition was met.
*/
default boolean waitUntilCondition(ExtensionContext context, Collection items,
Predicate condition, long amount, TimeUnit timeUnit) throws InterruptedException {
long amountInNanos = timeUnit.toNanos(amount);
long end = System.nanoTime() + amountInNanos;
KubernetesClient client = getKubernetesClient(context);
List notReady = items.stream().map(i -> client.resource(i).fromServer().get()).filter(condition.negate())
.collect(Collectors.toList());
while (System.nanoTime() < end && !notReady.isEmpty()) {
Thread.sleep(1000);
notReady = items.stream().map(i -> client.resource(i).fromServer().get()).filter(condition.negate())
.collect(Collectors.toList());
}
return notReady.isEmpty();
}
default boolean deleteAndWait(ExtensionContext context, T item, long amount, TimeUnit timeUnit)
throws InterruptedException {
return deleteAndWait(context, Arrays.asList(item), amount, timeUnit);
}
default boolean deleteAndWait(ExtensionContext context, Collection items, long amount,
TimeUnit timeUnit) throws InterruptedException {
long amountInNanos = timeUnit.toNanos(amount);
long end = System.nanoTime() + amountInNanos;
KubernetesClient client = getKubernetesClient(context);
for (T item : items) {
try {
T existing = client.resource(item).fromServer().get();
if (existing != null) {
client.resource(existing).delete();
}
} catch (KubernetesClientException e) {
if (e.getCode() != 404) {
throw e;
}
}
}
List notDeleted = new ArrayList<>(items);
while (System.nanoTime() < end && !notDeleted.isEmpty()) {
for (T item : items) {
if (!notDeleted.contains(item)) {
continue;
}
try {
if (client.resource(item).fromServer().get() == null) {
notDeleted.remove(item);
}
} catch (KubernetesClientException e) {
if (e.getCode() == 404) {
notDeleted.remove(item);
}
}
Thread.sleep(1000);
}
}
return notDeleted.isEmpty();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy