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

io.hyscale.controller.validator.impl.VolumeValidator Maven / Gradle / Ivy

There is a newer version: 1.0.0
Show 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.controller.validator.impl;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.Predicate;
import java.util.stream.Collectors;

import io.hyscale.commons.io.StructuredOutputHandler;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import com.fasterxml.jackson.core.type.TypeReference;

import io.hyscale.commons.constants.K8SRuntimeConstants;
import io.hyscale.commons.constants.ToolConstants;
import io.hyscale.commons.exception.HyscaleException;
import io.hyscale.commons.logger.LoggerTags;
import io.hyscale.commons.logger.WorkflowLogger;
import io.hyscale.commons.models.AnnotationKey;
import io.hyscale.commons.models.K8sAuthorisation;
import io.hyscale.commons.models.StorageClassAnnotation;
import io.hyscale.commons.utils.HyscaleStringUtil;
import io.hyscale.commons.utils.ResourceSelectorUtil;
import io.hyscale.commons.validator.Validator;
import io.hyscale.controller.activity.ValidatorActivity;
import io.hyscale.controller.model.WorkflowContext;
import io.hyscale.deployer.core.model.ResourceKind;
import io.hyscale.deployer.services.exception.DeployerErrorCodes;
import io.hyscale.deployer.services.handler.ResourceHandlers;
import io.hyscale.deployer.services.handler.impl.V1PersistentVolumeClaimHandler;
import io.hyscale.deployer.services.handler.impl.V1StorageClassHandler;
import io.hyscale.deployer.services.model.DeployerActivity;
import io.hyscale.deployer.services.provider.K8sClientProvider;
import io.hyscale.deployer.services.util.KubernetesVolumeUtil;
import io.hyscale.servicespec.commons.fields.HyscaleSpecFields;
import io.hyscale.servicespec.commons.model.service.ServiceSpec;
import io.hyscale.servicespec.commons.model.service.Volume;
import io.kubernetes.client.custom.Quantity;
import io.kubernetes.client.openapi.ApiClient;
import io.kubernetes.client.openapi.models.V1PersistentVolumeClaim;
import io.kubernetes.client.openapi.models.V1ResourceRequirements;
import io.kubernetes.client.openapi.models.V1StorageClass;

/**
 * Validate Volumes:
 * Storage Class should be valid
 *      persist error message and returns false
 * Size and Storage class modification 
 *      persist warn message and returns true
 * 
 * @author tushar
 *
 */
@Component
public class VolumeValidator implements Validator{
	private static final Logger logger = LoggerFactory.getLogger(VolumeValidator.class);
	
	private static final String STORAGE = "storage";

	@Autowired
	private K8sClientProvider clientProvider;

	@Autowired
	private StructuredOutputHandler outputHandler;

	private List storageClassList = new ArrayList<>();

	@Override
	public boolean validate(WorkflowContext context) throws HyscaleException {
		logger.debug("Validating volumes from the service spec");
		ServiceSpec serviceSpec = context.getServiceSpec();
		if (serviceSpec == null) {
			return false;
		}
		long startTime = System.currentTimeMillis();
		TypeReference> volTypeRef = new TypeReference>() {
		};
		List volumeList = serviceSpec.get(HyscaleSpecFields.volumes, volTypeRef);

		if (volumeList == null || volumeList.isEmpty()) {
			return true;
		}
		ApiClient apiClient = clientProvider.get((K8sAuthorisation) context.getAuthConfig());
		
		try {
		    if (!initStorageClass(apiClient)) {
		        return isValid(true, startTime);
		    }
		} catch (HyscaleException ex) {
		    throw ex;
		}
		
		// Validate Storage class
		if (!validateStorageClass(volumeList)) {
		    return isValid(true, startTime);
		}

		logger.debug("Storage class provided are valid");

		// Validate volume edit
		checkForVolumeEdit(apiClient, context, volumeList);
		return isValid(false, startTime);
	}
	
	private boolean isValid(boolean isFailed, long startTime) {
	    logger.debug("Time taken: {}", System.currentTimeMillis() - startTime);
	    if (isFailed) {
            return false;
	    }
        return true;
	}
	

	/**
	 * Validate Storage class
	 * Checks for the following conditions in the cluster and fails when
	 * 
    *
  1. There are no storage classes
  2. *
  3. No storage class is defined & default storage class does not exist in the cluster
  4. *
  5. Provided storage class does not exist in cluster
  6. *
* * @param volumeList */ private boolean validateStorageClass(List volumeList) { List defaultStorageClass = getDefaultStorageClass(); Set storageClassAllowed = storageClassList.stream().map(each -> each.getMetadata().getName()) .collect(Collectors.toSet()); logger.debug("Allowed Storage classes are : {}", storageClassAllowed); boolean isFailed = false; DeployerErrorCodes errorCode = null; StringBuilder failMsgBuilder = new StringBuilder(); for (Volume volume : volumeList) { String storageClass = volume.getStorageClass(); if (storageClass == null && isInvalidDefaultStorageClass(defaultStorageClass)) { isFailed = true; if (defaultStorageClass == null || defaultStorageClass.isEmpty()) { errorCode = DeployerErrorCodes.MISSING_DEFAULT_STORAGE_CLASS; } else { errorCode = DeployerErrorCodes.MISSING_UNIQUE_DEFAULT_STORAGE_CLASS; } failMsgBuilder.append(volume.getName()); failMsgBuilder.append(ToolConstants.COMMA); } if (storageClass != null && !storageClassAllowed.contains(storageClass)) { isFailed = true; errorCode = DeployerErrorCodes.INVALID_STORAGE_CLASS_FOR_VOLUME; failMsgBuilder.append(storageClass).append(" in volume ").append(volume.getName()); failMsgBuilder.append(ToolConstants.COMMA); } } if (isFailed) { String failMsg = HyscaleStringUtil.removeSuffixStr(failMsgBuilder, ToolConstants.COMMA); failMsg = new HyscaleException(errorCode, failMsg, storageClassAllowed.toString()).getMessage(); WorkflowLogger.persist(ValidatorActivity.VOLUME_VALIDATION_FAILED, LoggerTags.ERROR, failMsg); if(WorkflowLogger.isDisabled()){ outputHandler.addErrorMessage(ValidatorActivity.VOLUME_VALIDATION_FAILED.getActivityMessage(),failMsg); } logger.info("StroageClass validation failed. Message : {}", failMsg); return false; } return true; } private boolean isInvalidDefaultStorageClass(List defaultStorageClass) { if (defaultStorageClass == null || defaultStorageClass.isEmpty() || defaultStorageClass.size() > 1) { return true; } return false; } /** * Validate volume edit is supported, *

print warn message for size and storage class changes * Get all pvc for this service and app * Create map of volume name to pvc * For volumes check existing values through pvcs * @param apiClient * @param context * @param volumeList */ private void checkForVolumeEdit(ApiClient apiClient, WorkflowContext context, List volumeList) { V1PersistentVolumeClaimHandler pvcHandler = (V1PersistentVolumeClaimHandler) ResourceHandlers .getHandlerOf(ResourceKind.PERSISTENT_VOLUME_CLAIM.getKind()); String appName = context.getAppName(); String envName = context.getEnvName(); String serviceName = context.getServiceName(); String namespace = context.getNamespace(); String selector = ResourceSelectorUtil.getSelector(appName, envName, serviceName); List pvcList = null; try { pvcList = pvcHandler.getBySelector(apiClient, selector, true, namespace); } catch (HyscaleException ex) { logger.debug("Error while fetching PVCs with selector: {}, error: {}", selector, ex.getMessage()); } if (pvcList != null && !pvcList.isEmpty()) { Map volumeVsPVC = pvcList.stream().collect( Collectors.toMap(pvc -> KubernetesVolumeUtil.getVolumeName(pvc), pvc -> pvc, (key1, key2) -> { return key1; })); StringBuilder warnMsgBuilder = new StringBuilder(); volumeList.stream().forEach(volume -> { V1PersistentVolumeClaim pvc = volumeVsPVC.get(volume.getName()); if (pvc == null) { // new volume does not exist on cluster return; } String storageClass = pvc.getSpec().getStorageClassName(); if (StringUtils.isBlank(storageClass)) { logger.debug("Storage class not found in spec, getting from annotation"); storageClass = pvc.getMetadata().getAnnotations().get(AnnotationKey.K8S_STORAGE_CLASS.getAnnotation()); } Quantity existingSize = pvc.getStatus().getCapacity() != null ? pvc.getStatus().getCapacity().get(STORAGE) : null; if (existingSize == null) { logger.debug("Size not found in status, getting from spec"); V1ResourceRequirements resources = pvc.getSpec().getResources(); existingSize = resources != null ? (resources.getRequests() != null ? resources.getRequests().get(STORAGE) : null) : null; } Quantity newSize = Quantity.fromString(StringUtils.isNotBlank(volume.getSize()) ? volume.getSize() : K8SRuntimeConstants.DEFAULT_VOLUME_SIZE); boolean isStorageClassSame = matchStorageClass(storageClass, volume.getStorageClass()); boolean isSizeSame = newSize.equals(existingSize); if (!isStorageClassSame || !isSizeSame) { warnMsgBuilder.append(volume.getName()).append(ToolConstants.COMMA).append(ToolConstants.SPACE); } }); String warnMsg = warnMsgBuilder.toString(); if (StringUtils.isNotBlank(warnMsg)) { warnMsg = HyscaleStringUtil.removeSuffixStr(warnMsg, ToolConstants.COMMA + ToolConstants.SPACE); logger.debug(DeployerActivity.IGNORING_VOLUME_MODIFICATION.getActivityMessage(), serviceName, warnMsg); WorkflowLogger.persist(DeployerActivity.IGNORING_VOLUME_MODIFICATION, serviceName, warnMsg); } } } private Predicate isDefaultStorageClass() { return v1StorageClass -> { Map annotations = v1StorageClass.getMetadata().getAnnotations(); String annotationValue = StorageClassAnnotation.getDefaultAnnotaionValue(annotations); if (StringUtils.isNotBlank(annotationValue)) { return Boolean.valueOf(annotationValue); } return false; }; } private List getDefaultStorageClass() { List storageClasses = new ArrayList(); if (storageClassList != null && !storageClassList.isEmpty()) { storageClasses = storageClassList.stream().filter(isDefaultStorageClass()) .map(each -> each.getMetadata().getName()).collect(Collectors.toList()); } return storageClasses; } private boolean initStorageClass(ApiClient apiClient) throws HyscaleException { if (storageClassList != null && !storageClassList.isEmpty()) { return true; } V1StorageClassHandler storageClassHandler = (V1StorageClassHandler) ResourceHandlers .getHandlerOf(ResourceKind.STORAGE_CLASS.getKind()); // Storage class are cluster based no need of selector and namespace try { storageClassList = storageClassHandler.getAll(apiClient); } catch (HyscaleException ex) { logger.error("Error while getting storage class list, error {}", ex.getMessage()); WorkflowLogger.persist(ValidatorActivity.VOLUME_VALIDATION_FAILED, LoggerTags.ERROR, DeployerErrorCodes.NO_STORAGE_CLASS_IN_K8S.getMessage()); throw new HyscaleException(DeployerErrorCodes.NO_STORAGE_CLASS_IN_K8S); } if (storageClassList == null || storageClassList.isEmpty()) { return false; } return true; } private boolean matchStorageClass(String existing, String modified) { if (StringUtils.isBlank(modified)) { List defaultStorageClassList = getDefaultStorageClass(); return (existing != null && defaultStorageClassList != null && defaultStorageClassList.size() == 1) ? existing.equals(defaultStorageClassList.get(0)) : false; } if (StringUtils.isNotBlank(existing) && StringUtils.isNotBlank(modified)) { return existing.equals(modified); } return false; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy