com.isomorphic.maven.mojo.AbstractPackagerMojo Maven / Gradle / Ivy
Show all versions of isc-maven-plugin Show documentation
package com.isomorphic.maven.mojo;
/*
* 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.
*/
import com.google.common.base.Splitter;
import com.isomorphic.maven.packaging.Module;
import com.isomorphic.maven.packaging.*;
import com.isomorphic.maven.util.HttpRequestManager;
import org.apache.commons.io.FileUtils;
import org.apache.commons.io.FilenameUtils;
import org.apache.commons.io.IOCase;
import org.apache.commons.io.filefilter.FileFilterUtils;
import org.apache.commons.io.filefilter.IOFileFilter;
import org.apache.commons.io.filefilter.TrueFileFilter;
import org.apache.commons.io.filefilter.WildcardFileFilter;
import org.apache.commons.lang3.StringUtils;
import org.apache.http.HttpHost;
import org.apache.http.auth.UsernamePasswordCredentials;
import org.apache.maven.model.Model;
import org.apache.maven.model.Parent;
import org.apache.maven.model.building.DefaultModelBuildingRequest;
import org.apache.maven.model.building.ModelBuildingException;
import org.apache.maven.model.building.ModelBuildingRequest;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException;
import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.project.ProjectBuildingRequest.RepositoryMerging;
import org.apache.maven.project.ProjectModelResolver;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.util.FileSystemUtils;
import java.io.File;
import java.io.IOException;
import java.net.URI;
import java.nio.file.*;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.*;
import static com.isomorphic.maven.packaging.License.*;
/**
* A base class meant to deal with prerequisites to install / deploy goals,
* which are basically to resolve the files in a given distribution to a
* collection of Maven artifacts suitable for installation or deployment to some
* Maven repository.
*
* The resulting artifacts are provided to this object's {@link #doExecute(Set)}
* method.
*/
public abstract class AbstractPackagerMojo extends AbstractBaseMojo {
private static final Logger LOGGER = LoggerFactory.getLogger(AbstractPackagerMojo.class);
private static final HttpHost HOST = new HttpHost("www.smartclient.com", -1, "https");
private HttpRequestManager httpWorker;
/**
* If true, the optional analytics module (bundled and distributed
* separately) has been licensed and should be downloaded with the
* distribution specified by {@link #license}.
*
* @since 1.0.0
*/
@Parameter(property = "includeAnalytics", defaultValue = "false")
protected Boolean includeAnalytics;
/**
* The date on which the Isomorphic build was made publicly available at http://www.smartclient.com/builds
* /, in yyyy-MM-dd format. e.g., 2013-25-12. Used to determine both
* remote and local file locations.
*
* Note that if no value is provided, and {@link #skipDownload} is not true, an attempt is
* made to discover the date of the latest distribution currently published to the
* Isomorphic build server.
*
* Default value is: The date of the most recent distribution (with caveats)
.
*
* @since 1.0.0
*/
@Parameter(property = "buildDate")
protected String buildDate;
/**
* The Isomorphic version number of the specified {@link #product}. e.g.,
* 9.1d, 4.0p. Used to determine both remote and local file locations.
*
* @since 1.0.0
*/
@Parameter(property = "buildNumber", required = true)
protected String buildNumber;
/**
* Typically one of: LGPL, EVAL, PRO, POWER, ENTERPRISE. Although it is also
* valid to specify optional modules ANALYTICS_MODULE or MESSAGING_MODULE,
* generally prefer the {@link #includeAnalytics} /
* {@link #includeMessaging} properties, respectively, to cause the optional
* modules to be included with the base installation / deployment.
*
* @since 1.0.0
*/
@Parameter(property = "license", required = true)
protected License license;
/**
* Limits the skins installed with the runtime to the names in this comma-separated list.
* E.g., -Dskins=Tahoe,Stratus
will remove all skins except Tahoe and Stratus.
* Note that these deleted resources cannot be recovered except by reinstalling the artifact(s) to
* your repository.
*
* @since 1.4.6
*/
@Parameter(property = "skins")
protected String skins;
/**
* If true, the optional messaging module (bundled and distributed
* separately) has been licensed and should be downloaded with the
* distribution specified by {@link #license}.
*
* @since 1.0.0
*/
@Parameter(property = "includeMessaging", defaultValue = "false")
protected Boolean includeMessaging;
/**
* If true, any file previously downloaded / unpacked will be overwritten
* with this execution. Useful in the case of an interrupted download. Note
* that this setting has no effect on unzip operations.
*
* @since 1.0.0
*/
@Parameter(property = "overwrite", defaultValue = "false")
protected Boolean overwrite;
/**
* If true, makes a copy of the given distribution in a 'latest' subdirectory.
* Can be useful for bookmarking documentation, etc. but adds additional install time
* and storage requirements.
*
* @since 1.4.0
*/
@Parameter(property = "copyToLatestFolder", defaultValue = "false")
protected Boolean copyToLatestFolder;
/**
* If true, no attempt is made to download any remote distribution. Files
* will be loaded instead from a path constructed of the following parts
* (e.g., C:/downloads/SmartGWT/PowerEdition/4.1d/2013-12-25/zip):
*
*
* - {@link #workdir}
* - {@link #product}
* - {@link #license}
* - {@link #buildNumber}
* - {@link #buildDate}
* - "zip"
*
*
* @since 1.0.0
*/
@Parameter(property = "skipDownload", defaultValue = "false")
protected Boolean skipDownload;
/**
* If true, no attempt it made to extract the contents of any distribution.
* Only useful in the case where some manual intervention is required
* between download and another step. For example, it would be possible to
* first run the download goal, manipulate the version number of some
* dependency in some POM, and then run the install goal with
* skipExtraction=false to prevent the modified POM from being overwritten.
*
* This is the kind of thing that should generally be avoided, however.
*/
@Parameter(property = "skipExtraction", defaultValue = "false")
protected Boolean skipExtraction;
/**
* One of SMARTGWT, SMARTCLIENT.
*
* @since 1.0.0
*/
@Parameter(property = "product", defaultValue = "SMARTGWT")
protected Product product;
/**
* If true, artifacts should be versioned with the 'SNAPSHOT' qualifier, in the case of development
* builds only. The setting has no effect on patch builds.
*
* If false, each artifact's POM file is modified to remove the unwanted
* qualifier. This can be useful if you need to deploy a development build
* to a production environment.
*
* @since 1.0.0
*/
@Parameter(property = "snapshots", defaultValue = "true")
protected Boolean snapshots;
/**
* The path to some directory that is to be used for storing downloaded
* files, working copies, and so on.
*
* @since 1.0.0
*/
@Parameter(property = "workdir", defaultValue = "${java.io.tmpdir}/${project.artifactId}")
protected File workdir;
/**
* The id of a server
* configuration containing authentication credentials for the
* smartclient.com website, used to download licensed products.
*
* Not strictly necessary for unprotected (LGPL) distributions.
*
* @since 1.0.0
*/
@Parameter(property = "serverId", defaultValue = "smartclient-developer")
protected String serverId;
/**
* The point where a subclass is able to manipulate the collection of
* artifacts prepared for it by this object's {@link #execute()} method.
*
* @param artifacts
* A collection of Maven artifacts resulting from the download
* and preparation of a supported Isomorphic SDK.
* @throws MojoExecutionException
* When any fatal error occurs. e.g., there is no distribution
* to work with.
* @throws MojoFailureException
* When any non-fatal error occurs. e.g., documentation cannot
* be copied to some other folder.
*/
public abstract void doExecute(Set artifacts) throws MojoExecutionException,
MojoFailureException;
/**
* Provides some initialization and validation steps around the collection
* and transformation of an Isomorphic SDK.
*
* @throws MojoExecutionException
* When any fatal error occurs.
* @throws MojoFailureException
* When any non-fatal error occurs.
*/
public void execute() throws MojoExecutionException, MojoFailureException {
// allow execution to proceed without login credentials - it may be that
// they're not required
UsernamePasswordCredentials credentials = getCredentials(serverId);
if (credentials == null) {
LOGGER.warn("No server configured with id '{}'. Will be unable to authenticate.", serverId);
}
String buildNumberFormat = "\\d.*\\.\\d.*[d|p]";
if (!buildNumber.matches(buildNumberFormat)) {
throw new MojoExecutionException(String.format(
"buildNumber '%s' must take the form [major].[minor].[d|p]. e.g., 4.1d",
buildNumber, buildNumberFormat));
}
httpWorker = new HttpRequestManager(HOST, credentials, settings.getActiveProxy());
Downloads dl = new Downloads(httpWorker);
try {
if (! skipDownload) {
httpWorker.login();
}
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd");
dateFormat.setLenient(false);
if (buildDate == null) {
if (skipDownload) {
throw new MojoExecutionException("A buildDate value is required when the skipDownload parameter is true.");
}
Distribution d = Distribution.get(product, license);
LOGGER.info("No buildDate provided. Contacting Isomorphic build server to "
+ "look for the most recent distribution...");
String link = dl.findCurrentBuild(d, buildNumber);
if (link == null) {
throw new MojoExecutionException("No build found for the given distribution.");
}
LOGGER.debug("Extracting date from server response: '{}'", link);
buildDate = StringUtils.substringAfterLast(link, "/");
LOGGER.info("buildDate set to '{}'", buildDate);
}
dateFormat.parse(buildDate);
File basedir = FileUtils.getFile(workdir, product.toString(), license.toString(),
buildNumber, buildDate);
// add optional modules to the list of downloads
List licenses = new ArrayList();
licenses.add(license);
if (license == POWER || license == ENTERPRISE) {
if (includeAnalytics) {
licenses.add(ANALYTICS_MODULE);
}
if (includeMessaging) {
licenses.add(MESSAGING_MODULE);
}
}
// collect the maven artifacts and send them along to the abstract method
Set artifacts = collect(licenses, basedir);
String[] executables = { "bat", "sh", "command" };
Collection scripts = FileUtils.listFiles(basedir, executables, true);
if (copyToLatestFolder) {
File bookmarkable = new File(basedir.getParent(), "latest");
LOGGER.info("Copying distribution to '{}'", bookmarkable.getAbsolutePath());
try {
FileUtils.forceMkdir(bookmarkable);
FileUtils.cleanDirectory(bookmarkable);
FileUtils.copyDirectory(basedir, bookmarkable,
FileFilterUtils.notFileFilter(FileFilterUtils.nameFileFilter("zip")));
scripts.addAll(FileUtils.listFiles(bookmarkable, executables, true));
} catch (IOException e) {
throw new MojoFailureException("Unable to copy distribution contents", e);
}
}
for (File script : scripts) {
script.setExecutable(true);
LOGGER.debug("Enabled execute permissions on file '{}'", script.getAbsolutePath());
}
doExecute(artifacts);
} catch (ParseException e) {
throw new MojoExecutionException(String.format(
"buildDate '%s' must take the form yyyy-MM-dd.", buildDate));
} finally {
httpWorker.logout();
}
}
/**
* Download the specified distributions, if necessary, extract resources
* from them, and use the results to create Maven artifacts as appropriate:
*
* Try to install all of the main artifacts - e.g., those found in lib/*.jar
* and assembly/*.zip
*
* Try to match main artifacts to 'subartifacts' by name and attach them
* (with classifiers as necessary)
*
* @param downloads
* The list of licenses top be included in the distribution.
* Multiple licenses are only allowed to support the inclusion of
* optional modules.
* @param basedir
* The directory into which results should be written
* @return A collection of Maven artifacts resulting from the download and
* preparation of a supported Isomorphic SDK.
* @throws MojoExecutionException
* When any fatal error occurs.
*/
private Set collect(List downloads, File basedir)
throws MojoExecutionException {
File downloadTo = new File(basedir, "zip");
downloadTo.mkdirs();
Downloads downloadManager = new Downloads(httpWorker);
downloadManager.setToFolder(downloadTo);
downloadManager.setOverwriteExistingFiles(overwrite);
File[] existing = downloadTo.listFiles();
List distributions = new ArrayList();
try {
if (!skipDownload) {
distributions.addAll(downloadManager.fetch(product, buildNumber, buildDate,
downloads.toArray(new License[0])));
} else if (existing != null) {
LOGGER.info("Creating local distribution from '{}'",
downloadTo.getAbsolutePath());
Distribution distribution = Distribution.get(product, license);
distribution.getFiles().addAll(Arrays.asList(existing));
distributions.add(distribution);
}
if (!skipExtraction) {
LOGGER.info("Unpacking downloaded file/s to '{}'", basedir);
for (Distribution distribution : distributions) {
distribution.unpack(basedir);
if (skins != null) {
LOGGER.info("Pruning unwanted skins...");
skin(basedir, distribution.getSkinResources());
}
}
}
// TODO it'd be better if this didn't have to know where the files were located after unpacking
// it doesn't strictly read this way, but we're looking for
// lib/*.jar, pom/*.xml, assembly/*.zip
Collection files = FileUtils.listFiles(
basedir,
FileFilterUtils.or(FileFilterUtils.suffixFileFilter("jar"),
FileFilterUtils.suffixFileFilter("xml"),
FileFilterUtils.suffixFileFilter("zip")),
FileFilterUtils.or(FileFilterUtils.nameFileFilter("lib"),
FileFilterUtils.nameFileFilter("pom"),
FileFilterUtils.nameFileFilter("assembly"))
);
if (files.isEmpty()) {
throw new MojoExecutionException(String.format("There don't appear to be any files to work with at '%s'. Check earlier log entries for clues.", basedir.getAbsolutePath()));
}
Set result = new TreeSet();
for (File file : files) {
try {
String base = FilenameUtils.getBaseName(file.getName().replaceAll("_", "-"));
// poms don't need anything else
if ("xml".equals(FilenameUtils.getExtension(file.getName()))) {
result.add(new Module(getModelFromFile(file)));
continue;
}
// for each jar/zip, find the matching pom
IOFileFilter filter = new WildcardFileFilter(base + ".pom", IOCase.INSENSITIVE);
Collection poms = FileUtils.listFiles(basedir, filter,
TrueFileFilter.INSTANCE);
if (poms.size() != 1) {
LOGGER.warn("Expected to find exactly 1 POM matching artifact with name '{}', but found {}. Skpping installation.",base, poms.size());
continue;
}
Model model = getModelFromFile(poms.iterator().next());
Module module = new Module(model, file);
/*
* Find the right javadoc bundle, matched on prefix. e.g.,
* smartgwt-eval -> smartgwt-javadoc isomorphic-core-rpc ->
* isomorphic-javadoc and add it to the main artifact with
* the javadoc classifier. This seems appropriate as long as
* a) there is no per-jar javadoc b) naming conventions are
* adhered to (or can be corrected by plugin at extraction)
*/
int index = base.indexOf("-");
String prefix = base.substring(0, index);
Collection doc = FileUtils.listFiles(new File(basedir, "doc"),
FileFilterUtils.prefixFileFilter(prefix),
FileFilterUtils.nameFileFilter("lib"));
if (doc.size() != 1) {
LOGGER.debug("Found {} javadoc attachments with prefix '{}'. Skipping attachment.", doc.size(), prefix);
} else {
module.attach(doc.iterator().next(), "javadoc");
}
result.add(module);
} catch (ModelBuildingException e) {
throw new MojoExecutionException("Error building model from POM", e);
}
}
return result;
} catch (IOException e) {
throw new MojoExecutionException("Failure during assembly collection", e);
}
}
/**f
* Read the given POM so it can be used as the source of coordinates, etc.
* during artifact construction. Note that if this object's
* {@link #snapshots} property is true, and we're working with a development
* build ({@link #buildNumber} ends with 'd'), the POM is modified to remove
* the SNAPSHOT qualifier.
*
* @param pom
* the POM file containing the artifact metadata
* @return A Maven model to be used at
* {@link com.isomorphic.maven.packaging.Module#Module(Model)}
* Module construction
* @throws ModelBuildingException
* if the Model cannot be built from the given POM
* @throws IOException
* if the Model cannot be built from the given POM
*/
private Model getModelFromFile(File pom) throws ModelBuildingException, IOException {
if (buildNumber.endsWith("d") && !snapshots) {
LOGGER.info(
"Rewriting file to remove SNAPSHOT qualifier from development POM '{}'",
pom.getName());
String content = FileUtils.readFileToString(pom);
content = content.replaceAll("-SNAPSHOT", "");
FileUtils.write(pom, content);
}
ProjectModelResolver resolver = new ProjectModelResolver(repositorySystemSession, null, repositorySystem, remoteRepositoryManager, project.getRemoteProjectRepositories(), RepositoryMerging.POM_DOMINANT, null);
ModelBuildingRequest request = new DefaultModelBuildingRequest();
request.setModelResolver(resolver);
request.setPomFile(pom);
Model model = modelBuilder.buildRawModel(pom, 0, false).get();
Parent parent = model.getParent();
if (parent != null) {
model.setGroupId(parent.getGroupId());
model.setVersion(parent.getVersion());
}
return model;
}
private void skin(File basedir, Map skinResources) throws MojoExecutionException, IOException {
Map props = new HashMap<>();
props.put("create", "false");
List requested = new ArrayList<>(Splitter.on(",").trimResults().splitToList(skins.toLowerCase()));
// preserve for simplicity
requested.add("enterprise");
requested.add("fonts");
requested.add("toolskin");
requested.add("toolskinnative");
Set keys = skinResources.keySet();
for(String key : keys) {
String archive = basedir.getCanonicalPath() + "/" + key;
String skinDir = skinResources.get(key);
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("Opening file at '{}' for modification", archive);
}
URI uri = URI.create("jar:file:" + archive);
try (FileSystem fs = FileSystems.newFileSystem(uri, props)) {
try (DirectoryStream ds = Files.newDirectoryStream(fs.getPath(skinDir), Files::isDirectory)) {
for (Path path : ds) {
String skin = path.getFileName().toString().toLowerCase().replace("/", "");
if (requested.contains(skin)) {
LOGGER.debug("Preserving '{}' skin", skin);
continue;
}
LOGGER.info("Deleting '{}' skin resources at '{}'", skin, path);
FileSystemUtils.deleteRecursively(path);
}
} catch (NotDirectoryException ignore) {}
}
}
}
}