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

dev.galasa.boot.Launcher Maven / Gradle / Ivy

There is a newer version: 0.37.0
Show newest version
/*
 * Licensed Materials - Property of IBM
 * 
 * (c) Copyright IBM Corp. 2019.
 */
package dev.galasa.boot;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URL;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;
import java.util.Set;

import org.apache.commons.cli.CommandLine;
import org.apache.commons.cli.CommandLineParser;
import org.apache.commons.cli.DefaultParser;
import org.apache.commons.cli.Option;
import org.apache.commons.cli.Options;
import org.apache.commons.cli.ParseException;

import dev.galasa.boot.BootLogger.Level;
import dev.galasa.boot.felix.FelixFramework;

/**
 * Galasa command line launcher.
* * Required parameters:
*   --obr infra.obr --obr test.obr test.bundle/test.package.TestClass
* *
    *
  1. Start Felix OSGi framework
  2. *
  3. Loads required bundles
  4. *
  5. Loads OSGi Bundle repositories from one or more --obr * parameters
  6. *
  7. Loads test bundle test.bundle from OBR
  8. *
  9. Loads and runs test class test.package.TestClass
  10. *
* */ public class Launcher { private static final String OBR_OPTION = "obr"; private static final String BOOTSTRAP_OPTION = "bootstrap"; private static final String OVERRIDES_OPTION = "overrides"; private static final String RESOURCEMANAGEMENT_OPTION = "resourcemanagement"; private static final String K8SCONTROLLER_OPTION = "k8scontroller"; private static final String DOCKERCONTROLLER_OPTION = "dockercontroller"; private static final String METRICSERVER_OPTION = "metricserver"; private static final String TEST_OPTION = "test"; private static final String RUN_OPTION = "run"; private static final String BUNDLE_OPTION = "bundle"; private static final String METRICS_OPTION = "metrics"; private static final String HEALTH_OPTION = "health"; private static final String LOCALMAVEN_OPTION = "localmaven"; private static final String REMOTEMAVEN_OPTION = "remotemaven"; private static final String TRACE_OPTION = "trace"; private static final String USER_HOME = "user.home"; private static final BootLogger logger = new BootLogger(); private List bundleRepositories = new ArrayList<>(); private String testName; private String testBundleName; private String testClassName; private String runName; private FelixFramework felixFramework; private Properties boostrapProperties; private Properties overridesProperties; private boolean testRun; private boolean resourceManagement; private boolean k8sController; private boolean dockerController; private boolean metricsServer; private Integer metrics; private Integer health; private List bundles = new ArrayList<>(); private URL localMavenRepo; private List remoteMavenRepos = new ArrayList<>(); /** * Launcher main method */ public static void main(String[] args) throws Exception { Launcher launcher = new Launcher(); launcher.launch(args); System.exit(0); } /** * Process supplied command line arguments and run the test * * @param args test run parameters * @throws InterruptedException * @throws Exception */ protected void launch(String[] args) throws LauncherException, InterruptedException { felixFramework = new FelixFramework(); // Build Felix framework and install required bundles try { processCommandLine(args); } catch (ParseException e) { throw new LauncherException("Unable to parse command line arguments", e); } logger.debug("OBR Repository Files: " + bundleRepositories); logger.debug("Launching Framework..."); try { // Build the Framework buildFramework(); if (testRun) { if (runName == null) { // Run test class logger.debug("Test Bundle: " + testBundleName); logger.debug("Test Class: " + testClassName); overridesProperties.setProperty("framework.run.testbundleclass", this.testName); } else { logger.debug("Test Run: " + runName); overridesProperties.setProperty("framework.run.name", this.runName); } felixFramework.runTest(boostrapProperties, overridesProperties); } else if (resourceManagement) { logger.debug("Resource Management"); felixFramework.runResourceManagement(boostrapProperties, overridesProperties, bundles, metrics, health); } else if (k8sController) { logger.debug("Kubernetes Controller"); felixFramework.runK8sController(boostrapProperties, overridesProperties, bundles, metrics, health); } else if (dockerController) { logger.debug("Docker Controller"); felixFramework.runDockerController(boostrapProperties, overridesProperties, bundles, metrics, health); } else if (metricsServer) { logger.debug("Metrics Server"); felixFramework.runMetricsServer(boostrapProperties, overridesProperties, bundles, metrics, health); } } catch (LauncherException e) { logger.error("Unable run test class", e); throw e; } finally { if (felixFramework != null) { felixFramework.stopFramework(); } } logger.info("Boot complete"); } private void buildFramework() throws LauncherException { logger.debug("Launching Framework..."); try { felixFramework.buildFramework(bundleRepositories, this.boostrapProperties, localMavenRepo, remoteMavenRepos); } catch (Exception e) { throw new LauncherException("Unable to create and initialize Felix framework", e); } } /** * Process the supplied command line augments * * @param args supplied arguments * @throws ParseException */ private void processCommandLine(String[] args) throws ParseException { StringBuilder messageBuffer = new StringBuilder(); messageBuffer.append("Supplied command line arguments: "); for (String arg : args) { messageBuffer.append(arg + " "); } logger.debug(messageBuffer.toString()); Options options = new Options(); Option propertyOption = Option.builder().longOpt("D").argName("property=value").hasArgs().valueSeparator() .numberOfArgs(2).desc("use value for given properties").build(); options.addOption(propertyOption); options.addOption(null, OBR_OPTION, true, "Felix OBR Repository File name"); options.addOption(null, BOOTSTRAP_OPTION, true, "Bootstrap properties file url"); options.addOption(null, OVERRIDES_OPTION, true, "Overrides properties file url"); options.addOption(null, RESOURCEMANAGEMENT_OPTION, false, "A Resource Management server"); options.addOption(null, K8SCONTROLLER_OPTION, false, "A k8s Controller server"); options.addOption(null, DOCKERCONTROLLER_OPTION, false, "A Docker Controller server"); options.addOption(null, METRICSERVER_OPTION, false, "A Metrics server"); options.addOption(null, TEST_OPTION, true, "The test to run"); options.addOption(null, RUN_OPTION, true, "The run name"); options.addOption(null, BUNDLE_OPTION, true, "Extra bundles to load"); options.addOption(null, METRICS_OPTION, true, "The port the metrics server will open, 0 to disable"); options.addOption(null, HEALTH_OPTION, true, "The port the health server will open, 0 to disable"); options.addOption(null, LOCALMAVEN_OPTION, true, "The local maven repository, defaults to ~/.m2/repository"); options.addOption(null, REMOTEMAVEN_OPTION, true, "The remote maven repositories, defaults to central"); options.addOption(null, TRACE_OPTION, false, "Enable TRACE logging"); CommandLineParser parser = new DefaultParser(); CommandLine commandLine = null; commandLine = parser.parse(options, args); if (commandLine.hasOption("D")) { Properties commandLinePropety = commandLine.getOptionProperties("D"); Set keys = commandLinePropety.keySet(); for (Object key : keys) { System.setProperty((String) key, commandLinePropety.getProperty((String) key)); } } if (commandLine.hasOption(TRACE_OPTION)) { logger.setLevel(Level.TRACE); System.setProperty("log4j.configuration", "trace-log4j.properties"); } // *** Add any OBRs if coded if (commandLine.hasOption(OBR_OPTION)) { for (String option : commandLine.getOptionValues(OBR_OPTION)) { bundleRepositories.add(option); } } checkForBoostrap(commandLine); checkForOverrides(commandLine); checkForBundles(commandLine); checkForMetricsPort(commandLine); checkForHealthPort(commandLine); checkForLocalMaven(commandLine); checkForRemoteMaven(commandLine); testRun = commandLine.hasOption(TEST_OPTION) || commandLine.hasOption(RUN_OPTION); resourceManagement = commandLine.hasOption(RESOURCEMANAGEMENT_OPTION); k8sController = commandLine.hasOption(K8SCONTROLLER_OPTION); dockerController = commandLine.hasOption(DOCKERCONTROLLER_OPTION); metricsServer = commandLine.hasOption(METRICSERVER_OPTION); if (testRun) { runName = commandLine.getOptionValue(RUN_OPTION); if (runName != null) { runName = runName.toUpperCase(); } else { testName = commandLine.getOptionValue(TEST_OPTION); if (testName == null) { commandLineError("Error: A single test method must be supplied"); } String[] bundleclass = testName.split("/"); if (bundleclass.length != 2) { commandLineError("Error: Invalid test name format"); } testBundleName = bundleclass[0]; testClassName = bundleclass[1]; if (testBundleName.isEmpty() || testClassName.isEmpty()) { commandLineError("Error: Invalid test name format"); } } return; } if (resourceManagement) { return; } if (k8sController) { return; } if (dockerController) { return; } if (metricsServer) { return; } commandLineError( "Error: Must select either --test, --run, --k8scontroller, --metricserver or --resourcemanagement"); } private void checkForRemoteMaven(CommandLine commandLine) { try { if (commandLine.hasOption(REMOTEMAVEN_OPTION)) { for (String option : commandLine.getOptionValues(REMOTEMAVEN_OPTION)) { this.remoteMavenRepos.add(new URL(option)); } } else { this.remoteMavenRepos.add(new URL("http://cicscit.hursley.ibm.com/galasa/maven")); // DEFAULT } this.remoteMavenRepos.add(new URL("https://repo.maven.apache.org/maven2")); } catch (MalformedURLException e) { logger.error("invalid remote maven urls", e); commandLineError(null); } } private void checkForLocalMaven(CommandLine commandLine) { if (commandLine.hasOption(LOCALMAVEN_OPTION)) { String repo = commandLine.getOptionValue(LOCALMAVEN_OPTION); if (repo != null) { // *** Allowed with no parameter so that we can disable the local repository if ("disable".equals(repo)) { this.localMavenRepo = null; } else { try { this.localMavenRepo = new URL(repo); } catch (MalformedURLException e) { logger.error("--localmaven has an invalid URL", e); commandLineError(null); } if (!"file".equals(this.localMavenRepo.getProtocol())) { commandLineError("--localmaven must be a file: URL"); } } } } else { try { this.localMavenRepo = new File( System.getProperty(USER_HOME) + File.separator + ".m2" + File.separator + "repository").toURI() .toURL(); } catch (MalformedURLException e) { logger.error("internal error", e); commandLineError(null); } } } private void checkForMetricsPort(CommandLine commandLine) { if (commandLine.hasOption(METRICS_OPTION)) { String port = commandLine.getOptionValue(METRICS_OPTION); metrics = Integer.parseInt(port); } } private void checkForHealthPort(CommandLine commandLine) { if (commandLine.hasOption(HEALTH_OPTION)) { String port = commandLine.getOptionValue(HEALTH_OPTION); health = Integer.parseInt(port); } } private void checkForBundles(CommandLine commandLine) { if (commandLine.hasOption(BUNDLE_OPTION)) { for (String option : commandLine.getOptionValues(BUNDLE_OPTION)) { bundles.add(option); } } } /** * Retrieve the bootstrap URI from the command line and load the properties * file. If the the option is not provided, the default is * ~/.galasa/boostrap.properties which will be created if it does not exist. * * @param commandLine - The command line instance */ private void checkForBoostrap(CommandLine commandLine) { URI bootstrapUri = null; if (commandLine.hasOption(BOOTSTRAP_OPTION)) { String uri = commandLine.getOptionValue(BOOTSTRAP_OPTION); try { bootstrapUri = new URI(uri); } catch (URISyntaxException e) { logger.error("Invalid bootstrap URI provided", e); commandLineError(null); } } else { Path path = Paths.get(System.getProperty(USER_HOME), ".galasa", "bootstrap.properties"); try { if (!path.toFile().exists()) { if (!path.getParent().toFile().exists()) { Files.createDirectories(path.getParent()); } Files.createFile(path); } bootstrapUri = path.toUri(); } catch (IOException e) { logger.error("Unable to create empty default bootstrap file " + path.toUri().toString(), e); commandLineError(null); } } try (InputStream is = bootstrapUri.toURL().openStream()) { boostrapProperties = new Properties(); boostrapProperties.load(is); } catch (IOException e) { logger.error("Unable to load bootstrap properties", e); commandLineError(null); } } /** * Retrieve the bootsrap URI from the command line and load the properties file. * If the the option is not provided, the default is * ~/.galasa/boostrap.properties. It will not be created if it does not exist * * @param commandLine - The command line instance */ private void checkForOverrides(CommandLine commandLine) { URI overridesUri = null; if (commandLine.hasOption(OVERRIDES_OPTION)) { String uri = commandLine.getOptionValue(OVERRIDES_OPTION); try { overridesUri = new URI(uri); } catch (URISyntaxException e) { logger.error("Invalid overrides URI provided", e); commandLineError(null); } } else { Path path = Paths.get(System.getProperty(USER_HOME), ".galasa", "overrides.properties"); if (!path.toFile().exists()) { this.overridesProperties = new Properties(); return; } overridesUri = path.toUri(); } try (InputStream is = overridesUri.toURL().openStream()) { overridesProperties = new Properties(); overridesProperties.load(is); } catch (IOException e) { logger.error("Unable to load overrides properties", e); commandLineError(null); } } /** * Issue command line options error and exit */ private void commandLineError(String message) { if (message != null) { logger.error(message); } logger.error( "\nExample test run arguments: --obr infra.obr --obr test.obr --test test.bundle/test.package.TestClass\n" + "Example Resource Management arguments: --obr infra.obr --obr test.obr --resourcemanagement"); System.exit(-1); } }