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

com.metaeffekt.mirror.plugin.DataMirrorMojo Maven / Gradle / Ivy

There is a newer version: 0.132.0
Show newest version
/*
 * Copyright 2021-2024 the original author or authors.
 *
 * 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.metaeffekt.mirror.plugin;

import com.metaeffekt.artifact.analysis.utils.StringUtils;
import com.metaeffekt.mirror.download.Download;
import com.metaeffekt.mirror.download.advisor.*;
import com.metaeffekt.mirror.download.nvd.CpeDictionaryDownload;
import com.metaeffekt.mirror.download.nvd.NvdCpeApiDownload;
import com.metaeffekt.mirror.download.nvd.NvdCveApiDownload;
import com.metaeffekt.mirror.download.nvd.NvdDownload;
import com.metaeffekt.mirror.download.other.CisaKevDownload;
import com.metaeffekt.mirror.download.other.EolDownload;
import com.metaeffekt.mirror.download.other.EpssDownload;
import com.metaeffekt.mirror.index.Index;
import com.metaeffekt.mirror.index.advisor.*;
import com.metaeffekt.mirror.index.nvd.*;
import com.metaeffekt.mirror.index.other.EolIndex;
import com.metaeffekt.mirror.index.other.EpssIndex;
import com.metaeffekt.mirror.index.other.KevIndex;
import com.metaeffekt.mirror.initializer.*;
import org.apache.maven.plugin.AbstractMojo;
import org.apache.maven.plugin.MojoFailureException;
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.project.MavenProject;

import java.io.File;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;

@Mojo(name = "data-mirror", defaultPhase = LifecyclePhase.PROCESS_RESOURCES)
public class DataMirrorMojo extends AbstractMojo {

    @Parameter(required = true)
    private File mirrorDirectory;

    @Parameter
    private String proxyScheme, proxyHost, proxyUsername, proxyPassword;
    @Parameter
    private Integer proxyPort;

    @Parameter
    private DownloadInitializer msrcDownload;

    @Deprecated
    @Parameter
    private DownloadInitializer msrcCsvDownload;

    @Parameter
    private DownloadInitializer msrcSecurityUpdateGuideDownload;

    /**
     * Deprecated: use {@link #nvdCpeDownload} instead.
     */
    @Parameter
    @Deprecated
    private DownloadInitializer cpeDictionaryDownload;

    @Parameter
    private NvdApiDownloadInitializer nvdCpeDownload;

    @Parameter
    private DownloadInitializer certSeiDownload;

    @Parameter
    private DownloadInitializer certFrDownload;

    @Parameter
    private DownloadInitializer certEuDownload;

    @Parameter
    private DownloadInitializer cisaKevDownload;

    @Parameter
    private DownloadInitializer epssDownload;

    @Parameter
    private DownloadInitializer eolDownload;

    /**
     * Deprecated: use {@link #nvdCveDownload} instead.
     */
    @Parameter
    @Deprecated
    private DownloadInitializer nvdLegacyDownload;

    @Parameter
    private NvdApiDownloadInitializer nvdCveDownload;

    @Parameter
    private GitDownloadInitializer githubAdvisorDownload;


    @Parameter
    private IndexInitializer certSeiAdvisorIndex;


    @Parameter
    private IndexInitializer certEuAdvisorIndex;

    @Parameter
    private IndexInitializer certFrAdvisorIndex;

    /**
     * Deprecated: use {@link #nvdCpeIndex} instead.
     */
    @Parameter
    @Deprecated
    private IndexInitializer cpeDictionaryIndex;

    /**
     * Deprecated: use {@link #nvdCpeVendorProductIndex} instead.
     */
    @Parameter
    @Deprecated
    private IndexInitializer cpeDictionaryVendorProductIndex;

    @Parameter
    private IndexInitializer nvdCpeIndex;

    @Parameter
    private IndexInitializer nvdCpeVendorProductIndex;

    @Parameter
    private IndexInitializer msrcProductIndex;

    @Parameter
    private IndexInitializer msrcAdvisorIndex;

    @Parameter
    private IndexInitializer msrcKbChainIndex;

    /**
     * Deprecated: use {@link #nvdVulnerabilityIndex} instead.
     */
    @Parameter
    @Deprecated
    private IndexInitializer nvdLegacyVulnerabilityIndex;

    @Parameter
    private IndexInitializer nvdVulnerabilityIndex;

    @Parameter
    private IndexInitializer githubAdvisorIndex;

    @Parameter
    private IndexInitializer kevIndex;

    @Parameter
    private IndexInitializer epssIndex;

    @Parameter
    private IndexInitializer eolIndex;

    @Parameter(defaultValue = "${project}", required = true, readonly = true)
    private MavenProject project;

    public MavenProject getProject() {
        return project;
    }

    @Override
    public void execute() throws MojoFailureException {
        getLog().info("Starting mirroring process");

        final Map downloadResults = processDownload();
        final Map indexResults = processIndex();
        logResults(downloadResults, indexResults);

        if (hasProcessFailed(downloadResults, indexResults)) {
            throw new MojoFailureException("Mirroring process completed with errors - see log for details");
        }

        getLog().info("Mirroring process completed successfully");
    }

    private Map processDownload() {
        final Map mirrorResults = new LinkedHashMap<>();
        final List downloads = buildDownloaders();

        for (Download download : downloads) {
            // getLog().info("");
            // getLog().info("Downloading [" + Download.getDirectoryIdentifier(download.getClass()) + "]");

            final MirrorResult result = mirrorResults.computeIfAbsent(download, d -> new MirrorResult());

            try {
                download.performDownloadIfRequired();
                // getLog().info("Completed download [" + Download.getDirectoryIdentifier(download.getClass()) + "]");
            } catch (Exception e) {
                result.exception = e;
                getLog().error("Failed on download " + download.getDownloadIntoDirectory().getName(), e);
            } finally {
                result.stop();
                getLog().info("");
            }
        }

        return mirrorResults;
    }

    private Map processIndex() {
        final Map mirrorResults = new LinkedHashMap<>();
        final List indexes = buildIndexes();

        for (Index index : indexes) {
            // getLog().info("");
            // getLog().info("Indexing [" + Index.getDirectoryIdentifier(index.getClass()) + "]");

            final MirrorResult result = mirrorResults.computeIfAbsent(index, d -> new MirrorResult());

            try {
                index.createIndexIfRequired();
                // getLog().info("Completed indexing [" + Index.getDirectoryIdentifier(index.getClass()) + "]");
            } catch (Exception e) {
                result.exception = e;
                getLog().error("Failed on download " + index.getClass().getSimpleName(), e);
            } finally {
                result.stop();
                getLog().info("");
            }
        }

        return mirrorResults;
    }

    private List buildDownloaders() {
        final List downloads = new ArrayList<>();

        if (isActive(msrcDownload)) {
            applyProxySettings(msrcDownload);
            final MsrcDownload download = new MsrcDownload(mirrorDirectory);
            msrcDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(msrcCsvDownload)) {
            applyProxySettings(msrcCsvDownload);
            final MsrcManualCsvDownload download = new MsrcManualCsvDownload(mirrorDirectory);
            msrcCsvDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(msrcSecurityUpdateGuideDownload)) {
            applyProxySettings(msrcSecurityUpdateGuideDownload);
            final MsrcSecurityGuideDownload download = new MsrcSecurityGuideDownload(mirrorDirectory);
            msrcSecurityUpdateGuideDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(cpeDictionaryDownload)) {
            applyProxySettings(cpeDictionaryDownload);
            final CpeDictionaryDownload download = new CpeDictionaryDownload(mirrorDirectory);
            cpeDictionaryDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(nvdCpeDownload)) {
            applyProxySettings(nvdCpeDownload);
            final NvdCpeApiDownload download = new NvdCpeApiDownload(mirrorDirectory);
            nvdCpeDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(certSeiDownload)) {
            applyProxySettings(certSeiDownload);
            final CertSeiDownload download = new CertSeiDownload(mirrorDirectory);
            certSeiDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(certFrDownload)) {
            applyProxySettings(certFrDownload);
            final CertFrDownload download = new CertFrDownload(mirrorDirectory);
            certFrDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(certEuDownload)) {
            applyProxySettings(certEuDownload);
            final CertEuDownload download = new CertEuDownload(mirrorDirectory);
            certEuDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(cisaKevDownload)) {
            applyProxySettings(cisaKevDownload);
            final CisaKevDownload download = new CisaKevDownload(mirrorDirectory);
            cisaKevDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(epssDownload)) {
            applyProxySettings(epssDownload);
            final EpssDownload download = new EpssDownload(mirrorDirectory);
            epssDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(eolDownload)) {
            applyProxySettings(eolDownload);
            final EolDownload download = new EolDownload(mirrorDirectory);
            eolDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(githubAdvisorDownload)) {
            applyProxySettings(githubAdvisorDownload);
            final GhsaDownload download = new GhsaDownload(mirrorDirectory);
            githubAdvisorDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(nvdLegacyDownload)) {
            applyProxySettings(nvdLegacyDownload);
            final NvdDownload download = new NvdDownload(mirrorDirectory);
            nvdLegacyDownload.apply(download);
            downloads.add(download);
        }

        if (isActive(nvdCveDownload)) {
            applyProxySettings(nvdCveDownload);
            final NvdCveApiDownload download = new NvdCveApiDownload(mirrorDirectory);
            nvdCveDownload.apply(download);
            downloads.add(download);
        }

        if (!downloads.isEmpty()) {
            getLog().info("Downloads:");
            for (Download download : downloads) {
                getLog().info(" - " + Download.getDirectoryIdentifier(download.getClass()));
            }
            getLog().info("");
        }

        return downloads;
    }

    private List buildIndexes() {
        final List indexes = new ArrayList<>();

        if (isActive(certSeiAdvisorIndex)) {
            final CertSeiAdvisorIndex index = new CertSeiAdvisorIndex(mirrorDirectory);
            certSeiAdvisorIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(certFrAdvisorIndex)) {
            final CertFrAdvisorIndex index = new CertFrAdvisorIndex(mirrorDirectory);
            certFrAdvisorIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(certEuAdvisorIndex)) {
            final CertEuAdvisorIndex index = new CertEuAdvisorIndex(mirrorDirectory);
            certEuAdvisorIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(cpeDictionaryIndex)) {
            final CpeDictionaryIndex index = new CpeDictionaryIndex(mirrorDirectory);
            cpeDictionaryIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(cpeDictionaryVendorProductIndex)) {
            final CpeDictionaryVendorProductIndex index = new CpeDictionaryVendorProductIndex(mirrorDirectory);
            cpeDictionaryVendorProductIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(nvdCpeIndex)) {
            final NvdCpeApiIndex index = new NvdCpeApiIndex(mirrorDirectory);
            nvdCpeIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(nvdCpeVendorProductIndex)) {
            final NvdCpeApiVendorProductIndex index = new NvdCpeApiVendorProductIndex(mirrorDirectory);
            nvdCpeVendorProductIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(msrcProductIndex)) {
            final MsrcProductIndex index = new MsrcProductIndex(mirrorDirectory);
            msrcProductIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(msrcAdvisorIndex)) {
            final MsrcAdvisorIndex index = new MsrcAdvisorIndex(mirrorDirectory);
            msrcAdvisorIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(msrcKbChainIndex)) {
            final MsrcKbChainIndex index = new MsrcKbChainIndex(mirrorDirectory);
            msrcKbChainIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(githubAdvisorIndex)) {
            final GhsaAdvisorIndex index = new GhsaAdvisorIndex(mirrorDirectory);
            githubAdvisorIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(kevIndex)) {
            final KevIndex index = new KevIndex(mirrorDirectory);
            kevIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(epssIndex)) {
            final EpssIndex index = new EpssIndex(mirrorDirectory);
            epssIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(eolIndex)) {
            final EolIndex index = new EolIndex(mirrorDirectory);
            eolIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(nvdLegacyVulnerabilityIndex)) {
            final NvdVulnerabilityIndex index = new NvdVulnerabilityIndex(mirrorDirectory);
            nvdLegacyVulnerabilityIndex.apply(index);
            indexes.add(index);
        }

        if (isActive(nvdVulnerabilityIndex)) {
            final NvdCveApiIndex index = new NvdCveApiIndex(mirrorDirectory);
            nvdVulnerabilityIndex.apply(index);
            indexes.add(index);
        }

        if (!indexes.isEmpty()) {
            getLog().info("Indexes:");
            for (Index index : indexes) {
                getLog().info(" - " + Index.getDirectoryIdentifier(index.getClass()));
            }
            getLog().info("");
        }

        return indexes;
    }

    /**
     * Applies the global proxy settings to a specified {@link DownloadInitializer} instance.
* The settings are applied only if none of the fields in the initializer are already set; * if any of the fields in the initializer are set, no changes are made. * * @param initializer the {@link DownloadInitializer} instance to which the proxy settings should be applied */ private void applyProxySettings(DownloadInitializer initializer) { if (initializer.scheme != null || initializer.host != null || initializer.port != null || initializer.username != null || initializer.password != null) { getLog().info("Downloader using initializer [" + initializer.getClass().getSimpleName() + "] already has proxy settings specified, ignoring global proxy settings"); return; } if (isValidProxySetting(proxyScheme)) initializer.scheme = proxyScheme; if (isValidProxySetting(proxyHost)) initializer.host = proxyHost; if (isValidProxySetting(proxyPort)) initializer.port = proxyPort; if (isValidProxySetting(proxyUsername)) initializer.username = proxyUsername; if (isValidProxySetting(proxyPassword)) initializer.password = proxyPassword; } /** * Determines whether the specified value is a valid proxy setting. *

* To be considered a valid proxy setting, the following conditions must be met: *

    *
  • The value should not be null.
  • *
  • If the value is a string, it must not be: *
      *
    • an empty string,
    • *
    • "null" (ignoring case),
    • *
    • "none" (ignoring case).
    • *
    *
  • *
  • If the value is an integer, it must not be -1.
  • *
* If the value is of any other datatype, then it is always considered as invalid proxy setting. * * @param value the value to be checked. * @return true if the value is a valid proxy setting according to the rules defined above, false otherwise. */ private boolean isValidProxySetting(Object value) { if (value == null) return false; if (value instanceof String) { final String str = (String) value; if (StringUtils.isEmpty(str)) return false; if (str.equalsIgnoreCase("null")) return false; if (str.equalsIgnoreCase("none")) return false; } else if (value instanceof Integer) { final Integer i = (Integer) value; if (i == -1) return false; } else { return false; } return true; } private boolean isActive(MirrorInitializer initializer) { return initializer != null && initializer.active; } private boolean hasProcessFailed(Map downloadResults, Map indexResults) { for (MirrorResult result : downloadResults.values()) { if (result.hasFailed()) { return true; } } for (MirrorResult result : indexResults.values()) { if (result.hasFailed()) { return true; } } return false; } private void logResults(Map downloadResults, Map indexResults) { getLog().info("Mirror results:"); if (!downloadResults.isEmpty()) { getLog().info("Download:"); for (Map.Entry entry : downloadResults.entrySet()) { entry.getValue().logSummary(Download.getDirectoryIdentifier(entry.getKey().getClass())); } getLog().info(""); } if (!indexResults.isEmpty()) { getLog().info("Index:"); for (Map.Entry entry : indexResults.entrySet()) { entry.getValue().logSummary(Index.getDirectoryIdentifier(entry.getKey().getClass())); } getLog().info(""); } if (!downloadResults.isEmpty()) { for (Map.Entry entry : downloadResults.entrySet()) { if (entry.getValue().hasFailed()) { getLog().error(entry.getValue().exception); getLog().error(""); } } } if (!indexResults.isEmpty()) { for (Map.Entry entry : indexResults.entrySet()) { if (entry.getValue().hasFailed()) { getLog().error(entry.getValue().exception); getLog().error(""); } } } } private class MirrorResult { public Exception exception; private final long start = System.currentTimeMillis(); private long duration = -1; public void stop() { duration = System.currentTimeMillis() - start; } public String getDuration() { if (duration >= 0) { if (duration < 1000) { return String.format("%4s ms", duration); } else if (duration < 100000) { return String.format("%5.1f s", duration / 1000.0); } else { return String.format("%5.0f s", duration / 1000.0); } } else { return "unknown"; } } public boolean hasFailed() { return exception != null; } public void logSummary(String name) { if (hasFailed()) { getLog().error("- FAILED: [" + getDuration() + "] " + name + ": " + exception.getMessage()); } else { getLog().info("- SUCCESS: [" + getDuration() + "] " + name); } } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy