blob: a77d8cd8b63b6d13f2b6b8642e727f78b83fb3c8 [file] [log] [blame]
/**
*
* 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.karaf.tooling.utils;
import java.io.File;
import java.io.InputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.nio.file.Files;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import org.apache.karaf.util.StreamUtils;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.factory.ArtifactFactory;
import org.apache.maven.artifact.metadata.ArtifactMetadataSource;
import org.apache.maven.artifact.repository.ArtifactRepository;
import org.apache.maven.artifact.repository.DefaultArtifactRepository;
import org.apache.maven.artifact.repository.layout.DefaultRepositoryLayout;
import org.apache.maven.artifact.resolver.ArtifactResolver;
import org.apache.maven.artifact.versioning.InvalidVersionSpecificationException;
import org.apache.maven.artifact.versioning.VersionRange;
import org.apache.maven.execution.MavenSession;
import org.apache.maven.model.Dependency;
import org.apache.maven.model.DependencyManagement;
import org.apache.maven.plugin.AbstractMojo;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugins.annotations.Component;
import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.project.MavenProject;
import org.apache.maven.project.MavenProjectBuilder;
import org.apache.maven.project.MavenProjectHelper;
import org.apache.maven.project.ProjectBuildingException;
import org.apache.maven.settings.Proxy;
import org.codehaus.plexus.PlexusContainer;
@SuppressWarnings({"deprecation", "rawtypes", "unchecked"})
public abstract class MojoSupport extends AbstractMojo {
/**
* Maven ProjectHelper
*/
@Component
protected MavenProjectHelper projectHelper;
/**
* The Maven project.
*/
@Parameter(defaultValue = "${project}", readonly = true)
protected MavenProject project;
/**
* Directory that resources are copied to during the build.
*/
@Parameter(defaultValue = "${project.build.directory}/${project.artifactId}-${project.version}-installer")
protected File workDirectory;
@Component
protected MavenProjectBuilder projectBuilder;
@Parameter(defaultValue = "${localRepository}")
protected ArtifactRepository localRepo;
@Parameter(defaultValue = "${project.remoteArtifactRepositories}")
protected List<ArtifactRepository> remoteRepos;
@Component
protected ArtifactMetadataSource artifactMetadataSource;
@Component
protected ArtifactResolver artifactResolver;
@Component
protected ArtifactFactory factory;
/**
* The artifact type of a feature.
*/
@Parameter(defaultValue = "xml")
private String featureArtifactType = "xml";
/**
* The Maven session.
*/
@Parameter(defaultValue = "${session}", readonly = true)
protected MavenSession mavenSession;
/**
* <p>We can't autowire strongly typed RepositorySystem from Aether because it may be Sonatype (Maven 3.0.x)
* or Eclipse (Maven 3.1.x/3.2.x) version, so we switch to service locator by autowiring entire {@link PlexusContainer}</p>
*
* <p>It's a bit of a hack but we have not choice when we want to be usable both in Maven 3.0.x and 3.1.x/3.2.x</p>
*/
@Component
protected PlexusContainer container;
protected MavenProject getProject() {
return project;
}
protected File getWorkDirectory() {
return workDirectory;
}
public MavenProjectHelper getProjectHelper() {
return projectHelper;
}
// called by Plexus when injecting the mojo's session
public void setMavenSession(MavenSession mavenSession) {
this.mavenSession = mavenSession;
if (mavenSession != null) {
// check for custom settings.xml and pass it onto pax-url-aether
File settingsFile = mavenSession.getRequest().getUserSettingsFile();
if (settingsFile != null && settingsFile.isFile()) {
System.setProperty("org.ops4j.pax.url.mvn.settings", settingsFile.getPath());
}
}
}
protected Map createManagedVersionMap(String projectId,
DependencyManagement dependencyManagement) throws ProjectBuildingException {
Map map;
if (dependencyManagement != null
&& dependencyManagement.getDependencies() != null) {
map = new HashMap();
for (Iterator i = dependencyManagement.getDependencies().iterator(); i
.hasNext();) {
Dependency d = (Dependency) i.next();
try {
VersionRange versionRange = VersionRange
.createFromVersionSpec(d.getVersion());
Artifact artifact = factory.createDependencyArtifact(d
.getGroupId(), d.getArtifactId(), versionRange, d
.getType(), d.getClassifier(), d.getScope());
map.put(d.getManagementKey(), artifact);
} catch (InvalidVersionSpecificationException e) {
throw new ProjectBuildingException(projectId,
"Unable to parse version '" + d.getVersion()
+ "' for dependency '"
+ d.getManagementKey() + "': "
+ e.getMessage(), e);
}
}
} else {
map = Collections.EMPTY_MAP;
}
return map;
}
protected String translateFromMaven(String uri) {
if (uri.startsWith("mvn:")) {
String[] parts = uri.substring("mvn:".length()).split("/");
String groupId = parts[0];
String artifactId = parts[1];
String version = null;
String classifier = null;
String type = "jar";
if (parts.length > 2) {
version = parts[2];
if (parts.length > 3) {
type = parts[3];
if (parts.length > 4) {
classifier = parts[4];
}
}
}
String dir = groupId.replace('.', '/') + "/" + artifactId + "/" + version + "/";
String name = artifactId + "-" + version + (classifier != null ? "-" + classifier : "") + "." + type;
return getLocalRepoUrl() + "/" + dir + name;
}
if (System.getProperty("os.name").startsWith("Windows") && uri.startsWith("file:")) {
String baseDir = uri.substring(5).replace('\\', '/').replaceAll(" ", "%20");
String result = baseDir;
if (baseDir.indexOf(":") > 0) {
result = "file:///" + baseDir;
}
return result;
}
return uri;
}
protected String getLocalRepoUrl() {
if (System.getProperty("os.name").startsWith("Windows")) {
String baseDir = localRepo.getBasedir().replace('\\', '/').replaceAll(" ", "%20");
return extractProtocolFromLocalMavenRepo() + ":///" + baseDir;
} else {
return localRepo.getUrl();
}
}
/**
* Required because Maven 3 returns null in {@link ArtifactRepository#getProtocol()} (see KARAF-244)
*/
private String extractProtocolFromLocalMavenRepo() {
try {
return new URL(localRepo.getUrl()).getProtocol();
} catch (MalformedURLException e) {
// Basically this should not happen; if it does though cancel the process
throw new RuntimeException("Repository URL is not valid", e);
}
}
private Dependency findDependency(List<Dependency> dependencies, String artifactId, String groupId) {
for(Dependency dep : dependencies) {
if (artifactId.equals(dep.getArtifactId()) && groupId.equals(dep.getGroupId()) &&
featureArtifactType.equals(dep.getType())) {
if (dep.getVersion() != null)
return dep;
}
}
return null;
}
/**
* Convert a feature resourceLocation (bundle or configuration file) into an artifact.
*
* @param resourceLocation the feature resource location (bundle or configuration file).
* @param skipNonMavenProtocols flag to skip protocol different than mvn:
* @return the artifact corresponding to the resource.
* @throws MojoExecutionException
*/
protected Artifact resourceToArtifact(String resourceLocation, boolean skipNonMavenProtocols) throws MojoExecutionException {
resourceLocation = resourceLocation.replace("\r\n", "").replace("\n", "").replace(" ", "").replace("\t", "");
final int index = resourceLocation.indexOf("mvn:");
if (index < 0) {
if (skipNonMavenProtocols) {
return null;
}
throw new MojoExecutionException("Resource URL is not a Maven URL: " + resourceLocation);
} else {
resourceLocation = resourceLocation.substring(index + "mvn:".length());
}
// Truncate the URL when a '#', a '?' or a '$' is encountered
final int index1 = resourceLocation.indexOf('?');
final int index2 = resourceLocation.indexOf('#');
int endIndex = -1;
if (index1 > 0) {
if (index2 > 0) {
endIndex = Math.min(index1, index2);
} else {
endIndex = index1;
}
} else if (index2 > 0) {
endIndex = index2;
}
if (endIndex >= 0) {
resourceLocation = resourceLocation.substring(0, endIndex);
}
final int index3 = resourceLocation.indexOf('$');
if (index3 > 0) {
resourceLocation = resourceLocation.substring(0, index3);
}
//check if the resourceLocation descriptor contains also remote repository information.
ArtifactRepository repo = null;
if (resourceLocation.startsWith("http://")) {
final int repoDelimIntex = resourceLocation.indexOf('!');
String repoUrl = resourceLocation.substring(0, repoDelimIntex);
repo = new DefaultArtifactRepository(
repoUrl,
repoUrl,
new DefaultRepositoryLayout());
org.apache.maven.repository.Proxy mavenProxy = configureProxyToInlineRepo();
if (mavenProxy != null) {
repo.setProxy(mavenProxy);
}
resourceLocation = resourceLocation.substring(repoDelimIntex + 1);
}
String[] parts = resourceLocation.split("/");
String groupId = parts[0];
String artifactId = parts[1];
String version = null;
String classifier = null;
String type = "jar";
if (parts.length > 2) {
version = parts[2];
if (parts.length > 3) {
type = parts[3];
if (parts.length > 4) {
classifier = parts[4];
}
}
} else {
Dependency dep = findDependency(project.getDependencies(), artifactId, groupId);
if (dep == null && project.getDependencyManagement() != null) {
dep = findDependency(project.getDependencyManagement().getDependencies(), artifactId, groupId);
}
if (dep != null) {
version = dep.getVersion();
classifier = dep.getClassifier();
type = dep.getType();
}
}
if (version == null || version.isEmpty()) {
throw new MojoExecutionException("Cannot find version for: " + resourceLocation);
}
Artifact artifact = factory.createArtifactWithClassifier(groupId, artifactId, version, type, classifier);
artifact.setRepository(repo);
return artifact;
}
private org.apache.maven.repository.Proxy configureProxyToInlineRepo() {
if (mavenSession != null && mavenSession.getSettings() != null) {
Proxy proxy = mavenSession.getSettings().getActiveProxy();
org.apache.maven.repository.Proxy mavenProxy = new org.apache.maven.repository.Proxy();
if (proxy != null) {
mavenProxy.setProtocol(proxy.getProtocol());
mavenProxy.setHost(proxy.getHost());
mavenProxy.setPort(proxy.getPort());
mavenProxy.setNonProxyHosts(proxy.getNonProxyHosts());
mavenProxy.setUserName(proxy.getUsername());
mavenProxy.setPassword(proxy.getPassword());
return mavenProxy;
} else {
return null;
}
} else {
return null;
}
}
protected void copy(File sourceFile, File destFile) {
File targetDir = destFile.getParentFile();
ensureDirExists(targetDir);
try (InputStream is = Files.newInputStream(sourceFile.toPath())) {
try (OutputStream bos = Files.newOutputStream(destFile.toPath())) {
StreamUtils.copy(is, bos);
}
} catch (IOException e) {
throw new RuntimeException(e.getMessage(), e);
}
}
/**
* Make sure the target directory exists and
* that is actually a directory
* @param targetDir
* @throws IOException
*/
private static void ensureDirExists(File targetDir) {
if (!targetDir.exists()) {
if (!targetDir.mkdirs()) {
throw new RuntimeException("Unable to create target directory: " + targetDir);
}
} else if (!targetDir.isDirectory()) {
throw new RuntimeException("Target is not a directory: " + targetDir);
}
}
}