All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.hyscale.deployer.services.handler.impl.V1PodHandler Maven / Gradle / Ivy

The newest version!
/**
 * Copyright 2019 Pramati Prism, Inc.
 *
 * 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.hyscale.deployer.services.handler.impl;

import java.io.IOException;
import java.io.InputStream;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.concurrent.TimeUnit;

import io.hyscale.deployer.services.model.PodStatusUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.google.common.collect.Lists;
import com.google.gson.JsonSyntaxException;
import com.google.gson.reflect.TypeToken;

import io.hyscale.commons.exception.HyscaleException;
import io.hyscale.commons.framework.events.model.ActivityState;
import io.hyscale.commons.framework.events.publisher.EventPublisher;
import io.hyscale.commons.logger.ActivityContext;
import io.hyscale.commons.logger.WorkflowLogger;
import io.hyscale.commons.models.AnnotationKey;
import io.hyscale.commons.models.ResourceLabelKey;
import io.hyscale.commons.models.ServiceMetadata;
import io.hyscale.commons.models.Status;
import io.hyscale.commons.utils.GsonProviderUtil;
import io.hyscale.commons.utils.HyscaleContextUtil;
import io.hyscale.commons.utils.ResourceSelectorUtil;
import io.hyscale.deployer.core.model.ResourceKind;
import io.hyscale.deployer.core.model.ResourceOperation;
import io.hyscale.deployer.events.model.PodStageEvent;
import io.hyscale.deployer.services.config.DeployerEnvConfig;
import io.hyscale.deployer.services.constants.DeployerConstants;
import io.hyscale.deployer.services.exception.DeployerErrorCodes;
import io.hyscale.deployer.services.factory.PodParentFactory;
import io.hyscale.deployer.services.handler.PodParentHandler;
import io.hyscale.deployer.services.handler.ResourceLifeCycleHandler;
import io.hyscale.deployer.services.model.DeployerActivity;
import io.hyscale.deployer.services.model.PodParent;
import io.hyscale.deployer.services.predicates.PodPredicates;
import io.hyscale.deployer.services.processor.PodParentProvider;
import io.hyscale.deployer.services.util.ExceptionHelper;
import io.hyscale.deployer.services.util.K8sResourcePatchUtil;
import io.kubernetes.client.PodLogs;
import io.kubernetes.client.custom.V1Patch;
import io.kubernetes.client.openapi.ApiClient;
import io.kubernetes.client.openapi.ApiException;
import io.kubernetes.client.openapi.apis.CoreV1Api;
import io.kubernetes.client.openapi.models.V1DeleteOptions;
import io.kubernetes.client.openapi.models.V1Pod;
import io.kubernetes.client.openapi.models.V1PodList;
import io.kubernetes.client.util.Watch;
import okhttp3.Call;
import okhttp3.OkHttpClient;
import okhttp3.OkHttpClient.Builder;
import okhttp3.Response;

public class V1PodHandler implements ResourceLifeCycleHandler {

    private static final Logger LOGGER = LoggerFactory.getLogger(V1PodHandler.class);
    private static final long MAX_TIME_TO_CONTAINER_READY = 120 * 1000L;
    private static final long POD_RESTART_COUNT = DeployerEnvConfig.getPodRestartCount();
    private static final Integer POD_WATCH_TIMEOUT_IN_SEC = 10;


    @Override
    public V1Pod create(ApiClient apiClient, V1Pod resource, String namespace) throws HyscaleException {
        if (resource == null) {
            LOGGER.debug("Cannot create null Pod");
            return resource;
        }
        CoreV1Api coreV1Api = new CoreV1Api(apiClient);
        String name = resource.getMetadata().getName();
        V1Pod v1Pod = null;
        try {
            resource.getMetadata().putAnnotationsItem(
                    AnnotationKey.K8S_HYSCALE_LAST_APPLIED_CONFIGURATION.getAnnotation(), GsonProviderUtil.getPrettyGsonBuilder().toJson(resource));
            v1Pod = coreV1Api.createNamespacedPod(namespace, resource, DeployerConstants.TRUE, null, null);
        } catch (ApiException e) {
            HyscaleException ex = new HyscaleException(e, DeployerErrorCodes.FAILED_TO_CREATE_RESOURCE,
                    ExceptionHelper.getExceptionMessage(getKind(), e, ResourceOperation.CREATE));
            LOGGER.error("Error while creating Pod {} in namespace {}, error {}", name, namespace, ex.toString());
            throw ex;
        }
        return v1Pod;

    }

    @Override
    public boolean update(ApiClient apiClient, V1Pod resource, String namespace) throws HyscaleException {
        if (resource == null) {
            LOGGER.debug("Cannot update null Pod");
            return false;
        }
        CoreV1Api coreV1Api = new CoreV1Api(apiClient);
        String name = resource.getMetadata().getName();
        V1Pod existingPod = null;

        try {
            existingPod = get(apiClient, name, namespace);
        } catch (HyscaleException ex) {
            LOGGER.debug("Error while getting Pod {} in namespace {} for Update, creating new", name, namespace);
            V1Pod pod = create(apiClient, resource, namespace);
            return pod != null;
        }
        try {
            String resourceVersion = existingPod.getMetadata().getResourceVersion();
            resource.getMetadata().setResourceVersion(resourceVersion);
            coreV1Api.replaceNamespacedPod(name, namespace, resource, DeployerConstants.TRUE, null, null);
        } catch (ApiException e) {
            HyscaleException ex = new HyscaleException(e, DeployerErrorCodes.FAILED_TO_UPDATE_RESOURCE,
                    ExceptionHelper.getExceptionMessage(getKind(), e, ResourceOperation.UPDATE));
            LOGGER.error("Error while updating Pod {} in namespace {}, error {}", name, namespace, ex.toString());
            throw ex;
        }

        return true;
    }

    @Override
    public V1Pod get(ApiClient apiClient, String name, String namespace) throws HyscaleException {
        CoreV1Api coreV1Api = new CoreV1Api(apiClient);
        V1Pod v1Pod = null;
        try {
            v1Pod = coreV1Api.readNamespacedPod(name, namespace, DeployerConstants.TRUE, null, null);
        } catch (ApiException e) {
            HyscaleException ex = ExceptionHelper.buildGetException(getKind(), e, ResourceOperation.GET);
            LOGGER.error("Error while fetching Pod {} in namespace {}, error {}", name, namespace, ex.toString());
            throw ex;
        }
        return v1Pod;
    }

    @Override
    public List getBySelector(ApiClient apiClient, String selector, boolean label, String namespace)
            throws HyscaleException {
        CoreV1Api coreV1Api = new CoreV1Api(apiClient);
        String labelSelector = label ? selector : null;
        String fieldSelector = label ? null : selector;
        List v1Pods = null;
        try {
            V1PodList v1PodList = coreV1Api.listNamespacedPod(namespace, DeployerConstants.TRUE, null, null, fieldSelector, labelSelector,
                    null, null, null, null);
            v1Pods = v1PodList != null ? v1PodList.getItems() : null;
        } catch (ApiException e) {
            HyscaleException ex = ExceptionHelper.buildGetException(getKind(), e, ResourceOperation.GET_BY_SELECTOR);
            LOGGER.error("Error while listing Pods in namespace {}, with selectors {},  error {}", namespace, selector,
                    ex.toString());
            throw ex;
        }
        return v1Pods;
    }

    @Override
    public List listForAllNamespaces(ApiClient apiClient, String selector, boolean label)
            throws HyscaleException {
        CoreV1Api coreV1Api = new CoreV1Api(apiClient);
        String labelSelector = label ? selector : null;
        String fieldSelector = label ? null : selector;
        List v1Pods = null;
        try {
            V1PodList v1PodList = coreV1Api.listPodForAllNamespaces(null, null, fieldSelector, labelSelector, null,
                    DeployerConstants.TRUE, null, null, null);
            v1Pods = v1PodList != null ? v1PodList.getItems() : null;
        } catch (ApiException e) {
            HyscaleException ex = ExceptionHelper.buildGetException(getKind(), e, ResourceOperation.GET_BY_SELECTOR);
            LOGGER.error("Error while listing Pods in all namespace, with selectors {},  error {}", selector,
                    ex.toString());
            throw ex;
        }
        return v1Pods;
    }

    @Override
    public boolean patch(ApiClient apiClient, String name, String namespace, V1Pod target) throws HyscaleException {
        if (target == null) {
            LOGGER.debug("Cannot patch null Pod");
            return false;
        }
        CoreV1Api coreV1Api = new CoreV1Api(apiClient);
        target.getMetadata().putAnnotationsItem(AnnotationKey.K8S_HYSCALE_LAST_APPLIED_CONFIGURATION.getAnnotation(),
                GsonProviderUtil.getPrettyGsonBuilder().toJson(target));
        V1Pod sourcePod = null;
        try {
            sourcePod = get(apiClient, name, namespace);
        } catch (HyscaleException e) {
            LOGGER.debug("Error while getting Pod {} in namespace {} for Patch, creating new", name, namespace);
            V1Pod pod = create(apiClient, target, namespace);
            return pod != null;
        }
        Object patchObject = null;
        String lastAppliedConfig = sourcePod.getMetadata().getAnnotations()
                .get(AnnotationKey.K8S_HYSCALE_LAST_APPLIED_CONFIGURATION.getAnnotation());
        try {
            patchObject = K8sResourcePatchUtil.getJsonPatch(GsonProviderUtil.getPrettyGsonBuilder().fromJson(lastAppliedConfig, V1Pod.class), target,
                    V1Pod.class);
            V1Patch v1Patch = new V1Patch(patchObject.toString());
            coreV1Api.patchNamespacedPod(name, namespace, v1Patch, DeployerConstants.TRUE, null, null, null);
        } catch (HyscaleException e) {
            LOGGER.error("Error while creating patch for Pod {}, source {}, target {}", name, sourcePod, target);
            throw e;
        } catch (ApiException e) {
            HyscaleException ex = new HyscaleException(e, DeployerErrorCodes.FAILED_TO_PATCH_RESOURCE,
                    ExceptionHelper.getExceptionMessage(getKind(), e, ResourceOperation.PATCH));
            LOGGER.error("Error while patching Pod {} in namespace {} , error {}", name, namespace, ex.toString());
            throw ex;
        }
        return true;
    }

    @Override
    public boolean delete(ApiClient apiClient, String name, String namespace, boolean wait) throws HyscaleException {
        try {
            delete(apiClient, name, namespace);
            List podList = Lists.newArrayList();
            podList.add(name);
            if (wait) {
                waitForResourceDeletion(apiClient, podList, namespace, null);
            }
        } catch (ApiException e) {
            if (e.getCode() == 404) {
                return false;
            }
            HyscaleException ex = new HyscaleException(e, DeployerErrorCodes.FAILED_TO_DELETE_RESOURCE,
                    ExceptionHelper.getExceptionMessage(getKind(), e, ResourceOperation.DELETE));
            LOGGER.error("Error while deleting Pod {} in namespace {} , error {}", name, namespace, ex.toString());
            throw ex;
        }
        return true;
    }

    private void delete(ApiClient apiClient, String name, String namespace) throws ApiException {
        CoreV1Api coreV1Api = new CoreV1Api(apiClient);
        V1DeleteOptions deleteOptions = getDeleteOptions();
        deleteOptions.setApiVersion("apps/v1");
        try {
            coreV1Api.deleteNamespacedPod(name, namespace, DeployerConstants.TRUE, null, null, null, null,
                    deleteOptions);
        } catch (JsonSyntaxException e) {
            // K8s end exception ignore
        }
    }

    @Override
    public boolean deleteBySelector(ApiClient apiClient, String selector, boolean label, String namespace, boolean wait)
            throws HyscaleException {
        try {
            List v1PodList = getBySelector(apiClient, selector, label, namespace);
            if (v1PodList == null || v1PodList.isEmpty()) {
                return false;
            }
            for (V1Pod V1Pod : v1PodList) {
                delete(apiClient, V1Pod.getMetadata().getName(), namespace, wait);
            }
        } catch (HyscaleException e) {
            if (DeployerErrorCodes.RESOURCE_NOT_FOUND.equals(e.getHyscaleError())) {
                LOGGER.error("Error while deleting Pods for selector {} in namespace {}, error {}", selector, namespace,
                        e.toString());
                return false;
            }
            throw e;
        }
        return true;
    }

    @Override
    public String getKind() {
        return ResourceKind.POD.getKind();
    }

    public InputStream tailLogs(ApiClient apiClient, String name, String namespace, Integer readLines)
            throws HyscaleException {
        return tailLogs(apiClient, name, namespace, null, name, readLines);
    }

    public InputStream tailLogs(ApiClient apiClient, String serviceName, String namespace, String podName, String containerName, Integer readLines)
            throws HyscaleException {
        if (podName == null) {
            List v1Pods = getBySelector(apiClient, ResourceLabelKey.SERVICE_NAME.getLabel() + "=" + serviceName, true,
                    namespace);
            if (v1Pods == null || v1Pods.isEmpty()) {
                throw new HyscaleException(DeployerErrorCodes.FAILED_TO_RETRIEVE_POD, serviceName, namespace);
            }
            podName = v1Pods.get(0).getMetadata().getName();
        }
        OkHttpClient existingHttpClient = apiClient.getHttpClient();
        try {
            // Update timeout
            apiClient.setHttpClient(getUpdatedHttpClient(existingHttpClient));
            PodLogs logs = new PodLogs(apiClient);
            return logs.streamNamespacedPodLog(namespace, podName, containerName, null, readLines,
                    true);
        } catch (IOException | ApiException e) {
            LOGGER.error("Failed to tail Pod logs for service {} in namespace {} ", serviceName, namespace);
            throw new HyscaleException(DeployerErrorCodes.FAILED_TO_TAIL_POD, serviceName, namespace);
        } finally {
            apiClient.setHttpClient(existingHttpClient);
        }
    }

    private OkHttpClient getUpdatedHttpClient(OkHttpClient existingHttpClient) {
        Builder newClientBuilder = existingHttpClient.newBuilder()
                .readTimeout(120, TimeUnit.MINUTES);

        return newClientBuilder.build();
    }

    public InputStream getLogs(ApiClient apiClient, String name, String namespace, Integer readLines)
            throws HyscaleException {
        return getLogs(apiClient, name, namespace, null, name, readLines);
    }

    public InputStream getLogs(ApiClient apiClient, String serviceName, String namespace, String podName, String containerName, Integer readLines)
            throws HyscaleException {
        if (podName == null) {
            List v1Pods = getBySelector(apiClient, ResourceLabelKey.SERVICE_NAME.getLabel() + "=" + serviceName, true,
                    namespace);
            if (v1Pods == null || v1Pods.isEmpty()) {
                throw new HyscaleException(DeployerErrorCodes.FAILED_TO_RETRIEVE_POD, serviceName, namespace);
            }
            podName = v1Pods.get(0).getMetadata().getName();
        }
        try {
            CoreV1Api coreClient = new CoreV1Api(apiClient);
            Call call = coreClient.readNamespacedPodLogCall(podName, namespace, containerName, Boolean.FALSE, Boolean.TRUE,
                    null,DeployerConstants.TRUE, Boolean.FALSE, null,readLines, Boolean.TRUE, null);
            Response response = call.execute();
            if (!response.isSuccessful()) {
                LOGGER.error("Failed to get Pod logs for service {} in namespace {} : {}", serviceName, namespace,
                        response.body());
                throw new HyscaleException(DeployerErrorCodes.FAILED_TO_GET_LOGS, serviceName, namespace);
            }
            return response.body().byteStream();
        } catch (IOException | ApiException e) {
            LOGGER.error("Error while fetching Pod logs for service {} in namespace {}", serviceName, namespace, e);
            throw new HyscaleException(DeployerErrorCodes.FAILED_TO_GET_LOGS, serviceName, namespace);
        }
    }

    @Override
    public boolean cleanUp() {
        return false;
    }

    @Override
    public int getWeight() {
        return ResourceKind.POD.getWeight();
    }

    /**
     * It will fetch pod parent by using of servicename and namespace, based on pod parent it will get  latestpodselector.
     * Now it will start watching pod event changes on cluster based on namespace and latestpodselector.
     * While watching pods it is reading status, here we are mainly reading three status (initialization, creations and readiness),
     * after completion of each status we are showing status done message, whenever we get any pod has failed status immediately come out from watch
     * and show deployment fail message.
     *
     * @param apiClient
     * @param appName
     * @param serviceName
     * @param namespace
     */

    public void watch(ApiClient apiClient, String appName, String serviceName, String namespace)
            throws HyscaleException {
        ServiceMetadata serviceMetadata = new ServiceMetadata();
        serviceMetadata.setAppName(appName);
        serviceMetadata.setServiceName(serviceName);
        String selector = ResourceSelectorUtil.getServiceSelector(appName, serviceName);
        PodParent podParent = HyscaleContextUtil.getSpringBean(PodParentProvider.class).getPodParent(apiClient, appName, serviceName, namespace);
        if (podParent == null) {
            LOGGER.error("Error while fetching pod parent of service {}", serviceName);
            throw new HyscaleException(DeployerErrorCodes.FAILED_TO_RETRIEVE_SERVICE_REPLICAS);
        }

        Integer replicas = null;
        String latestPodSelector = null;

        PodParentHandler podParentHandler = PodParentFactory.getHandler(podParent.getKind());
        latestPodSelector = podParentHandler.getPodSelector(apiClient, podParent.getParent(), selector);
        replicas = podParentHandler.getReplicas(podParent.getParent());

        if (replicas == 0) {
            WorkflowLogger.info(DeployerActivity.SERVICE_WITH_ZERO_REPLICAS);
            return;
        }

        if (latestPodSelector == null) {
            throw new HyscaleException(DeployerErrorCodes.FAILED_TO_RETRIEVE_SERVICE_REPLICAS);
        }
        OkHttpClient previousHttpClient = apiClient.getHttpClient();
        OkHttpClient newHttpClient = getUpdatedHttpClient(previousHttpClient);
        apiClient.setHttpClient(newHttpClient);
        try {
            watchPods(apiClient, serviceMetadata, namespace, latestPodSelector, replicas);
        } finally {
            apiClient.setHttpClient(previousHttpClient);
        }
    }

    private void watchPods(ApiClient apiClient, ServiceMetadata serviceMetadata, String namespace,
                           String latestPodSelector, Integer replicas) throws HyscaleException {
        EventPublisher publisher = EventPublisher.getInstance();
        PodStageEvent.PodStage currentStage = PodStageEvent.PodStage.INITIALIZATION;

        Set readyPods = new HashSet<>();
        Set initializedPods = new HashSet<>();
        Set createdPods = new HashSet<>();

        ActivityContext initializedActivityContext = new ActivityContext(DeployerActivity.POD_INITIALIZED);
        ActivityContext creationActivityContext = new ActivityContext(DeployerActivity.POD_CREATION);
        ActivityContext readyActivityContext = new ActivityContext(DeployerActivity.POD_READINESS);
        ActivityContext currentActivityContext = initializedActivityContext;
        boolean initializationActivityDone = false;
        boolean creationActivityStarted = false;
        boolean creationActivityDone = false;
        boolean readyActivityStarted = false;
        WorkflowLogger.startActivity(initializedActivityContext);
        Long startTime = System.currentTimeMillis();
        boolean isTimeout = true;
        while (System.currentTimeMillis() - startTime < replicas * MAX_TIME_TO_CONTAINER_READY) {
            WorkflowLogger.continueActivity(currentActivityContext);
            try (Watch watch = getWatch(apiClient, namespace, latestPodSelector)) {
                for (Watch.Response item : watch) {
                    String status = PodStatusUtil.currentStatusOf(item.object);
                    WorkflowLogger.continueActivity(currentActivityContext);
                    /*if pod  status is failed then watch will exit*/
                    if (PodPredicates.isPodFailed().test(item.object)) {
                        isTimeout = false;
                        break;
                    }

                    /*if pod restart count is reached to specified pod_restart_count then watch will exit*/
                    if (PodPredicates.isPodRestarted().test(item.object, POD_RESTART_COUNT)) {
                        isTimeout = false;
                        break;
                    }

                    if (PodPredicates.isPodInitialized().test(item.object)) {
                        initializedPods.add(item.object.getMetadata().getName());
                        /*pod initialization activity completed*/
                        if (initializedPods.size() == replicas && !initializationActivityDone) {
                            initializationActivityDone = true;
                            WorkflowLogger.endActivity(initializedActivityContext, Status.DONE);
                            publisher.publishEvent(new PodStageEvent(serviceMetadata, namespace, currentStage, status));
                        }
                    }

                    if (initializationActivityDone && !creationActivityStarted) {
                        currentActivityContext = creationActivityContext;
                        WorkflowLogger.startActivity(creationActivityContext);
                        creationActivityStarted = true;
                        currentStage = PodStageEvent.PodStage.CREATION;
                    }
                    if (PodPredicates.isPodCreated().test(item.object)) {
                        createdPods.add(item.object.getMetadata().getName());
                        /*pod creation activity completed*/
                        if (createdPods.size() == replicas && !creationActivityDone) {
                            WorkflowLogger.endActivity(creationActivityContext, Status.DONE);
                            creationActivityDone = true;
                            publisher.publishEvent(new PodStageEvent(serviceMetadata, namespace, currentStage, status));
                        }
                    }

                    if (creationActivityDone && !readyActivityStarted) {
                        currentActivityContext = readyActivityContext;
                        WorkflowLogger.startActivity(readyActivityContext);
                        readyActivityStarted = true;
                        currentStage = PodStageEvent.PodStage.READINESS;
                    }
                    if (PodPredicates.isPodReady().test(item.object)) {
                        readyPods.add(item.object.getMetadata().getName());
                        /*pod readiness activity completed*/
                        if (readyPods.size() == replicas) {
                            WorkflowLogger.endActivity(readyActivityContext, Status.DONE);
                            publisher.publishEvent(new PodStageEvent(serviceMetadata, namespace, currentStage, status));
                            isTimeout = false;
                            break;
                        }
                    }

                    if (readyPods.size() == replicas) {
                        isTimeout = false;
                        break;
                    }

                }
                if (!isTimeout) {
                    break;
                }
            } catch (IOException e) {
                LOGGER.error("Error while watching pods with selector {} in namespace {}", latestPodSelector, namespace, e);
            }
        }
        PodStageEvent event;
        if (isTimeout) {
            WorkflowLogger.endActivity(currentActivityContext, Status.FAILED);
            event = new PodStageEvent(serviceMetadata, namespace, currentStage);
            publisher.publishEvent(event);
            throw new HyscaleException(DeployerErrorCodes.TIMEOUT_WHILE_WAITING_FOR_DEPLOYMENT);
        }

        if (!initializationActivityDone) {
            event = new PodStageEvent(serviceMetadata, namespace, currentStage);
            event.setFailure(true);
            publisher.publishEvent(event);
            WorkflowLogger.endActivity(currentActivityContext, Status.FAILED);
            throw new HyscaleException(DeployerErrorCodes.FAILED_TO_INITIALIZE_POD);
        }
        if (!creationActivityDone) {
            event = new PodStageEvent(serviceMetadata, namespace, currentStage);
            event.setFailure(true);
            publisher.publishEvent(event);
            WorkflowLogger.endActivity(currentActivityContext, Status.FAILED);
            throw new HyscaleException(DeployerErrorCodes.FAILED_TO_CREATE_POD);
        }
        if (readyPods.size() != replicas) {
            event = new PodStageEvent(serviceMetadata, namespace, currentStage);
            event.setFailure(true);
            publisher.publishEvent(event);
            WorkflowLogger.endActivity(currentActivityContext, Status.FAILED);
            throw new HyscaleException(DeployerErrorCodes.POD_FAILED_READINESS);
        }

    }

    private Watch getWatch(ApiClient apiClient, String namespace, String latestPodSelector)
            throws HyscaleException {
        CoreV1Api api = new CoreV1Api(apiClient);
        try {
            return Watch.createWatch(
                    apiClient, api.listNamespacedPodCall(namespace, null, false, null, null, latestPodSelector, null,
                            null, POD_WATCH_TIMEOUT_IN_SEC, Boolean.TRUE, null),
                    new TypeToken>() {
                    }.getType());
        } catch (ApiException e) {
            LOGGER.error("Failed to watch pod events ", e);
            throw new HyscaleException(DeployerErrorCodes.FAILED_TO_RETRIEVE_POD);
        }
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy