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

com.mooltiverse.oss.nyx.configuration.Configuration Maven / Gradle / Ivy

/*
 * Copyright 2020 Mooltiverse
 *
 * 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 com.mooltiverse.oss.nyx.configuration;

import static com.mooltiverse.oss.nyx.log.Markers.CONFIGURATION;

import java.io.File;

import java.util.ArrayList;
import java.util.EnumMap;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.mooltiverse.oss.nyx.Nyx;
import com.mooltiverse.oss.nyx.configuration.presets.Presets;
import com.mooltiverse.oss.nyx.entities.Attachment;
import com.mooltiverse.oss.nyx.entities.ChangelogConfiguration;
import com.mooltiverse.oss.nyx.entities.CommitMessageConvention;
import com.mooltiverse.oss.nyx.entities.CommitMessageConventions;
import com.mooltiverse.oss.nyx.entities.GitConfiguration;
import com.mooltiverse.oss.nyx.entities.GitRemoteConfiguration;
import com.mooltiverse.oss.nyx.entities.IllegalPropertyException;
import com.mooltiverse.oss.nyx.entities.ReleaseType;
import com.mooltiverse.oss.nyx.entities.ReleaseTypes;
import com.mooltiverse.oss.nyx.entities.ServiceConfiguration;
import com.mooltiverse.oss.nyx.entities.Substitution;
import com.mooltiverse.oss.nyx.entities.Substitutions;
import com.mooltiverse.oss.nyx.entities.Verbosity;
import com.mooltiverse.oss.nyx.io.DataAccessException;
import com.mooltiverse.oss.nyx.io.FileMapper;
import com.mooltiverse.oss.nyx.version.Scheme;

/**
 * The Nyx configuration. The configuration is a live object that resolves each option lazily, only when required.
 * This not only improves the overall performances but is also safer as in case of malformed configuration options, only
 * those actually needed are resolved.
 * 
* This means that even if the configuration sources don't change throughout a release process, the state of the * configuration may change every time a not yet resolved option is requested and evaluated. *
* The configuration is layered, where each layer represents a source of configuration options. There is a clear definition * of priorities among different layers so there is a clear precedence of options coming from one layer or another. * Thanks to this, each option can be overridden by other layer with higher priority. *
* There must be only one instance of this class for every execution and it's retrieved by {@link Nyx#configuration()}. */ public class Configuration implements ConfigurationRoot { /** * The private logger instance */ private static final Logger logger = LoggerFactory.getLogger(Configuration.class); /** * The private instance of the changelog configuration section. */ private ChangelogConfiguration changelogSection = null; /** * The private instance of the commit message convention configuration section. */ private CommitMessageConventions commitMessageConventionsSection = null; /** * The private instance of the Git configuration section. */ private GitConfiguration gitSection = null; /** * The private instance of the release assets configuration section. */ private Map releaseAssetsSection = null; /** * The private instance of the release types configuration section. */ private ReleaseTypes releaseTypesSection = null; /** * The private instance of the services configuration section. */ private Map servicesSection = null; /** * The private instance of the substitutions configuration section. */ private Substitutions substitutionsSection = null; /** * The internal representation of the configuration layers and their priorities. * * Map entries are ordered by the natural order of the {@link LayerPriority} keys so by iterating over the * entries in this map we are assured that the evaluation order is safe. * * This instance is initialized with the {@link DefaultLayer} (which has the least priority) as a last resort * when looking up configuration options. */ private final EnumMap layers = new EnumMap(LayerPriority.class){ // The default serial version UID private static final long serialVersionUID = 1L; { // Initialize the layers with the default values, which have the least priority // and the environment variables layer, with an high priority put(LayerPriority.DEFAULT, DefaultLayer.getInstance()); put(LayerPriority.ENVIRONMENT, EnvironmentConfigurationLayer.getInstance()); } }; /** * Default constructor. Returns a new configuration object at its initial state. * * Instances of this class are created using the {@link Nyx#configuration()} method so this constructor should never be used alone. * * See {@link Nyx#configuration()} * * @throws DataAccessException in case data cannot be read or accessed. * @throws IllegalPropertyException in case some option has been defined but has incorrect values or it can't be resolved. */ public Configuration() throws DataAccessException, IllegalPropertyException { super(); logger.trace(CONFIGURATION, "New configuration object"); loadStandardConfigurationFileLayers(); } /** * Returns a file built on the given path if it's already an absolute file path, otherwise make it absolute by resolving it with the * configured (or default) directory. * * @param path the file to make absolute * * @return the absolute representation of the file * * @see #getDirectory() * * @throws DataAccessException in case data cannot be read or accessed. * @throws IllegalPropertyException in case some option has been defined but has incorrect values or it can't be resolved. */ private File getAbsoluteFilePath(String path) throws DataAccessException, IllegalPropertyException { File res = new File(path); return res.isAbsolute() ? res : new File(new File(getDirectory()), path); } /** * Loads the various standard configuration file layers, if found, searched at their default locations. * * @throws DataAccessException in case data cannot be read or accessed. * @throws IllegalPropertyException in case some option has been defined but has incorrect values or it can't be resolved. */ private void loadStandardConfigurationFileLayers() throws DataAccessException, IllegalPropertyException { logger.debug(CONFIGURATION, "Searching for standard configuration files..."); // load standard local configuration files first, if any for (String fileName: List.of(".nyx.json", ".nyx.yaml", ".nyx.yml")) { File file = getAbsoluteFilePath(fileName); if (file.exists() && file.isFile()) { logger.debug(CONFIGURATION, "Standard local configuration file found at '{}'. Loading...", file.getAbsolutePath()); layers.put(LayerPriority.STANDARD_LOCAL_FILE, FileMapper.load(file, SimpleConfigurationLayer.class)); logger.debug(CONFIGURATION, "Standard local configuration file '{}' loaded", file.getAbsolutePath()); break; } else logger.debug(CONFIGURATION, "Standard local configuration file '{}' not found.", file.getAbsolutePath()); } // then load standard shared configuration files, if any for (String fileName: List.of(".nyx-shared.json", ".nyx-shared.yaml", ".nyx-shared.yml")) { File file = getAbsoluteFilePath(fileName); if (file.exists() && file.isFile()) { logger.debug(CONFIGURATION, "Standard shared configuration file found at '{}'. Loading...", file.getAbsolutePath()); layers.put(LayerPriority.STANDARD_SHARED_FILE, FileMapper.load(file, SimpleConfigurationLayer.class)); logger.debug(CONFIGURATION, "Standard shared configuration file '{}' loaded", file.getAbsolutePath()); break; } else logger.debug(CONFIGURATION, "Standard shared configuration file '{}' not found.", file.getAbsolutePath()); } if (layers.containsKey(LayerPriority.STANDARD_LOCAL_FILE) || layers.containsKey(LayerPriority.STANDARD_SHARED_FILE)) updateConfiguredConfigurationLayers(); } /** * Resets the cache of resolved options. */ private synchronized void resetCache() { logger.trace(CONFIGURATION, "Clearing the configuration cache"); changelogSection = null; commitMessageConventionsSection = null; gitSection = null; releaseAssetsSection = null; releaseTypesSection = null; servicesSection = null; substitutionsSection = null; } /** * Makes sure that the configuration layers that can be configured (custom configuration file, custom shared * configuration file, preset) are added or removed into the internal layers representation, according to the * actual configuration parameters. * * This method must be invoked after any core layer (not among the configured ones) changes, or after a batch * of those changes. * * @throws DataAccessException in case data cannot be read or accessed. * @throws IllegalPropertyException in case some option has been defined but has incorrect values or it can't be resolved. */ private synchronized void updateConfiguredConfigurationLayers() throws DataAccessException, IllegalPropertyException { // follow the evaluation order from top to bottom to make sure that any change applied here does not // require further changes to the layers that we already checked // start with the local custom configuration file if (Objects.isNull(getConfigurationFile())) { logger.debug(CONFIGURATION, "Clearing the custom local configuration file, if any"); layers.remove(LayerPriority.CUSTOM_LOCAL_FILE); } else if (getConfigurationFile().isBlank()) { logger.error(CONFIGURATION, "An empty path has been defined for the local custom configuration file and it will be ignored"); layers.remove(LayerPriority.CUSTOM_LOCAL_FILE); } else { File customLocalConfigurationFile = getAbsoluteFilePath(getConfigurationFile()); logger.debug(CONFIGURATION, "Loading custom local configuration file at '{}'", customLocalConfigurationFile.getAbsolutePath()); layers.put(LayerPriority.CUSTOM_LOCAL_FILE, FileMapper.load(customLocalConfigurationFile, SimpleConfigurationLayer.class)); logger.debug(CONFIGURATION, "Custom local configuration file '{}' loaded", customLocalConfigurationFile.getAbsolutePath()); } // now the local shared configuration file if (Objects.isNull(getSharedConfigurationFile())) { logger.debug(CONFIGURATION, "Clearing the custom shared configuration file, if any"); layers.remove(LayerPriority.CUSTOM_SHARED_FILE); } else if (getSharedConfigurationFile().isBlank()) { logger.error(CONFIGURATION, "An empty path has been defined for the local shared configuration file and it will be ignored"); layers.remove(LayerPriority.CUSTOM_SHARED_FILE); } else { File customSharedConfigurationFile = getAbsoluteFilePath(getSharedConfigurationFile()); logger.debug(CONFIGURATION, "Loading custom shared configuration file at '{}'", customSharedConfigurationFile.getAbsolutePath()); layers.put(LayerPriority.CUSTOM_SHARED_FILE, FileMapper.load(customSharedConfigurationFile, SimpleConfigurationLayer.class)); logger.debug(CONFIGURATION, "Custom shared configuration file '{}' loaded", customSharedConfigurationFile.getAbsolutePath()); } // now the preset if (Objects.isNull(getPreset())) { logger.debug(CONFIGURATION, "Clearing the preset configuration, if any"); layers.remove(LayerPriority.PRESET); } else if (getPreset().isBlank()) { logger.error(CONFIGURATION, "An empty name has been defined for the preset configuration and it will be ignored"); layers.remove(LayerPriority.PRESET); } else { logger.debug(CONFIGURATION, "Loading preset configuration '{}'", getPreset()); layers.put(LayerPriority.PRESET, Presets.byName(getPreset())); logger.debug(CONFIGURATION, "Preset configuration '{}' loaded", getPreset()); } } /** * Adds, replaces or removes the layer at the {@link LayerPriority#COMMAND_LINE} level. * * @param layer the configuration layer to set at the {@link LayerPriority#COMMAND_LINE} level. * If {@code null} any existing configuration layer at the same level is removed (if any). * * @return a reference to this same object. * * @throws DataAccessException in case data cannot be read or accessed. * @throws IllegalPropertyException in case some option has been defined but has incorrect values or it can't be resolved. */ public Configuration withCommandLineConfiguration(ConfigurationLayer layer) throws DataAccessException, IllegalPropertyException { if (Objects.isNull(layer)) { logger.debug(CONFIGURATION, "Removing the existing '{}' configuration layer, if any", LayerPriority.COMMAND_LINE); layers.remove(LayerPriority.COMMAND_LINE); } else { logger.debug(CONFIGURATION, "Adding or replacing the '{}' configuration layer", LayerPriority.COMMAND_LINE); layers.put(LayerPriority.COMMAND_LINE, layer); } updateConfiguredConfigurationLayers(); resetCache(); return this; } /** * Adds, replaces or removes the layer at the {@link LayerPriority#PLUGIN} level. * * @param layer the configuration layer to set at the {@link LayerPriority#PLUGIN} level. * If {@code null} any existing configuration layer at the same level is removed (if any). * * @return a reference to this same object. * * @throws DataAccessException in case data cannot be read or accessed. * @throws IllegalPropertyException in case some option has been defined but has incorrect values or it can't be resolved. */ public Configuration withPluginConfiguration(ConfigurationLayer layer) throws DataAccessException, IllegalPropertyException { if (Objects.isNull(layer)) { logger.debug(CONFIGURATION, "Removing the existing '{}' configuration layer, if any", LayerPriority.PLUGIN); layers.remove(LayerPriority.PLUGIN); } else { logger.debug(CONFIGURATION, "Adding or replacing the '{}' configuration layer", LayerPriority.PLUGIN); layers.put(LayerPriority.PLUGIN, layer); } updateConfiguredConfigurationLayers(); resetCache(); return this; } /** * Adds, replaces or removes the layer at the {@link LayerPriority#RUNTIME} level, which is the one that can override * all other layers. * * @param layer the configuration layer to set at the {@link LayerPriority#RUNTIME} level. * If {@code null} any existing configuration layer at the same level is removed (if any). * * @return a reference to this same object. * * @throws DataAccessException in case data cannot be read or accessed. * @throws IllegalPropertyException in case some option has been defined but has incorrect values or it can't be resolved. */ public Configuration withRuntimeConfiguration(ConfigurationLayer layer) throws DataAccessException, IllegalPropertyException { if (Objects.isNull(layer)) { logger.debug(CONFIGURATION, "Removing the existing '{}' configuration layer, if any", LayerPriority.RUNTIME); layers.remove(LayerPriority.RUNTIME); } else { logger.debug(CONFIGURATION, "Adding or replacing the '{}' configuration layer", LayerPriority.RUNTIME); layers.put(LayerPriority.RUNTIME, layer); } updateConfiguredConfigurationLayers(); resetCache(); return this; } /** * {@inheritDoc} */ @Override public String getBump() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "bump"); for (ConfigurationLayer layer: layers.values()) { String bump = layer.getBump(); if (!Objects.isNull(bump)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "bump", bump); return bump; } } return DefaultLayer.getInstance().getBump(); } /** * {@inheritDoc} */ @Override public ChangelogConfiguration getChangelog() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the changelog configuration"); if (Objects.isNull(changelogSection)) { changelogSection = new ChangelogConfiguration(); for (ConfigurationLayer layer: layers.values()) { // Since all attributes of the changelog configuration are objects we assume that if they are null // they have the default values and we keep non null values as those overriding defaults. // The sections map is assumed to override inherited values if its size is not 0 if (Objects.isNull(changelogSection.getAppend())) changelogSection.setAppend(layer.getChangelog().getAppend()); if (Objects.isNull(changelogSection.getPath())) changelogSection.setPath(layer.getChangelog().getPath()); if (Objects.isNull(changelogSection.getSections()) || changelogSection.getSections().isEmpty()) changelogSection.setSections(layer.getChangelog().getSections()); if (Objects.isNull(changelogSection.getSubstitutions()) || changelogSection.getSubstitutions().isEmpty()) changelogSection.setSubstitutions(layer.getChangelog().getSubstitutions()); if (Objects.isNull(changelogSection.getTemplate())) changelogSection.setTemplate(layer.getChangelog().getTemplate()); } logger.trace(CONFIGURATION, "The '{}' configuration option has been resolved", "changelog"); } return changelogSection; } /** * {@inheritDoc} */ @Override public CommitMessageConventions getCommitMessageConventions() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the commit message conventions"); if (Objects.isNull(commitMessageConventionsSection)) { // parse the 'enabled' items list List enabled = new ArrayList(); for (ConfigurationLayer layer: layers.values()) { if (!Objects.isNull(layer.getCommitMessageConventions().getEnabled()) && !layer.getCommitMessageConventions().getEnabled().isEmpty()) { enabled = layer.getCommitMessageConventions().getEnabled(); logger.trace(CONFIGURATION, "The '{}.{}' configuration option value is: '{}'", "commitMessageConventions", "enabled", String.join(", ", enabled)); break; } } // parse the 'items' map Map items = new HashMap(); for (String enabledItem: enabled) { for (ConfigurationLayer layer: layers.values()) { CommitMessageConvention item = layer.getCommitMessageConventions().getItems().get(enabledItem); if (!Objects.isNull(item)) { items.put(enabledItem, item); logger.trace(CONFIGURATION, "The '{}.{}[{}]' configuration option has been resolved", "commitMessageConventions", "items", enabledItem); break; } } } commitMessageConventionsSection = new CommitMessageConventions(enabled, items); logger.trace(CONFIGURATION, "The '{}' configuration option has been resolved", "commitMessageConventions"); } return commitMessageConventionsSection; } /** * {@inheritDoc} */ @Override public String getConfigurationFile() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "configurationFile"); for (Map.Entry layerEntry: layers.entrySet()) { // custom configuration file configuration is ignored on custom configuration file layers to avoid chaining if (!LayerPriority.CUSTOM_LOCAL_FILE.equals(layerEntry.getKey())) { String configurationFile = layerEntry.getValue().getConfigurationFile(); if (!Objects.isNull(configurationFile)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "configurationFile", configurationFile); return configurationFile; } } } return DefaultLayer.getInstance().getConfigurationFile(); } /** * {@inheritDoc} */ @Override public String getDirectory() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "directory"); for (ConfigurationLayer layer: layers.values()) { String directory = layer.getDirectory(); if (!Objects.isNull(directory)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "directory", directory); return directory; } } return DefaultLayer.getInstance().getDirectory(); } /** * This method allows to override the default directory that will be returned by {@link #getDirectory()}. * This method must be invoked before instances of {@link Nyx} or other classes are created or the given value may be ignored. * * @param directory the new default directory. If {@code null} then the standard default directory will be used. */ public static void setDefaultDirectory(File directory) { logger.trace(CONFIGURATION, "Setting the default directory '{}'", Objects.isNull(directory) ? "null" : directory.getAbsolutePath()); DefaultLayer.getInstance().setDirectory(Objects.isNull(directory) ? null : directory.getAbsolutePath()); } /** * {@inheritDoc} */ @Override public Boolean getDryRun() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "dryRun"); for (ConfigurationLayer layer: layers.values()) { Boolean dryRun = layer.getDryRun(); if (!Objects.isNull(dryRun)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "dryRun", dryRun); return dryRun; } } return DefaultLayer.getInstance().getDryRun(); } /** * {@inheritDoc} */ @Override public GitConfiguration getGit() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the Git configuration"); if (Objects.isNull(gitSection)) { // parse the 'remotes' map Map remotes = new HashMap(); for (ConfigurationLayer layer: layers.values()) { for (String remoteName: layer.getGit().getRemotes().keySet()) { GitRemoteConfiguration remote = layer.getGit().getRemotes().get(remoteName); if (!Objects.isNull(remote) && !remotes.containsKey(remoteName)) { remotes.put(remoteName, remote); logger.trace(CONFIGURATION, "The '{}.{}[{}]' configuration option has been resolved", "git", "remotes", remoteName); } } } gitSection = new GitConfiguration(remotes); } return gitSection; } /** * {@inheritDoc} */ @Override public String getInitialVersion() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "initialVersion"); for (ConfigurationLayer layer: layers.values()) { String initialVersion = layer.getInitialVersion(); if (!Objects.isNull(initialVersion)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "initialVersion", initialVersion); return initialVersion; } } return DefaultLayer.getInstance().getInitialVersion(); } /** * {@inheritDoc} */ @Override public String getPreset() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "preset"); for (Map.Entry layerEntry: layers.entrySet()) { // preset configuration is ignored on preset layers to avoid chaining if (!LayerPriority.PRESET.equals(layerEntry.getKey())) { String preset = layerEntry.getValue().getPreset(); if (!Objects.isNull(preset)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "preset", preset); return preset; } } } return DefaultLayer.getInstance().getPreset(); } /** * {@inheritDoc} */ @Override public Map getReleaseAssets() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the release assets"); if (Objects.isNull(releaseAssetsSection)) { // parse the 'releaseAssets' map releaseAssetsSection = new HashMap(); for (ConfigurationLayer layer: layers.values()) { for (String releaseAssetName: layer.getReleaseAssets().keySet()) { if (!releaseAssetsSection.containsKey(releaseAssetName)) { releaseAssetsSection.put(releaseAssetName, layer.getReleaseAssets().get(releaseAssetName)); logger.trace(CONFIGURATION, "The '{}[{}]' configuration option has been resolved", "releaseAssets", releaseAssetName); } } } } return releaseAssetsSection; } /** * {@inheritDoc} */ @Override public Boolean getReleaseLenient() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "releaseLenient"); for (ConfigurationLayer layer: layers.values()) { Boolean releaseLenient = layer.getReleaseLenient(); if (!Objects.isNull(releaseLenient)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "releaseLenient", releaseLenient); return releaseLenient; } } return DefaultLayer.getInstance().getReleaseLenient(); } /** * {@inheritDoc} */ @Override public String getReleasePrefix() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "releasePrefix"); for (ConfigurationLayer layer: layers.values()) { String releasePrefix = layer.getReleasePrefix(); if (!Objects.isNull(releasePrefix)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "releasePrefix", releasePrefix); return releasePrefix; } } return DefaultLayer.getInstance().getReleasePrefix(); } /** * {@inheritDoc} */ @Override public ReleaseTypes getReleaseTypes() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the release types"); if (Objects.isNull(releaseTypesSection)) { // parse the 'enabled' items list List enabled = new ArrayList(); for (ConfigurationLayer layer: layers.values()) { if (!Objects.isNull(layer.getReleaseTypes().getEnabled()) && !layer.getReleaseTypes().getEnabled().isEmpty()) { enabled = layer.getReleaseTypes().getEnabled(); logger.trace(CONFIGURATION, "The '{}.{}' configuration option value is: '{}'", "releaseTypes", "enabled", String.join(", ", enabled)); break; } } // parse the 'publicationServices' items list List publicationServices = new ArrayList(); for (ConfigurationLayer layer: layers.values()) { if (!Objects.isNull(layer.getReleaseTypes().getPublicationServices()) && !layer.getReleaseTypes().getPublicationServices().isEmpty()) { publicationServices = layer.getReleaseTypes().getPublicationServices(); logger.trace(CONFIGURATION, "The '{}.{}' configuration option value is: '{}'", "releaseTypes", "publicationServices", String.join(", ", publicationServices)); break; } } // parse the 'remoteRepositories' items list List remoteRepositories = new ArrayList(); for (ConfigurationLayer layer: layers.values()) { if (!Objects.isNull(layer.getReleaseTypes().getRemoteRepositories()) && !layer.getReleaseTypes().getRemoteRepositories().isEmpty()) { remoteRepositories = layer.getReleaseTypes().getRemoteRepositories(); logger.trace(CONFIGURATION, "The '{}.{}' configuration option value is: '{}'", "releaseTypes", "remoteRepositories", String.join(", ", remoteRepositories)); break; } } // parse the 'items' map Map items = new HashMap(); for (String enabledItem: enabled) { for (ConfigurationLayer layer: layers.values()) { ReleaseType item = layer.getReleaseTypes().getItems().get(enabledItem); if (!Objects.isNull(item)) { items.put(enabledItem, item); logger.trace(CONFIGURATION, "The '{}.{}[{}]' configuration option has been resolved", "releaseTypes", "items", enabledItem); break; } } } releaseTypesSection = new ReleaseTypes(enabled, publicationServices, remoteRepositories, items); } return releaseTypesSection; } /** * {@inheritDoc} */ @Override public Boolean getResume() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "resume"); for (ConfigurationLayer layer: layers.values()) { Boolean resume = layer.getResume(); if (!Objects.isNull(resume)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "resume", resume); return resume; } } return DefaultLayer.getInstance().getResume(); } /** * {@inheritDoc} */ @Override public Scheme getScheme() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "scheme"); for (ConfigurationLayer layer: layers.values()) { Scheme scheme = layer.getScheme(); if (!Objects.isNull(scheme)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "scheme", scheme); return scheme; } } return DefaultLayer.getInstance().getScheme(); } /** * {@inheritDoc} */ @Override public Map getServices() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the services"); if (Objects.isNull(servicesSection)) { // parse the 'services' map servicesSection = new HashMap(); for (ConfigurationLayer layer: layers.values()) { for (String serviceName: layer.getServices().keySet()) { if (!servicesSection.containsKey(serviceName)) { servicesSection.put(serviceName, layer.getServices().get(serviceName)); logger.trace(CONFIGURATION, "The '{}[{}]' configuration option has been resolved", "services", serviceName); } } } } return servicesSection; } /** * {@inheritDoc} */ @Override public String getSharedConfigurationFile() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "sharedConfigurationFile"); for (Map.Entry layerEntry: layers.entrySet()) { // custom shared configuration file configuration is ignored on custom shared configuration file layers to avoid chaining if (!LayerPriority.CUSTOM_SHARED_FILE.equals(layerEntry.getKey())) { String sharedConfigurationFile = layerEntry.getValue().getSharedConfigurationFile(); if (!Objects.isNull(sharedConfigurationFile)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "sharedConfigurationFile", sharedConfigurationFile); return sharedConfigurationFile; } } } return DefaultLayer.getInstance().getSharedConfigurationFile(); } /** * {@inheritDoc} */ @Override public String getStateFile() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "stateFile"); for (ConfigurationLayer layer: layers.values()) { String stateFile = layer.getStateFile(); if (!Objects.isNull(stateFile)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "stateFile", stateFile); return stateFile; } } return DefaultLayer.getInstance().getStateFile(); } /** * {@inheritDoc} */ @Override public Substitutions getSubstitutions() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the substitutions"); if (Objects.isNull(substitutionsSection)) { // parse the 'enabled' items list List enabled = new ArrayList(); for (ConfigurationLayer layer: layers.values()) { if (!Objects.isNull(layer.getSubstitutions().getEnabled()) && !layer.getSubstitutions().getEnabled().isEmpty()) { enabled = layer.getSubstitutions().getEnabled(); logger.trace(CONFIGURATION, "The '{}.{}' configuration option value is: '{}'", "substitutions", "enabled", String.join(", ", enabled)); break; } } // parse the 'items' map Map items = new HashMap(); for (String enabledItem: enabled) { for (ConfigurationLayer layer: layers.values()) { Substitution item = layer.getSubstitutions().getItems().get(enabledItem); if (!Objects.isNull(item)) { items.put(enabledItem, item); logger.trace(CONFIGURATION, "The '{}.{}[{}]' configuration option has been resolved", "substitutions", "items", enabledItem); break; } } } substitutionsSection = new Substitutions(enabled, items); logger.trace(CONFIGURATION, "The '{}' configuration option has been resolved", "substitutions"); } return substitutionsSection; } /** * {@inheritDoc} */ @Override public Boolean getSummary() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "summary"); for (ConfigurationLayer layer: layers.values()) { Boolean summary = layer.getSummary(); if (!Objects.isNull(summary)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "summary", summary); return summary; } } return DefaultLayer.getInstance().getSummary(); } /** * {@inheritDoc} */ @Override public String getSummaryFile() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "summaryFile"); for (ConfigurationLayer layer: layers.values()) { String summaryFile = layer.getSummaryFile(); if (!Objects.isNull(summaryFile)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "summaryFile", summaryFile); return summaryFile; } } return DefaultLayer.getInstance().getSummaryFile(); } /** * {@inheritDoc} */ @Override public Verbosity getVerbosity() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "verbosity"); for (ConfigurationLayer layer: layers.values()) { Verbosity verbosity = layer.getVerbosity(); if (!Objects.isNull(verbosity)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "verbosity", verbosity); return verbosity; } } return DefaultLayer.getInstance().getVerbosity(); } /** * {@inheritDoc} */ @Override public String getVersion() throws DataAccessException, IllegalPropertyException { logger.trace(CONFIGURATION, "Retrieving the '{}' configuration option", "version"); for (ConfigurationLayer layer: layers.values()) { String version = layer.getVersion(); if (!Objects.isNull(version)) { logger.trace(CONFIGURATION, "The '{}' configuration option value is: '{}'", "version", version); return version; } } return DefaultLayer.getInstance().getVersion(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy