io.hyscale.controller.provider.EffectiveServiceSpecProvider Maven / Gradle / Ivy
/**
* 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.provider;
import java.io.File;
import java.io.IOException;
import java.util.AbstractMap.SimpleEntry;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.stream.Collectors;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;
import com.fasterxml.jackson.databind.ObjectMapper;
import io.hyscale.commons.exception.HyscaleException;
import io.hyscale.commons.io.HyscaleFilesUtil;
import io.hyscale.commons.logger.WorkflowLogger;
import io.hyscale.commons.models.ServiceMetadata;
import io.hyscale.commons.models.Status;
import io.hyscale.commons.utils.ObjectMapperFactory;
import io.hyscale.controller.activity.ControllerActivity;
import io.hyscale.controller.constants.WorkflowConstants;
import io.hyscale.controller.exception.ControllerErrorCodes;
import io.hyscale.controller.model.EffectiveServiceSpec;
import io.hyscale.controller.model.HyscaleInputSpec;
import io.hyscale.controller.util.ServiceProfileUtil;
import io.hyscale.controller.util.ServiceSpecUtil;
import io.hyscale.servicespec.commons.builder.EffectiveServiceSpecBuilder;
import io.hyscale.servicespec.commons.builder.MapFieldDataProvider;
import io.hyscale.servicespec.commons.builder.ServiceInputType;
import io.hyscale.servicespec.commons.model.service.ServiceSpec;
@Component
public class EffectiveServiceSpecProvider {
private static final Logger logger = LoggerFactory.getLogger(EffectiveServiceSpecProvider.class);
/**
* Calls {@link #getEffectiveServiceSpec(List, List)}
* For service spec and profile in {@link HyscaleInputSpec}
* @param hyscaleInputSpec
* @return List of {@link EffectiveServiceSpec}
* @throws HyscaleException
*/
public List getEffectiveServiceSpec(HyscaleInputSpec hyscaleInputSpec)
throws HyscaleException {
return getEffectiveServiceSpec(hyscaleInputSpec.getServiceSpecFiles(), hyscaleInputSpec.getProfileFiles());
}
/**
* From List of service specs and profiles create a dependency graph of service to profile
* Using service spec files and dependency graph merge service spec and profile to get effective service spec
* @param serviceSpecFiles
* @param profileFiles
* @return List of {@link EffectiveServiceSpec}
* @throws HyscaleException
*/
public List getEffectiveServiceSpec(List serviceSpecFiles, List profileFiles)
throws HyscaleException {
Map> serviceVsProfile = getDependencyMap(serviceSpecFiles, profileFiles);
return mergeServiceSpec(serviceSpecFiles, serviceVsProfile);
}
/**
* For each service spec file, read service spec data
* if profile available for service spec
* uses {@link EffectiveServiceSpecBuilder} to merge profile and spec
* to get updated service spec data
*
* Uses service spec data to create {@link ServiceSpec}
*
* @param serviceSpecFiles
* @param serviceVsProfile
* @return List of {@link EffectiveServiceSpec}
* @throws HyscaleException
*/
private List mergeServiceSpec(List serviceSpecFiles,
Map> serviceVsProfile) throws HyscaleException {
List effectiveServiceSpecList = new ArrayList<>();
ObjectMapper mapper = null;
for (File serviceSpecFile : serviceSpecFiles) {
EffectiveServiceSpec effectiveServiceSpec = new EffectiveServiceSpec();
ServiceMetadata serviceMetadata = new ServiceMetadata();
effectiveServiceSpec.setServiceMetadata(serviceMetadata);
mapper = ObjectMapperFactory.yamlMapper();
String serviceName = ServiceSpecUtil.getServiceName(serviceSpecFile);
serviceMetadata.setServiceName(serviceName);
Entry profileDetail = serviceVsProfile.remove(serviceName);
String serviceSpecData = HyscaleFilesUtil.readFileData(serviceSpecFile);
if (profileDetail != null) {
String profileName = profileDetail.getKey();
serviceMetadata.setEnvName(profileName);
File profileFile = profileDetail.getValue();
WorkflowLogger.startActivity(ControllerActivity.APPLYING_PROFILE_FOR_SERVICE, profileName, serviceName);
mapper = ObjectMapperFactory.jsonMapper();
try {
MapFieldDataProvider mapFieldDataProvider = new MapFieldDataProvider();
// Merge
serviceSpecData = new EffectiveServiceSpecBuilder().type(ServiceInputType.YAML)
.withServiceSpec(serviceSpecData).withProfile(HyscaleFilesUtil.readFileData(profileFile))
.withFieldMetaDataProvider(mapFieldDataProvider).build();
WorkflowLogger.endActivity(Status.DONE);
} catch (HyscaleException e) {
logger.error("Error while applying profile {} for service {}", profileName, serviceName, e);
WorkflowLogger.endActivity(Status.FAILED);
throw e;
}
}
if (serviceMetadata.getEnvName() == null) {
serviceMetadata.setEnvName(WorkflowConstants.DEV_ENV);
}
try {
effectiveServiceSpec.setServiceSpec(new ServiceSpec(mapper.readTree(serviceSpecData)));
effectiveServiceSpecList.add(effectiveServiceSpec);
} catch (IOException e) {
logger.error("Error while processing service spec ", e);
throw new HyscaleException(ControllerErrorCodes.SERVICE_SPEC_PROCESSING_FAILED, e.getMessage());
}
}
return effectiveServiceSpecList;
}
private Map> getDependencyMap(List serviceSpecFiles, List profileFiles)
throws HyscaleException {
Map> serviceVsProfile = new HashMap<>();
List invalidServiceList = new ArrayList<>();
if (profileFiles != null && !profileFiles.isEmpty()) {
for (File profileFile : profileFiles) {
String profileName = ServiceProfileUtil.getProfileName(profileFile);
String serviceName = ServiceProfileUtil.getServiceNameFromProfile(profileFile);
if (serviceVsProfile.get(serviceName) != null) {
// Multiple profiles for a single service
invalidServiceList.add(serviceName);
}
serviceVsProfile.put(serviceName, new SimpleEntry<>(profileName, profileFile));
}
}
if (!invalidServiceList.isEmpty()) {
String invalidServices = invalidServiceList.toString();
logger.error("Multiple profiles found for services {}", invalidServices);
throw new HyscaleException(ControllerErrorCodes.UNIQUE_PROFILE_REQUIRED, invalidServices);
}
Map serviceVsSpecFile = new HashMap<>();
for (File serviceSpecFile : serviceSpecFiles) {
serviceVsSpecFile.put(ServiceSpecUtil.getServiceName(serviceSpecFile), serviceSpecFile);
}
// Services specified in profile not found
invalidServiceList = serviceVsProfile.entrySet().stream().map(Entry::getKey)
.filter(service -> !serviceVsSpecFile.containsKey(service)).collect(Collectors.toList());
if (invalidServiceList != null && !invalidServiceList.isEmpty()) {
String invalidServices = invalidServiceList.toString();
logger.error("Services {} mentioned in profiles not available in deployment", invalidServices);
throw new HyscaleException(ControllerErrorCodes.SERVICES_NOT_PROVIDED_FOR_PROFILE, invalidServices);
}
return serviceVsProfile;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy