Github user ottobackwards commented on a diff in the pull request:

    https://github.com/apache/metron/pull/530#discussion_r129826663
  
    --- Diff: 
bundles-maven-plugin/src/main/java/org/apache/metron/BundleMojo.java ---
    @@ -0,0 +1,742 @@
    +/*
    + * Licensed to the Apache Software Foundation (ASF) under one or more
    + * contributor license agreements.  See the NOTICE file distributed with
    + * this work for additional information regarding copyright ownership.
    + * The ASF licenses this file to You 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 org.apache.metron;
    +
    +import java.io.File;
    +import java.io.IOException;
    +import java.text.SimpleDateFormat;
    +import java.util.Date;
    +import java.util.HashSet;
    +import java.util.List;
    +import java.util.Set;
    +import org.apache.maven.archiver.MavenArchiveConfiguration;
    +import org.apache.maven.archiver.MavenArchiver;
    +import org.apache.maven.artifact.Artifact;
    +import org.apache.maven.artifact.DependencyResolutionRequiredException;
    +import org.apache.maven.artifact.factory.ArtifactFactory;
    +import org.apache.maven.artifact.installer.ArtifactInstaller;
    +import org.apache.maven.artifact.metadata.ArtifactMetadataSource;
    +import org.apache.maven.artifact.repository.ArtifactRepository;
    +import org.apache.maven.artifact.repository.ArtifactRepositoryFactory;
    +import org.apache.maven.artifact.resolver.ArtifactCollector;
    +import org.apache.maven.artifact.resolver.ArtifactNotFoundException;
    +import org.apache.maven.artifact.resolver.ArtifactResolutionException;
    +import org.apache.maven.artifact.resolver.ArtifactResolver;
    +import org.apache.maven.plugin.AbstractMojo;
    +import org.apache.maven.plugin.MojoExecutionException;
    +import org.apache.maven.plugin.MojoFailureException;
    +import org.apache.maven.plugin.dependency.utils.DependencyStatusSets;
    +import org.apache.maven.plugin.dependency.utils.DependencyUtil;
    +import org.apache.maven.plugin.dependency.utils.filters.DestFileFilter;
    +import 
org.apache.maven.plugin.dependency.utils.resolvers.ArtifactsResolver;
    +import 
org.apache.maven.plugin.dependency.utils.resolvers.DefaultArtifactsResolver;
    +import 
org.apache.maven.plugin.dependency.utils.translators.ArtifactTranslator;
    +import 
org.apache.maven.plugin.dependency.utils.translators.ClassifierTypeTranslator;
    +import org.apache.maven.plugins.annotations.LifecyclePhase;
    +import org.apache.maven.plugins.annotations.Mojo;
    +import org.apache.maven.plugins.annotations.Parameter;
    +import org.apache.maven.plugins.annotations.ResolutionScope;
    +import org.apache.maven.project.MavenProject;
    +import org.apache.maven.execution.MavenSession;
    +import org.apache.maven.plugins.annotations.Component;
    +import org.apache.maven.project.MavenProjectHelper;
    +import 
org.apache.maven.shared.artifact.filter.collection.ArtifactFilterException;
    +import org.apache.maven.shared.artifact.filter.collection.ArtifactIdFilter;
    +import org.apache.maven.shared.artifact.filter.collection.ArtifactsFilter;
    +import org.apache.maven.shared.artifact.filter.collection.ClassifierFilter;
    +import org.apache.maven.shared.artifact.filter.collection.FilterArtifacts;
    +import org.apache.maven.shared.artifact.filter.collection.GroupIdFilter;
    +import org.apache.maven.shared.artifact.filter.collection.ScopeFilter;
    +import 
org.apache.maven.shared.artifact.filter.collection.ProjectTransitivityFilter;
    +import org.apache.maven.shared.artifact.filter.collection.TypeFilter;
    +import org.codehaus.plexus.archiver.ArchiverException;
    +import org.codehaus.plexus.archiver.jar.JarArchiver;
    +import org.codehaus.plexus.archiver.jar.ManifestException;
    +import org.codehaus.plexus.archiver.manager.ArchiverManager;
    +import org.codehaus.plexus.util.FileUtils;
    +import org.codehaus.plexus.util.StringUtils;
    +
    +/**
    + * Packages the current project as an Apache Metron Bundle Archive 
(BUNDLE).
    + * Apache Metron Bundles are based on Apache Nifi Archives (NAR)
    + *
    + * The following code is derived from maven-dependencies-plugin and
    + * maven-jar-plugin. The functionality of CopyDependenciesMojo and JarMojo 
was
    + * simplified to the use case of NarMojo.
    + *
    + */
    +@Mojo(name = "bundle", defaultPhase = LifecyclePhase.PACKAGE, threadSafe = 
false, requiresDependencyResolution = ResolutionScope.RUNTIME)
    +public class BundleMojo extends AbstractMojo {
    +
    +    private static final String[] DEFAULT_EXCLUDES = new 
String[]{"**/package.html"};
    +    private static final String[] DEFAULT_INCLUDES = new String[]{"**/**"};
    +
    +    private static final String BUILD_TIMESTAMP_FORMAT = 
"yyyy-MM-dd'T'HH:mm:ss'Z'";
    +
    +    /**
    +     * POM
    +     *
    +     */
    +    @Parameter(defaultValue = "${project}", readonly = true, required = 
true)
    +    protected MavenProject project;
    +
    +    @Parameter(defaultValue = "${session}", readonly = true, required = 
true)
    +    protected MavenSession session;
    +
    +    /**
    +     * List of files to include. Specified as fileset patterns.
    +     */
    +    @Parameter(property = "includes")
    +    protected String[] includes;
    +    /**
    +     * List of files to exclude. Specified as fileset patterns.
    +     */
    +    @Parameter(property = "excludes")
    +    protected String[] excludes;
    +    /**
    +     * Name of the generated NAR.
    +     *
    +     */
    +    @Parameter(alias = "bundleName", property = "bundle.finalName", 
defaultValue = "${project.build.finalName}", required = true)
    +    protected String finalName;
    +    /**
    +     * Name of the prefix for package identifiers like Nar-Id, where Nar 
is the identifier
    +     */
    +    @Parameter(alias = "packageIDPrefix", property = 
"bundle.packageIDPrefix", defaultValue = "Bundle", required = true)
    +    protected String packageIDPrefix;
    +
    +    /**
    +     * The Jar archiver.
    +     *
    +     * \@\component role="org.codehaus.plexus.archiver.Archiver" 
roleHint="jar"
    +     */
    +    @Component(role = org.codehaus.plexus.archiver.Archiver.class, hint = 
"jar")
    +    private JarArchiver jarArchiver;
    +    /**
    +     * The archive configuration to use.
    +     *
    +     * See <a
    +     * href="http://maven.apache.org/shared/maven-archiver/index.html";>the
    +     * documentation for Maven Archiver</a>.
    +     *
    +     */
    +    @Parameter(property = "archive")
    +    protected final MavenArchiveConfiguration archive = new 
MavenArchiveConfiguration();
    +    /**
    +     * Path to the default MANIFEST file to use. It will be used if
    +     * <code>useDefaultManifestFile</code> is set to <code>true</code>.
    +     *
    +     */
    +    @Parameter(property = "defaultManifestFiles", defaultValue = 
"${project.build.outputDirectory}/META-INF/MANIFEST.MF", readonly = true, 
required = true)
    +    protected File defaultManifestFile;
    +
    +    /**
    +     * Set this to <code>true</code> to enable the use of the
    +     * <code>defaultManifestFile</code>.
    +     *
    +     * @since 2.2
    +     */
    +    @Parameter(property = "bundle.useDefaultManifestFile", defaultValue = 
"false")
    +    protected boolean useDefaultManifestFile;
    +
    +    @Component
    +    protected MavenProjectHelper projectHelper;
    +
    +    /**
    +     * Whether creating the archive should be forced.
    +     *
    +     */
    +    @Parameter(property = "bundle.forceCreation", defaultValue = "false")
    +    protected boolean forceCreation;
    +
    +    /**
    +     * Classifier to add to the artifact generated. If given, the artifact 
will
    +     * be an attachment instead.
    +     *
    +     */
    +    @Parameter(property = "classifier")
    +    protected String classifier;
    +
    +    @Component
    +    protected ArtifactInstaller installer;
    +
    +    @Component
    +    protected ArtifactRepositoryFactory repositoryFactory;
    +
    +    /**
    +     * This only applies if the classifier parameter is used.
    +     *
    +     */
    +    @Parameter(property = "mdep.failOnMissingClassifierArtifact", 
defaultValue = "true", required = false)
    +    protected boolean failOnMissingClassifierArtifact = true;
    +
    +    /**
    +     * Comma Separated list of Types to include. Empty String indicates 
include
    +     * everything (default).
    +     *
    +     */
    +    @Parameter(property = "includeTypes", required = false)
    +    protected String includeTypes;
    +
    +    /**
    +     * Comma Separated list of Types to exclude. Empty String indicates 
don't
    +     * exclude anything (default).
    +     *
    +     */
    +    @Parameter(property = "excludeTypes", required = false)
    +    protected String excludeTypes;
    +
    +    /**
    +     * Scope to include. An Empty string indicates all scopes (default).
    +     *
    +     */
    +    @Parameter(property = "includeScope", required = false)
    +    protected String includeScope;
    +
    +    /**
    +     * Scope to exclude. An Empty string indicates no scopes (default).
    +     *
    +     */
    +    @Parameter(property = "excludeScope", required = false)
    +    protected String excludeScope;
    +
    +    /**
    +     * Comma Separated list of Classifiers to include. Empty String 
indicates
    +     * include everything (default).
    +     *
    +     */
    +    @Parameter(property = "includeClassifiers", required = false)
    +    protected String includeClassifiers;
    +
    +    /**
    +     * Comma Separated list of Classifiers to exclude. Empty String 
indicates
    +     * don't exclude anything (default).
    +     *
    +     */
    +    @Parameter(property = "excludeClassifiers", required = false)
    +    protected String excludeClassifiers;
    +
    +    /**
    +     * Specify classifier to look for. Example: sources
    +     *
    +     */
    +    @Parameter(property = "classifier", required = false)
    +    protected String copyDepClassifier;
    +
    +    /**
    +     * Specify type to look for when constructing artifact based on 
classifier.
    +     * Example: java-source,jar,war,bundle
    +     *
    +     */
    +    @Parameter(property = "type", required = false, defaultValue = 
"bundle")
    +    protected String type;
    +
    +    /**
    +     * Comma separated list of Artifact names too exclude.
    +     *
    +     */
    +    @Parameter(property = "excludeArtifacts", required = false)
    +    protected String excludeArtifactIds;
    +
    +    /**
    +     * Comma separated list of Artifact names to include.
    +     *
    +     */
    +    @Parameter(property = "includeArtifacts", required = false)
    +    protected String includeArtifactIds;
    +
    +    /**
    +     * Comma separated list of GroupId Names to exclude.
    +     *
    +     */
    +    @Parameter(property = "excludeArtifacts", required = false)
    +    protected String excludeGroupIds;
    +
    +    /**
    +     * Comma separated list of GroupIds to include.
    +     *
    +     */
    +    @Parameter(property = "includeGroupIds", required = false)
    +    protected String includeGroupIds;
    +
    +    /**
    +     * Directory to store flag files
    +     *
    +     */
    +    @Parameter(property = "markersDirectory", required = false, 
defaultValue = "${project.build.directory}/dependency-maven-plugin-markers")
    +    protected File markersDirectory;
    +
    +    /**
    +     * Overwrite release artifacts
    +     *
    +     */
    +    @Parameter(property = "overWriteReleases", required = false)
    +    protected boolean overWriteReleases;
    +
    +    /**
    +     * Overwrite snapshot artifacts
    +     *
    +     */
    +    @Parameter(property = "overWriteSnapshots", required = false)
    +    protected boolean overWriteSnapshots;
    +
    +    /**
    +     * Overwrite artifacts that don't exist or are older than the source.
    +     *
    +     */
    +    @Parameter(property = "overWriteIfNewer", required = false, 
defaultValue = "true")
    +    protected boolean overWriteIfNewer;
    +
    +    @Parameter(property = "projectBuildDirectory", required = false, 
defaultValue = "${project.build.directory}")
    +    protected File projectBuildDirectory;
    +
    +    /**
    +     * Used to look up Artifacts in the remote repository.
    +     */
    +    @Component
    +    protected ArtifactFactory factory;
    +
    +    /**
    +     * Used to look up Artifacts in the remote repository.
    +     *
    +     */
    +    @Component
    +    protected ArtifactResolver resolver;
    +
    +    /**
    +     * Artifact collector, needed to resolve dependencies.
    +     *
    +     */
    +    @Component(role = 
org.apache.maven.artifact.resolver.ArtifactCollector.class)
    +    protected ArtifactCollector artifactCollector;
    +
    +    @Component(role = 
org.apache.maven.artifact.metadata.ArtifactMetadataSource.class)
    +    protected ArtifactMetadataSource artifactMetadataSource;
    +
    +    /**
    +     * Location of the local repository.
    +     *
    +     */
    +    @Parameter(property = "localRepository", required = true, readonly = 
true)
    +    protected ArtifactRepository local;
    +
    +    /**
    +     * List of Remote Repositories used by the resolver
    +     *
    +     */
    +    @Parameter(property = "project.remoteArtifactRepositories", required = 
true, readonly = true)
    +    protected List remoteRepos;
    +
    +    /**
    +     * To look up Archiver/UnArchiver implementations
    +     *
    +     */
    +    @Component
    +    protected ArchiverManager archiverManager;
    +
    +    /**
    +     * Contains the full list of projects in the reactor.
    +     *
    +     */
    +    @Parameter(property = "reactorProjects", required = true, readonly = 
true)
    +    protected List reactorProjects;
    +
    +    /**
    +     * If the plugin should be silent.
    +     *
    +     */
    +    @Parameter(property = "silent", required = false, defaultValue = 
"false")
    +    public boolean silent;
    +
    +    /**
    +     * Output absolute filename for resolved artifacts
    +     *
    +     */
    +    @Parameter(property = "outputAbsoluteArtifactFilename", defaultValue = 
"false", required = false)
    +    protected boolean outputAbsoluteArtifactFilename;
    +
    +    /* The values to use for populating the Nar-Group, Nar-Id, and 
Nar-Version in the MANIFEST file. By default
    +     * these values will be set to the standard Maven project equivalents, 
but they may be overridden through properties.
    +     *
    +     * For example if the pom.xml for the nifi-test-bundle contained the 
following:
    +     *
    +     *    <groupId>org.apache.nifi</groupId>
    +     *    <artifactId>nifi-test-nar</artifactId>
    +     *    <version>1.0</version>
    +     *
    +     *    <properties>
    +     *       <bundleGroup>org.apache.nifi.overridden</bundleGroup>
    +     *       <bundleId>nifi-overridden-test-nar</bundleId>
    +     *       <bundleVersion>2.0</bundleVersion>
    +     *   </properties>
    +     *
    +     * It would produce a MANIFEST with:
    +     *
    +     *   Nar-Id: nifi-overridden-test-nar
    +     *   Nar-Group: org.apache.nifi.overridden
    +     *   Nar-Version: 2.0
    +     *
    +    */
    +
    +    @Parameter(property = "bundleGroup", defaultValue = 
"${project.groupId}", required = true)
    +    protected String bundleGroup;
    +
    +    @Parameter(property = "bundleId", defaultValue = 
"${project.artifactId}", required = true)
    +    protected String bundleId;
    +
    +    @Parameter(property = "bundleVersion", defaultValue = 
"${project.version}", required = true)
    +    protected String bundleVersion;
    +
    +    @Parameter(property = "bundleDependencyGroup", required = false)
    +    protected String bundleDependencyGroup = null;
    +
    +    @Parameter(property = "bundleDependencyId", required = false)
    +    protected String bundleDependencyId = null;
    +
    +    @Parameter(property = "bundleDependencyVersion", required = false)
    +    protected String bundleDependencyVersion = null;
    +
    +
    +    /**
    +     * Build info to be populated in MANIFEST.
    +     */
    +
    +    @Parameter(property = "buildTag", defaultValue = "${project.scm.tag}", 
required = false)
    +    protected String buildTag;
    +
    +    @Parameter(property = "buildBranch", defaultValue = "${buildBranch}", 
required = false)
    +    protected String buildBranch;
    +
    +    @Parameter(property = "buildRevision", defaultValue = 
"${buildRevision}", required = false)
    +    protected String buildRevision;
    +
    +    /**
    +     * Allows a Bundle to specify if it's resources should be cloned when 
a component that depends on this Bundle
    +     * is performing class loader isolation.
    +     */
    +    @Parameter(property = "cloneDuringInstanceClassLoading", defaultValue 
= "false", required = false)
    +    protected boolean cloneDuringInstanceClassLoading;
    +
    +
    +    @Override
    +    public void execute() throws MojoExecutionException, 
MojoFailureException {
    +        copyDependencies();
    +        makeNar();
    +    }
    +
    +    private void copyDependencies() throws MojoExecutionException {
    +        DependencyStatusSets dss = 
getDependencySets(this.failOnMissingClassifierArtifact);
    +        Set artifacts = dss.getResolvedDependencies();
    +
    +        for (Object artifactObj : artifacts) {
    +            copyArtifact((Artifact) artifactObj);
    +        }
    +
    +        artifacts = dss.getSkippedDependencies();
    +        for (Object artifactOjb : artifacts) {
    +            Artifact artifact = (Artifact) artifactOjb;
    +            getLog().info(artifact.getFile().getName() + " already exists 
in destination.");
    +        }
    +    }
    +
    +    protected void copyArtifact(Artifact artifact) throws 
MojoExecutionException {
    +        String destFileName = 
DependencyUtil.getFormattedFileName(artifact, false);
    +        final File destDir = 
DependencyUtil.getFormattedOutputDirectory(false, false, false, false, false, 
getDependenciesDirectory(), artifact);
    +        final File destFile = new File(destDir, destFileName);
    +        copyFile(artifact.getFile(), destFile);
    +    }
    +
    +    protected Artifact getResolvedPomArtifact(Artifact artifact) {
    +        Artifact pomArtifact = 
this.factory.createArtifact(artifact.getGroupId(), artifact.getArtifactId(), 
artifact.getVersion(), "", "pom");
    +        // Resolve the pom artifact using repos
    +        try {
    +            this.resolver.resolve(pomArtifact, this.remoteRepos, 
this.local);
    +        } catch (ArtifactResolutionException | ArtifactNotFoundException 
e) {
    +            getLog().info(e.getMessage());
    +        }
    +        return pomArtifact;
    +    }
    +
    +    protected ArtifactsFilter getMarkedArtifactFilter() {
    +        return new DestFileFilter(this.overWriteReleases, 
this.overWriteSnapshots, this.overWriteIfNewer, false, false, false, false, 
false, getDependenciesDirectory());
    +    }
    +
    +    protected DependencyStatusSets getDependencySets(boolean 
stopOnFailure) throws MojoExecutionException {
    +        // add filters in well known order, least specific to most specific
    +        FilterArtifacts filter = new FilterArtifacts();
    +
    +        filter.addFilter(new 
ProjectTransitivityFilter(project.getDependencyArtifacts(), false));
    +        filter.addFilter(new ScopeFilter(this.includeScope, 
this.excludeScope));
    +        filter.addFilter(new TypeFilter(this.includeTypes, 
this.excludeTypes));
    +        filter.addFilter(new ClassifierFilter(this.includeClassifiers, 
this.excludeClassifiers));
    +        filter.addFilter(new GroupIdFilter(this.includeGroupIds, 
this.excludeGroupIds));
    +        filter.addFilter(new ArtifactIdFilter(this.includeArtifactIds, 
this.excludeArtifactIds));
    +
    +        // explicitly filter our bundle dependencies as defined by type
    +        filter.addFilter(new TypeFilter("", type));
    +
    +        // start with all artifacts.
    +        Set artifacts = project.getArtifacts();
    +
    +        // perform filtering
    +        try {
    +            artifacts = filter.filter(artifacts);
    +        } catch (ArtifactFilterException e) {
    +            throw new MojoExecutionException(e.getMessage(), e);
    +        }
    +
    +        // transform artifacts if classifier is set
    +        final DependencyStatusSets status;
    +        if (StringUtils.isNotEmpty(copyDepClassifier)) {
    +            status = getClassifierTranslatedDependencies(artifacts, 
stopOnFailure);
    +        } else {
    +            status = filterMarkedDependencies(artifacts);
    +        }
    +
    +        return status;
    +    }
    +
    +    protected DependencyStatusSets getClassifierTranslatedDependencies(Set 
artifacts, boolean stopOnFailure) throws MojoExecutionException {
    +        Set unResolvedArtifacts = new HashSet();
    +        Set resolvedArtifacts = artifacts;
    +        DependencyStatusSets status = new DependencyStatusSets();
    +
    +        // possibly translate artifacts into a new set of artifacts based 
on the
    +        // classifier and type
    +        // if this did something, we need to resolve the new artifacts
    +        if (StringUtils.isNotEmpty(copyDepClassifier)) {
    +            ArtifactTranslator translator = new 
ClassifierTypeTranslator(this.copyDepClassifier, this.type, this.factory);
    +            artifacts = translator.translate(artifacts, getLog());
    +
    +            status = filterMarkedDependencies(artifacts);
    +
    +            // the unskipped artifacts are in the resolved set.
    +            artifacts = status.getResolvedDependencies();
    +
    +            // resolve the rest of the artifacts
    +            ArtifactsResolver artifactsResolver = new 
DefaultArtifactsResolver(this.resolver, this.local,
    +                    this.remoteRepos, stopOnFailure);
    +            resolvedArtifacts = artifactsResolver.resolve(artifacts, 
getLog());
    +
    +            // calculate the artifacts not resolved.
    +            unResolvedArtifacts.addAll(artifacts);
    +            unResolvedArtifacts.removeAll(resolvedArtifacts);
    +        }
    +
    +        // return a bean of all 3 sets.
    +        status.setResolvedDependencies(resolvedArtifacts);
    +        status.setUnResolvedDependencies(unResolvedArtifacts);
    +
    +        return status;
    +    }
    +
    +    protected DependencyStatusSets filterMarkedDependencies(Set artifacts) 
throws MojoExecutionException {
    +        // remove files that have markers already
    +        FilterArtifacts filter = new FilterArtifacts();
    +        filter.clearFilters();
    +        filter.addFilter(getMarkedArtifactFilter());
    +
    +        Set unMarkedArtifacts;
    +        try {
    +            unMarkedArtifacts = filter.filter(artifacts);
    +        } catch (ArtifactFilterException e) {
    +            throw new MojoExecutionException(e.getMessage(), e);
    +        }
    +
    +        // calculate the skipped artifacts
    +        Set skippedArtifacts = new HashSet();
    +        skippedArtifacts.addAll(artifacts);
    +        skippedArtifacts.removeAll(unMarkedArtifacts);
    +
    +        return new DependencyStatusSets(unMarkedArtifacts, null, 
skippedArtifacts);
    +    }
    +
    +    protected void copyFile(File artifact, File destFile) throws 
MojoExecutionException {
    +        try {
    +            getLog().info("Copying " + 
(this.outputAbsoluteArtifactFilename ? artifact.getAbsolutePath() : 
artifact.getName()) + " to " + destFile);
    +            FileUtils.copyFile(artifact, destFile);
    +        } catch (Exception e) {
    +            throw new MojoExecutionException("Error copying artifact from 
" + artifact + " to " + destFile, e);
    +        }
    +    }
    +
    +    private File getClassesDirectory() {
    +        final File outputDirectory = projectBuildDirectory;
    +        return new File(outputDirectory, "classes");
    +    }
    +
    +    private File getDependenciesDirectory() {
    +        return new File(getClassesDirectory(), 
"META-INF/bundled-dependencies");
    +    }
    +
    +    private void makeNar() throws MojoExecutionException {
    --- End diff --
    
    done


---
If your project is set up for it, you can reply to this email and have your
reply appear on GitHub as well. If your project does not have this feature
enabled and wishes so, or if the feature is enabled but not working, please
contact infrastructure at infrastruct...@apache.org or file a JIRA ticket
with INFRA.
---

Reply via email to