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

com.mooltiverse.oss.nyx.command.Make 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.command;

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

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.IOException;
import java.io.Reader;
import java.io.StringWriter;
import java.net.URL;
import java.nio.file.Files;
import java.nio.file.FileSystem;
import java.nio.file.FileSystems;
import java.nio.file.FileVisitor;
import java.nio.file.FileVisitResult;
import java.nio.file.InvalidPathException;
import java.nio.file.Path;
import java.nio.file.PathMatcher;
import java.nio.file.Paths;
import java.nio.file.SimpleFileVisitor;
import java.nio.file.attribute.BasicFileAttributes;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URISyntaxException;
import java.text.SimpleDateFormat;
import java.time.ZoneOffset;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Date;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.Objects;
import java.util.TimeZone;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

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

import com.mooltiverse.oss.nyx.ReleaseException;
import com.mooltiverse.oss.nyx.entities.Changelog;
import com.mooltiverse.oss.nyx.entities.CommitMessageConvention;
import com.mooltiverse.oss.nyx.entities.IllegalPropertyException;
import com.mooltiverse.oss.nyx.entities.git.Commit;
import com.mooltiverse.oss.nyx.git.GitException;
import com.mooltiverse.oss.nyx.git.Repository;
import com.mooltiverse.oss.nyx.io.DataAccessException;
import com.mooltiverse.oss.nyx.state.State;
import com.mooltiverse.oss.nyx.template.Templates;

/**
 * The Make command takes care of building the release artifacts.
 * 
 * This class is not meant to be used in multi-threaded environments.
 */
public class Make extends AbstractCommand {
    /**
     * The name of the resource to load for the default template. Value: {@value}.
     */
    public static final String DEFAULT_TEMPLATE_RESOURCE_NAME = "changelog.tpl";

    /**
     * The private logger instance
     */
    private static final Logger logger = LoggerFactory.getLogger(Make.class);

    /**
     * The common prefix used for all the internal state attributes managed by this class.
     */
    private static final String INTERNAL_ATTRIBUTE_PREFIX = "make";

    /**
     * The common prefix used for all the internal state attributes managed by this class, representing an input.
     */
    private static final String INTERNAL_INPUT_ATTRIBUTE_PREFIX = INTERNAL_ATTRIBUTE_PREFIX.concat(".").concat("input");

    /**
     * The name used for the internal state attribute where we store the path to the changelog file.
     */
    private static final String INTERNAL_INPUT_ATTRIBUTE_CHANGELOG_FILE = INTERNAL_INPUT_ATTRIBUTE_PREFIX.concat(".").concat("changelog").concat(".").concat("file");

    /**
     * The name used for the internal state attribute where we store current branch name.
     */
    private static final String INTERNAL_INPUT_ATTRIBUTE_REPOSITORY_CURRENT_BRANCH = INTERNAL_INPUT_ATTRIBUTE_PREFIX.concat(".").concat("repository").concat(".").concat("current").concat(".").concat("branch");

    /**
     * The name used for the internal state attribute where we store the SHA-1 of the last
     * commit in the current branch by the time this command was last executed.
     */
    private static final String INTERNAL_INPUT_ATTRIBUTE_REPOSITORY_LAST_COMMIT = INTERNAL_INPUT_ATTRIBUTE_PREFIX.concat(".").concat("repository").concat(".").concat("last").concat(".").concat("commit");

    /**
     * The name used for the internal state attribute where we store the initial commit.
     */
    private static final String INTERNAL_INPUT_ATTRIBUTE_STATE_INITIAL_COMMIT = INTERNAL_INPUT_ATTRIBUTE_PREFIX.concat(".").concat("state").concat(".").concat("initialCommit");

    /**
     * The flag telling if the current version is new.
     */
    private static final String INTERNAL_INPUT_ATTRIBUTE_STATE_NEW_VERSION = INTERNAL_INPUT_ATTRIBUTE_PREFIX.concat(".").concat("state").concat(".").concat("newVersion");

    /**
     * The name used for the internal state attribute where we store the version.
     */
    private static final String INTERNAL_INPUT_ATTRIBUTE_VERSION = INTERNAL_INPUT_ATTRIBUTE_PREFIX.concat(".").concat("version");

    /**
     * Standard constructor.
     * 
     * @param state the state reference
     * @param repository the repository reference
     * 
     * @throws NullPointerException if a given argument is {@code null}
     */
    public Make(State state, Repository repository) {
        super(state, repository);
        logger.debug(COMMAND, "New Make command object");
    }

    /**
     * Returns the reference to the configured changelog file, if configured, or {@code null}
     * of no destination file has been set by the configuration.
     * 
     * @return the reference to the configured changelog file, if configured
     * 
     * @throws DataAccessException in case the configuration can't be loaded for some reason.
     * @throws IllegalPropertyException in case the configuration has some illegal options.
     */
    protected File getChangelogFile() 
        throws DataAccessException, IllegalPropertyException {
        if (Objects.isNull(state().getConfiguration().getChangelog()) || Objects.isNull(state().getConfiguration().getChangelog().getPath()) || Objects.isNull(state().getConfiguration().getChangelog().getPath().isBlank()))
            return null;

        File changelogFile = new File(state().getConfiguration().getChangelog().getPath());
        // if the file path is relative make it relative to the configured directory
        if (!changelogFile.isAbsolute())
            changelogFile = new File(state().getConfiguration().getDirectory(), state().getConfiguration().getChangelog().getPath());
        
        return changelogFile;
    }

    /**
     * Returns a reader object that reads the template to be used for rendering.
     * If the configuration overrides the template then the reader will point to that
     * template otherwise the default template will be returned.
     * 
     * @return a reader object that reads the template to be used for rendering. Never {@code null}.
     * 
     * @throws DataAccessException if an exception occurs when reading the template.
     * @throws IllegalPropertyException in case the configuration has some illegal options.
     */
    private Reader getChangelogTemplateReader()
        throws DataAccessException, IllegalPropertyException {
        if (Objects.isNull(state().getConfiguration().getChangelog().getTemplate()) || state().getConfiguration().getChangelog().getTemplate().isBlank()) {
            logger.debug(COMMAND, "The changelog template has not been overridden by configuration. Loading the default template.");
            try {
                InputStream is = getClass().getClassLoader().getResourceAsStream(DEFAULT_TEMPLATE_RESOURCE_NAME);
                if (Objects.isNull(is))
                    throw new DataAccessException(String.format("Unable to load the default changelog template resource from '%s'", DEFAULT_TEMPLATE_RESOURCE_NAME));
                else return new InputStreamReader(is);
            }
            catch (SecurityException se) {
                throw new DataAccessException(String.format("Unable to load the default changelog template resource from '%s'", DEFAULT_TEMPLATE_RESOURCE_NAME), se);
            }
        }
        else {
            String templatePath = state().getConfiguration().getChangelog().getTemplate();
            try {
                // try loading the file as an URL
                URL templateURL = new URI(templatePath).toURL();
                try {
                    return new InputStreamReader(templateURL.openStream());
                }
                catch (IOException ioe) {
                    throw new DataAccessException(String.format("Unable to load the configured changelog template file from URL '%s'", templatePath), ioe);
                }
            }
            catch (MalformedURLException | URISyntaxException | IllegalArgumentException ue) {
                // it's a local file, not an URL, so load it as such
                try {
                    return new FileReader(templatePath);
                }
                catch (FileNotFoundException fnfe) {
                    throw new DataAccessException(String.format("Unable to load the configured changelog template file from '%s'", templatePath), fnfe);
                }
            }
        }
    }

    /**
     * Builds the changelog assets.
     * 
     * @throws DataAccessException in case the configuration can't be loaded for some reason.
     * @throws IllegalPropertyException in case the configuration has some illegal options.
     * @throws GitException in case of unexpected issues when accessing the Git repository.
     * @throws ReleaseException if the task is unable to complete for reasons due to the release process.
     */
    private void buildChangelog()
        throws DataAccessException, IllegalPropertyException, GitException, ReleaseException {
        // The destination path is also used as a flag to enable or disable the changelog generation, so if it's not configured the changelog is not generated
        File changelogFile = getChangelogFile();

        if (Objects.isNull(changelogFile))
            logger.debug(COMMAND, "Changelog has not been configured or it has no destination path. Skipping the changelog generation.");
        else if (!state().getNewVersion())
            logger.debug(COMMAND, "No new version has been inferred so the changelog generation will be skipped.");
        else {
            logger.debug(COMMAND, "Building the changelog data model");

            Changelog changelog = new Changelog();
            state().setChangelog(changelog);

            // As of this version we can just pick the commits from the current release scope but in future versions,
            // when we need to have the 'Unreleased' fictional release plus, optionally, older releases,
            // (as per https://github.com/mooltiverse/nyx/issues/79) we'll need to walk the commit history just like the
            // Infer command does.

            // Create the timestamp string
            Date date = new Date(Long.valueOf(state().getTimestamp()).longValue());
            SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd");
            dateFormat.setTimeZone(TimeZone.getTimeZone(ZoneOffset.UTC));
            String dateString = dateFormat.format(date);

            // As of now we just have one release: the one being issued
            Changelog.Release release = new Changelog.Release(state().getVersion(), dateString);
            changelog.getReleases().add(release);

            // As of now we can just pick the commits from the current release scope.
            // We just need to distribute the commits among sections, which means filtering and translating
            // the sections if the user has configured them, or just use the commit 'type's as section names
            // if the user didn't map the sections
            for (Commit commit: state().getReleaseScope().getCommits()) {
                // Now we need to infer the commit type by using the commit message conventions
                Set commitTypes = new HashSet();
                if (!Objects.isNull(state().getConfiguration().getCommitMessageConventions().getItems())) {
                    logger.debug(COMMAND, "Trying to infer the commit type based on the commit message of commit '{}'", commit.getSHA());
                    for (Map.Entry cmcEntry: state().getConfiguration().getCommitMessageConventions().getItems().entrySet()) {
                        logger.debug(COMMAND, "Evaluating commit '{}' against message convention '{}'", commit.getSHA(), cmcEntry.getKey());                                
                        Matcher messageMatcher = Pattern.compile(cmcEntry.getValue().getExpression()).matcher(commit.getMessage().getFullMessage());
                        try {
                            // if the commit message matches multiple times we need to determine the commit type for all matches
                            while (messageMatcher.find()) {
                                logger.debug(COMMAND, "Commit message convention '{}' matches commit '{}'", cmcEntry.getKey(), commit.getSHA());
                                String commitType = messageMatcher.group("type");
                                commitTypes.add(commitType);
                                logger.debug(COMMAND, "The commit '{}' is of type '{}'", commit.getSHA(), commitType);
                            }
                        }
                        catch (IllegalArgumentException iae) {
                            // the regular expression doesn't match the name capturing group, no commit type is inferred
                            //throw new IllegalPropertyException(String.format("The regular expression '%s' defined for commit message convention '%s' does not define the 'type' named capturing group", cmcEntry.getValue().getExpression(), cmcEntry.getKey()), iae);
                        }
                        catch (IllegalStateException ise) {
                            throw new ReleaseException(String.format("Cannot infer the commit type for commit '%s' match operation failed for the regular expression '%s' from commit message convention '%s'", commit.getSHA(), cmcEntry.getValue().getExpression(), cmcEntry.getKey()), ise);
                        }
                    }
                }
                if (Objects.isNull(commitTypes) || commitTypes.isEmpty())
                    logger.debug(COMMAND, "Unable infer the 'type' for commit '{}'. The commit will not appear in the changelog.", commit.getSHA());
                else {
                    for (String commitType: commitTypes) {
                        // If the user has defined some sections mapping we need to map the commit type to those sections,
                        // otherwise the section will be the commit type
                        if (Objects.isNull(state().getConfiguration().getChangelog().getSections()) || state().getConfiguration().getChangelog().getSections().isEmpty()) {
                            logger.debug(COMMAND, "Changelog sections haven't been defined by user. Commit '{}' will appear in section '{}' (same as the commit type)", commit.getSHA(), commitType);
                            release.getSection(commitType, true).getCommits().add(commit);
                        }
                        else {
                            for (Map.Entry sectionEntry: state().getConfiguration().getChangelog().getSections().entrySet()) {
                                logger.debug(COMMAND, "Evaluating commit type '{}' against changelog section '{}'", commitType, sectionEntry.getKey());
                                if (Pattern.matches(sectionEntry.getValue(), commitType)) {
                                    logger.debug(COMMAND, "Expression '{}' for section '{}' successfully matches type '{}' so commit '{}' will appear under the '{}' section", sectionEntry.getValue(), sectionEntry.getKey(), commitType, commit.getSHA(), sectionEntry.getKey());
                                    release.getSection(sectionEntry.getKey(), true).getCommits().add(commit);
                                    break;
                                }
                                else {
                                    logger.debug(COMMAND, "Expression '{}' for section '{}' does not match type '{}'. Trying with next sections, if any.", sectionEntry.getValue(), sectionEntry.getKey(), commitType);
                                    continue;
                                }
                            }
                        }
                    }
                }
            }

            logger.debug(COMMAND, "Rendering the changelog");
            if (state().getConfiguration().getDryRun()) {
                logger.info(COMMAND, "Changelog rendering skipped due to dry run");
            }
            else {
                try {
                    StringWriter stringWriter = new StringWriter();
                    Templates.render(getChangelogTemplateReader(), changelog, stringWriter);
                    stringWriter.flush();
                    stringWriter.close();

                    String changelogBuffer = stringWriter.toString();

                    // if substitutions have been defined, let's apply them
                    if (!Objects.isNull(state().getConfiguration().getChangelog().getSubstitutions())) {
                        logger.debug(COMMAND, "Applying configured substitutions to the changelog");
                        for (Map.Entry substitutionEntry: state().getConfiguration().getChangelog().getSubstitutions().entrySet()) {
                            Pattern substitutionPattern = Pattern.compile(substitutionEntry.getKey());
                            Matcher substitutionMatcher = substitutionPattern.matcher(changelogBuffer);
                            StringBuffer buffer = new StringBuffer();
                            while (substitutionMatcher.find()) {
                                String substitutionValue = substitutionMatcher.group(1); // by the docs, only the group 1 is considered
                                // we need to detect the exact number of occurrences of '%s' and pass the Sprintf parameter that exact number of times
                                // since we don't have a 'count'-like method in String we use the split method for that, and we know the number of occurrences
                                // is the number of splits -1
                                int occurrences = substitutionEntry.getValue().split("%s").length-1;
                                if (occurrences < 0) {
                                    occurrences = 0;
                                }
                                Object[] replacementOccurrences = new Object[occurrences];
                                Arrays.fill(replacementOccurrences, substitutionValue);
                                substitutionMatcher.appendReplacement(buffer, String.format(substitutionEntry.getValue(), replacementOccurrences));
                            }
                            substitutionMatcher.appendTail(buffer);
                            changelogBuffer = buffer.toString();
                        }
                        logger.debug(COMMAND, "Configured substitutions have been applied to the changelog");
                    }

                    // now write to the actual destination file
                    if (Objects.isNull(state().getConfiguration().getChangelog().getAppend()) || state().getConfiguration().getChangelog().getAppend().isBlank()) {
                        logger.debug(COMMAND, "No append flag was defined for the changelog so the original file '{}', if any, will be overwritten", changelogFile.getAbsolutePath());
                        // overwrite the file
                        FileWriter fileWriter = new FileWriter(changelogFile);
                        fileWriter.write(changelogBuffer);
                        fileWriter.flush();
                        fileWriter.close();
                    }
                    else if ("tail".equalsIgnoreCase(state().getConfiguration().getChangelog().getAppend().trim())) {
                        logger.debug(COMMAND, "The '{}' append flag has been defined for the changelog so new contents will be appended to the end of the existing file '{}', if any", state().getConfiguration().getChangelog().getAppend(), changelogFile.getAbsolutePath());
                        // append to end
                        FileWriter fileWriter = new FileWriter(changelogFile, true);
                        fileWriter.write(changelogBuffer);
                        fileWriter.flush();
                        fileWriter.close();
                    }
                    else if ("head".equalsIgnoreCase(state().getConfiguration().getChangelog().getAppend().trim())) {
                        logger.debug(COMMAND, "The '{}' append flag has been defined for the changelog so new contents will be inserted to the beginning of the existing file '{}', if any", state().getConfiguration().getChangelog().getAppend(), changelogFile.getAbsolutePath());
                        // save the previous contents to a temporary buffer
                        StringWriter previousContentBuffer = new StringWriter();
                        if (changelogFile.exists()) {
                            FileReader reader = new FileReader(changelogFile);
                            reader.transferTo(previousContentBuffer);
                            reader.close();
                        }

                        // overwrite the file with the new content and append the previous contents to the file
                        FileWriter fileWriter = new FileWriter(changelogFile);
                        fileWriter.write(changelogBuffer);
                        fileWriter.write(previousContentBuffer.toString());
                        fileWriter.flush();
                        fileWriter.close();
                    } else {
                        throw new IllegalPropertyException(String.format("Illegal option '%s' has been defined for the changelog append option", state().getConfiguration().getChangelog().getAppend()));
                    }
                }
                catch (IOException ioe) {
                    throw new DataAccessException(String.format("Unable to render the changelog to file '%s'. Make sure the path to the file exists and can be written.", changelogFile.getAbsolutePath()), ioe);
                }

                logger.debug(COMMAND, "The changelog has been saved to '{}'", changelogFile.getAbsolutePath());
            }
        }
    }

    /**
     * Builds the configured assets.
     * 
     * @throws DataAccessException in case the configuration can't be loaded for some reason.
     * @throws IllegalPropertyException in case the configuration has some illegal options.
     * @throws GitException in case of unexpected issues when accessing the Git repository.
     * @throws ReleaseException if the task is unable to complete for reasons due to the release process.
     */
    private void buildAssets()
        throws DataAccessException, IllegalPropertyException, GitException, ReleaseException {
        // The only asset to build is the changelog
        buildChangelog();
    }

    /**
     * Applies the configured substitutions to project files.
     * 
     * @throws DataAccessException in case the configuration can't be loaded for some reason.
     * @throws IllegalPropertyException in case the configuration has some illegal options.
     * @throws GitException in case of unexpected issues when accessing the Git repository.
     * @throws ReleaseException if the task is unable to complete for reasons due to the release process.
     */
    private void applySubstitutions()
        throws DataAccessException, IllegalPropertyException, GitException, ReleaseException {
        logger.debug(COMMAND, "Applying substitutions to project files...");
        if (Objects.isNull(state().getConfiguration().getSubstitutions()) || Objects.isNull(state().getConfiguration().getSubstitutions().getEnabled()) || state().getConfiguration().getSubstitutions().getEnabled().isEmpty() || Objects.isNull(state().getConfiguration().getSubstitutions().getItems()) || state().getConfiguration().getSubstitutions().getItems().isEmpty()) {
            logger.debug(COMMAND, "No substitutions have been configured or enabled.");
            return;
        }

        for (String ruleName : state().getConfiguration().getSubstitutions().getEnabled()) {
            logger.debug(COMMAND, "Applying substitution rule '{}' to project files...", ruleName);
            if (Objects.isNull(state().getConfiguration().getSubstitutions().getItems().get(ruleName)))
                throw new IllegalPropertyException(String.format("Substitution rule '%s' is enabled but not configured", ruleName));
            if (Objects.isNull(state().getConfiguration().getSubstitutions().getItems().get(ruleName).getFiles()) || state().getConfiguration().getSubstitutions().getItems().get(ruleName).getFiles().isBlank() ||
                Objects.isNull(state().getConfiguration().getSubstitutions().getItems().get(ruleName).getMatch()) || state().getConfiguration().getSubstitutions().getItems().get(ruleName).getMatch().isBlank() ||
                Objects.isNull(state().getConfiguration().getSubstitutions().getItems().get(ruleName).getReplace()))
                throw new IllegalPropertyException(String.format("Substitution rule '%s' is enabled but not all of its required attributes have been set", ruleName));
            
            try {
                String globPattern = state().getConfiguration().getSubstitutions().getItems().get(ruleName).getFiles();

                String rootDirectory = System.getProperty("user.dir");
                if (!Objects.isNull(state().getConfiguration().getDirectory())) {
                    rootDirectory = state().getConfiguration().getDirectory();
                }

                List matches = new ArrayList();

                // First try to match a single file both as absolute or relative path
                // unless the path contains a glob character
                if (!globPattern.contains("*") && !globPattern.contains("?")) {
                    try {
                        File regularFile = Paths.get(globPattern).toFile();
                        if (regularFile.exists()) {
                            // If the regular file exists add it to the list of matches
                            logger.debug(COMMAND, "Expression '{}' matches a local absolute file path that is added to the substitution targets", globPattern);
                            matches.add(regularFile);
                        } else {
                            logger.debug(COMMAND, "Expression '{}' doesn't match a local absolute file path", globPattern);
                        }
                    } catch (InvalidPathException ipe) {
                        logger.debug(COMMAND, "Expression '{}' is not a valid absolute file path for a local file", globPattern);
                    }
                    try {
                        File regularFile = Paths.get(rootDirectory, globPattern).toFile();
                        if (regularFile.exists()) {
                            // If the regular file exists add it to the list of matches
                            logger.debug(COMMAND, "Expression '{}' matches a local relative file path that is added to the substitution targets", globPattern);
                            matches.add(regularFile);
                        } else {
                            logger.debug(COMMAND, "Expression '{}' doesn't match a local relative file path", globPattern);
                        }
                    } catch (InvalidPathException ipe) {
                        logger.debug(COMMAND, "Expression '{}' is not a valid relative file path for a local file", globPattern);
                    }
                }

                // Then try to find files matching the glob
                // A FileVisitor to walk the directory and select only the files whose names match the glob pattern
                FileVisitor matcherVisitor = new SimpleFileVisitor() {
                    @Override
                    public FileVisitResult visitFile(Path file, BasicFileAttributes attribs)
                        throws IOException {
                        FileSystem fs = FileSystems.getDefault();
                        PathMatcher matcher = fs.getPathMatcher("glob:"+globPattern);
                        if (matcher.matches(file)) {
                            matches.add(file.toFile());
                        }
                        return FileVisitResult.CONTINUE;
                    }
                };
                Files.walkFileTree(Paths.get(rootDirectory), matcherVisitor);

                if (matches.isEmpty()) {
                    logger.debug(COMMAND, "Expression '{}' doesn't match any file in directory '{}'", state().getConfiguration().getSubstitutions().getItems().get(ruleName).getFiles(), rootDirectory);
                }
                else {
                    for (File file: matches) {
                        logger.debug(COMMAND, "Applying substitutions in file '{}'...", file.getCanonicalPath());
                        FileReader fileReader = new FileReader(file);
                        StringWriter stringWriter = new StringWriter();
                        fileReader.transferTo(stringWriter);
                        fileReader.close();
                        String fileBuffer = stringWriter.toString();

                        String replacement = renderTemplate(state().getConfiguration().getSubstitutions().getItems().get(ruleName).getReplace());
                        Pattern substitutionPattern = Pattern.compile(state().getConfiguration().getSubstitutions().getItems().get(ruleName).getMatch());
                        Matcher substitutionMatcher = substitutionPattern.matcher(fileBuffer);
                        fileBuffer = substitutionMatcher.replaceAll(replacement);

                        FileWriter fileWriter = new FileWriter(file);
                        fileWriter.write(fileBuffer);
                        fileWriter.flush();
                        fileWriter.close();
                        logger.debug(COMMAND, "Substitutions have been applied in file '{}'.", file.getCanonicalPath());
                    }
                    logger.debug(COMMAND, "Substitution rule '{}' applied to project files.", ruleName);
                }
            }
            catch (InvalidPathException ipe) {
                throw new IllegalPropertyException(ipe);
            }
            catch (IOException ioe) {
                throw new DataAccessException(ioe);
            }
        }
    }

    /**
     * Reset the attributes store by this command into the internal state object.
     * This is required before running the command in order to make sure that the new execution is not affected
     * by a stale status coming from previous runs.
     * 
     * @throws DataAccessException in case the configuration can't be loaded for some reason.
     * @throws IllegalPropertyException in case the configuration has some illegal options.
     * 
     * @see #isUpToDate()
     * @see State#getInternals()
     */
    private void clearStateOutputAttributes() 
        throws DataAccessException, IllegalPropertyException {
        logger.debug(COMMAND, "Clearing the state from Make outputs");
        state().setChangelog(null);
    }

    /**
     * This method stores the state internal attributes used for up-to-date checks so that subsequent invocations
     * of the {@link #isUpToDate()} method can find them and determine if the command is already up to date.
     * 
     * This method is meant to be invoked at the end of a successful {@link #run()}.
     * 
     * @throws DataAccessException in case the configuration can't be loaded for some reason.
     * @throws IllegalPropertyException in case the configuration has some illegal options.
     * @throws GitException in case of unexpected issues when accessing the Git repository.
     * 
     * @see #isUpToDate()
     * @see State#getInternals()
     */
    private void storeStatusInternalAttributes()
        throws DataAccessException, IllegalPropertyException, GitException {
        logger.debug(COMMAND, "Storing the Make command internal attributes to the State");
        if (!state().getConfiguration().getDryRun()) {
            File changelogFile = getChangelogFile();
            putInternalAttribute(INTERNAL_INPUT_ATTRIBUTE_CHANGELOG_FILE, Objects.isNull(changelogFile) ? "null" : changelogFile.getAbsolutePath());
            putInternalAttribute(INTERNAL_INPUT_ATTRIBUTE_REPOSITORY_CURRENT_BRANCH, getCurrentBranch());
            putInternalAttribute(INTERNAL_INPUT_ATTRIBUTE_REPOSITORY_LAST_COMMIT, getLatestCommit());
            putInternalAttribute(INTERNAL_INPUT_ATTRIBUTE_STATE_INITIAL_COMMIT, state().getReleaseScope().getInitialCommit());
            putInternalAttribute(INTERNAL_INPUT_ATTRIBUTE_STATE_NEW_VERSION, state().getNewVersion());
            putInternalAttribute(INTERNAL_INPUT_ATTRIBUTE_VERSION, state().getVersion());
        }
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean isUpToDate()
        throws DataAccessException, IllegalPropertyException, GitException {
        logger.debug(COMMAND, "Checking whether the Make command is up to date");
        // Never up to date if this command hasn't stored a version yet into the state
        if (Objects.isNull(state().getVersion())) {
            logger.debug(COMMAND, "The Make command is not up to date because the internal state has no version yet");
            return false;
        }

        if (state().getNewVersion()) {
            File changelogFile = getChangelogFile();
            if (!Objects.isNull(changelogFile)) {
                // The command is never considered up to date when the configuration requires a changelog file but the state has no such object reference
                if (Objects.isNull(state().getChangelog())) {
                    logger.debug(COMMAND, "The Make command is not up to date because a changelog file has been configured ('{}') but the internal state has no changelog yet", changelogFile.getAbsolutePath());
                    return false;
                }
                // The command is never considered up to date when the changelog file hasn't been saved yet or it has changed
                if (!isInternalAttributeUpToDate(INTERNAL_INPUT_ATTRIBUTE_CHANGELOG_FILE, changelogFile.getAbsolutePath())) {
                    logger.debug(COMMAND, "The Make command is not up to date because a changelog file has been configured ('{}') but the configured path has changed", changelogFile.getAbsolutePath());
                    return false;
                }
                if (!changelogFile.exists()) {
                    logger.debug(COMMAND, "The Make command is not up to date because a changelog file has been configured ('{}') but the file does not exist", changelogFile.getAbsolutePath());
                    return false;
                }
            }
        }
        else {
            logger.debug(COMMAND, "No new version has been generated so up-to-date checks for the Make command in regards to the changelog file are skipped");
        }

        // The command is never considered up to date when the repository branch or last commit has changed
        if ((!isInternalAttributeUpToDate(INTERNAL_INPUT_ATTRIBUTE_REPOSITORY_CURRENT_BRANCH, getCurrentBranch())) || (!isInternalAttributeUpToDate(INTERNAL_INPUT_ATTRIBUTE_REPOSITORY_LAST_COMMIT, getLatestCommit())) || (!isInternalAttributeUpToDate(INTERNAL_INPUT_ATTRIBUTE_STATE_INITIAL_COMMIT, state().getReleaseScope().getInitialCommit()))) {
            logger.debug(COMMAND, "The Make command is not up to date because the range of commits or the current branch has changed");
            return false;
        }

        // Check if configuration parameters have changed
        boolean res = isInternalAttributeUpToDate(INTERNAL_INPUT_ATTRIBUTE_VERSION, state().getVersion()) &&
            isInternalAttributeUpToDate(INTERNAL_INPUT_ATTRIBUTE_STATE_NEW_VERSION, state().getNewVersion());
        if (res) {
            logger.debug(COMMAND, "The Make command is up to date");
        }
        else {
            logger.debug(COMMAND, "The Make command is not up to date because the configuration or the internal state has changed");
        }
        return res;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public State run()
        throws DataAccessException, IllegalPropertyException, GitException, ReleaseException {
        logger.debug(COMMAND, "Running the Make command...");

        clearStateOutputAttributes();

        buildAssets();

        applySubstitutions();

        storeStatusInternalAttributes();
        
        return state();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy