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

io.hyscale.controller.commands.generate.HyscaleGenerateServiceManifestsCommand 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.controller.commands.generate;

import java.io.File;
import java.util.*;
import java.util.concurrent.Callable;

import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;
import javax.validation.constraints.Pattern;

import io.hyscale.commons.validator.Validator;
import io.hyscale.controller.profile.ServiceSpecProcessor;
import io.hyscale.controller.validator.impl.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import io.hyscale.commons.config.SetupConfig;
import io.hyscale.commons.constants.ToolConstants;
import io.hyscale.commons.constants.ValidationConstants;
import io.hyscale.commons.exception.HyscaleException;
import io.hyscale.commons.logger.WorkflowLogger;
import io.hyscale.controller.activity.ControllerActivity;
import io.hyscale.controller.model.WorkflowContextBuilder;
import io.hyscale.controller.commands.args.FileConverter;
import io.hyscale.controller.commands.input.ProfileArg;
import io.hyscale.controller.constants.WorkflowConstants;
import io.hyscale.controller.invoker.ManifestGeneratorComponentInvoker;
import io.hyscale.controller.model.EffectiveServiceSpec;
import io.hyscale.controller.model.WorkflowContext;
import io.hyscale.controller.util.CommandUtil;
import io.hyscale.controller.util.ServiceSpecUtil;
import picocli.CommandLine;
import picocli.CommandLine.ArgGroup;

/**
 * This class executes 'hyscale generate service manifests' command
 * It is a sub-command of the 'hyscale generate service' command
 *
 * Options:
 *  appName - name of the app
 *  serviceSpecs - list of service specs
 *  profiles - list of profiles for services
 *  profile - profile name to look for. Profile file should be present for all services in service spec (profiles and profile are mutually exclusive)
 * 

* Eg 1: hyscale generate service manifests -f svc.hspec -f svcb.hspec -p dev-svc.hprof -a sample * Eg 2: hyscale generate service manifests -f svc.hspec -P dev -a sample *

* Performs a validation of input before starting manifest generation. * Generates the manifests from the given hspec and writes the manifests * to {@literal}/hyscale/apps/{@literal}/{@literal}/generated-files/manifests/ * @see HyscaleGenerateServiceCommand * Every command/sub-command has to implement the Runnable so that * whenever the command is executed the {@link #call()} * method will be invoked */ @CommandLine.Command(name = "manifests", aliases = {"manifest"}, description = {"Generates manifests from the given service specs"}) @Component public class HyscaleGenerateServiceManifestsCommand implements Callable { private static final Logger logger = LoggerFactory.getLogger(HyscaleGenerateServiceManifestsCommand.class); @CommandLine.Option(names = {"-h", "--help"}, usageHelp = true, description = "Display help message about the specified command") private boolean helpRequested = false; @Pattern(regexp = ValidationConstants.APP_NAME_REGEX, message = ValidationConstants.INVALID_APP_NAME_MSG) @CommandLine.Option(names = {"-a", "--app"}, required = true, description = "Application name") private String appName; @CommandLine.Option(names = {"-f", "--files"}, required = true, description = "Service specs files.", split = ",", converter = FileConverter.class) private List serviceSpecsFiles; @ArgGroup(exclusive = true, heading = "Profile Options", order = 10) private ProfileArg profileArg; @Autowired private InputSpecPostValidator inputSpecPostValidator; @Autowired private ManifestGeneratorComponentInvoker manifestGeneratorComponentInvoker; @Autowired private ServiceSpecInputValidator serviceSpecInputValidator; @Autowired private ServiceSpecProcessor serviceSpecProcessor; @Autowired private RegistryValidator registryValidator; @Autowired private ManifestValidator manifestValidator; @Autowired private PortsValidator portsValidator; @Autowired private NetworkPoliciesValidator networkPoliciesValidator; private List> postValidators; @PostConstruct public void init() { this.postValidators = new LinkedList<>(); this.postValidators.add(registryValidator); this.postValidators.add(manifestValidator); this.postValidators.add(portsValidator); this.postValidators.add(networkPoliciesValidator); } @Override public Integer call() throws Exception { if (!CommandUtil.isInputValid(this)) { return ToolConstants.INVALID_INPUT_ERROR_CODE; } // Validate Service specs with schema if (!serviceSpecInputValidator.validate(serviceSpecsFiles)) { return ToolConstants.INVALID_INPUT_ERROR_CODE; } Map serviceVsSpecFile = new HashMap<>(); for (File serviceSpec : serviceSpecsFiles) { serviceVsSpecFile.put(ServiceSpecUtil.getServiceName(serviceSpec), serviceSpec); } // Construct EffectiveServiceSpec from ProfileArg & ServiceSpecFiles List effectiveServiceSpecs = serviceSpecProcessor.process(profileArg, serviceSpecsFiles); List contextList = new ArrayList<>(); for (EffectiveServiceSpec each : effectiveServiceSpecs) { if (each != null && each.getServiceSpec() != null && each.getServiceMetadata() != null) { WorkflowContextBuilder builder = new WorkflowContextBuilder(appName); try { builder.withProfile(each.getServiceMetadata().getEnvName()); builder.withService(each.getServiceSpec()); contextList.add(builder.get()); } catch (HyscaleException e) { logger.error("Error while preparing workflow context ",e); throw e; } } } postValidators.forEach(each -> inputSpecPostValidator.addValidator(each)); if (!inputSpecPostValidator.validate(contextList)) { return ToolConstants.INVALID_INPUT_ERROR_CODE; } boolean isFailed = false; for (WorkflowContext workflowContext : contextList) { isFailed = !executeInvoker(serviceVsSpecFile, workflowContext) || isFailed; } return isFailed ? ToolConstants.HYSCALE_ERROR_CODE : 0; } private boolean executeInvoker(Map serviceVsSpecFile, WorkflowContext workflowContext) throws HyscaleException { boolean isFailed = workflowContext.isFailed(); String serviceName = workflowContext.getServiceName(); WorkflowLogger.header(ControllerActivity.SERVICE_NAME, serviceName); SetupConfig.clearAbsolutePath(); SetupConfig.setAbsolutePath(serviceVsSpecFile.get(serviceName).getAbsoluteFile().getParent()); try { manifestGeneratorComponentInvoker.execute(workflowContext); } catch (HyscaleException e) { logger.error("Error while generating manifest for app: {}, service: {}", appName, serviceName, e); throw e; } if (workflowContext.isFailed()) { isFailed = true; } WorkflowLogger.footer(); WorkflowLogger.logPersistedActivities(); CommandUtil.logMetaInfo( SetupConfig.getMountPathOf((String) workflowContext.getAttribute(WorkflowConstants.MANIFESTS_PATH)), ControllerActivity.MANIFESTS_GENERATION_PATH); return !isFailed; } @PreDestroy public void clear() { SetupConfig.clearAbsolutePath(); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy