1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19 package org.apache.maven.cli.internal;
20
21 import javax.inject.Inject;
22 import javax.inject.Named;
23
24 import java.io.File;
25 import java.util.ArrayList;
26 import java.util.Collections;
27 import java.util.List;
28 import java.util.NoSuchElementException;
29 import java.util.Set;
30 import java.util.stream.Collectors;
31
32 import org.apache.maven.RepositoryUtils;
33 import org.apache.maven.api.Service;
34 import org.apache.maven.api.Session;
35 import org.apache.maven.api.model.Plugin;
36 import org.apache.maven.api.services.ArtifactCoordinateFactory;
37 import org.apache.maven.api.services.ArtifactManager;
38 import org.apache.maven.api.services.ArtifactResolver;
39 import org.apache.maven.api.services.RepositoryFactory;
40 import org.apache.maven.api.services.VersionParser;
41 import org.apache.maven.api.services.VersionRangeResolver;
42 import org.apache.maven.cli.internal.extension.model.CoreExtension;
43 import org.apache.maven.execution.DefaultMavenExecutionResult;
44 import org.apache.maven.execution.MavenExecutionRequest;
45 import org.apache.maven.execution.MavenSession;
46 import org.apache.maven.extension.internal.CoreExports;
47 import org.apache.maven.extension.internal.CoreExtensionEntry;
48 import org.apache.maven.internal.impl.DefaultArtifactCoordinateFactory;
49 import org.apache.maven.internal.impl.DefaultArtifactManager;
50 import org.apache.maven.internal.impl.DefaultArtifactResolver;
51 import org.apache.maven.internal.impl.DefaultModelVersionParser;
52 import org.apache.maven.internal.impl.DefaultRepositoryFactory;
53 import org.apache.maven.internal.impl.DefaultSession;
54 import org.apache.maven.internal.impl.DefaultVersionParser;
55 import org.apache.maven.internal.impl.DefaultVersionRangeResolver;
56 import org.apache.maven.internal.impl.InternalSession;
57 import org.apache.maven.plugin.PluginResolutionException;
58 import org.apache.maven.plugin.internal.DefaultPluginDependenciesResolver;
59 import org.apache.maven.resolver.MavenChainedWorkspaceReader;
60 import org.apache.maven.resolver.RepositorySystemSessionFactory;
61 import org.codehaus.plexus.DefaultPlexusContainer;
62 import org.codehaus.plexus.PlexusContainer;
63 import org.codehaus.plexus.classworlds.ClassWorld;
64 import org.codehaus.plexus.classworlds.realm.ClassRealm;
65 import org.codehaus.plexus.interpolation.InterpolationException;
66 import org.codehaus.plexus.interpolation.Interpolator;
67 import org.codehaus.plexus.interpolation.MapBasedValueSource;
68 import org.codehaus.plexus.interpolation.StringSearchInterpolator;
69 import org.eclipse.aether.RepositorySystem;
70 import org.eclipse.aether.RepositorySystemSession;
71 import org.eclipse.aether.RepositorySystemSession.CloseableSession;
72 import org.eclipse.aether.artifact.Artifact;
73 import org.eclipse.aether.graph.DependencyFilter;
74 import org.eclipse.aether.internal.impl.DefaultChecksumPolicyProvider;
75 import org.eclipse.aether.internal.impl.DefaultRemoteRepositoryManager;
76 import org.eclipse.aether.internal.impl.DefaultUpdatePolicyAnalyzer;
77 import org.eclipse.aether.repository.RemoteRepository;
78 import org.eclipse.aether.repository.WorkspaceReader;
79 import org.eclipse.aether.resolution.ArtifactResult;
80 import org.eclipse.aether.resolution.DependencyResult;
81 import org.eclipse.aether.util.filter.ExclusionsDependencyFilter;
82 import org.eclipse.aether.util.version.GenericVersionScheme;
83 import org.eclipse.sisu.Nullable;
84 import org.slf4j.Logger;
85 import org.slf4j.LoggerFactory;
86
87
88
89
90 @Named
91 public class BootstrapCoreExtensionManager {
92 public static final String STRATEGY_PARENT_FIRST = "parent-first";
93 public static final String STRATEGY_PLUGIN = "plugin";
94 public static final String STRATEGY_SELF_FIRST = "self-first";
95
96 private final Logger log = LoggerFactory.getLogger(getClass());
97
98 private final DefaultPluginDependenciesResolver pluginDependenciesResolver;
99
100 private final RepositorySystemSessionFactory repositorySystemSessionFactory;
101
102 private final CoreExports coreExports;
103
104 private final ClassWorld classWorld;
105
106 private final ClassRealm parentRealm;
107
108 private final WorkspaceReader ideWorkspaceReader;
109
110 private final RepositorySystem repoSystem;
111
112 @Inject
113 public BootstrapCoreExtensionManager(
114 DefaultPluginDependenciesResolver pluginDependenciesResolver,
115 RepositorySystemSessionFactory repositorySystemSessionFactory,
116 CoreExports coreExports,
117 PlexusContainer container,
118 @Nullable @Named("ide") WorkspaceReader ideWorkspaceReader,
119 RepositorySystem repoSystem) {
120 this.pluginDependenciesResolver = pluginDependenciesResolver;
121 this.repositorySystemSessionFactory = repositorySystemSessionFactory;
122 this.coreExports = coreExports;
123 this.classWorld = ((DefaultPlexusContainer) container).getClassWorld();
124 this.parentRealm = container.getContainerRealm();
125 this.ideWorkspaceReader = ideWorkspaceReader;
126 this.repoSystem = repoSystem;
127 }
128
129 public List<CoreExtensionEntry> loadCoreExtensions(
130 MavenExecutionRequest request, Set<String> providedArtifacts, List<CoreExtension> extensions)
131 throws Exception {
132 try (CloseableSession repoSession = repositorySystemSessionFactory
133 .newRepositorySessionBuilder(request)
134 .setWorkspaceReader(new MavenChainedWorkspaceReader(request.getWorkspaceReader(), ideWorkspaceReader))
135 .build()) {
136 MavenSession mSession = new MavenSession(repoSession, request, new DefaultMavenExecutionResult());
137 InternalSession iSession = new SimpleSession(mSession, repoSystem, null);
138 InternalSession.associate(repoSession, iSession);
139
140 List<RemoteRepository> repositories = RepositoryUtils.toRepos(request.getPluginArtifactRepositories());
141 Interpolator interpolator = createInterpolator(request);
142
143 return resolveCoreExtensions(repoSession, repositories, providedArtifacts, extensions, interpolator);
144 }
145 }
146
147 private List<CoreExtensionEntry> resolveCoreExtensions(
148 RepositorySystemSession repoSession,
149 List<RemoteRepository> repositories,
150 Set<String> providedArtifacts,
151 List<CoreExtension> configuration,
152 Interpolator interpolator)
153 throws Exception {
154 List<CoreExtensionEntry> extensions = new ArrayList<>();
155
156 DependencyFilter dependencyFilter = new ExclusionsDependencyFilter(providedArtifacts);
157
158 for (CoreExtension extension : configuration) {
159 List<Artifact> artifacts =
160 resolveExtension(extension, repoSession, repositories, dependencyFilter, interpolator);
161 if (!artifacts.isEmpty()) {
162 extensions.add(createExtension(extension, artifacts));
163 }
164 }
165
166 return Collections.unmodifiableList(extensions);
167 }
168
169 private CoreExtensionEntry createExtension(CoreExtension extension, List<Artifact> artifacts) throws Exception {
170 String realmId = "coreExtension>" + extension.getGroupId() + ":" + extension.getArtifactId() + ":"
171 + extension.getVersion();
172 final ClassRealm realm = classWorld.newRealm(realmId, null);
173 Set<String> providedArtifacts = Collections.emptySet();
174 String classLoadingStrategy = extension.getClassLoadingStrategy();
175 if (STRATEGY_PARENT_FIRST.equals(classLoadingStrategy)) {
176 realm.importFrom(parentRealm, "");
177 } else if (STRATEGY_PLUGIN.equals(classLoadingStrategy)) {
178 coreExports.getExportedPackages().forEach((p, cl) -> realm.importFrom(cl, p));
179 providedArtifacts = coreExports.getExportedArtifacts();
180 } else if (STRATEGY_SELF_FIRST.equals(classLoadingStrategy)) {
181 realm.setParentRealm(parentRealm);
182 } else {
183 throw new IllegalArgumentException("Unsupported class-loading strategy '"
184 + classLoadingStrategy + "'. Supported values are: " + STRATEGY_PARENT_FIRST
185 + ", " + STRATEGY_PLUGIN + " and " + STRATEGY_SELF_FIRST);
186 }
187 log.debug("Populating class realm {}", realm.getId());
188 for (Artifact artifact : artifacts) {
189 String id = artifact.getGroupId() + ":" + artifact.getArtifactId();
190 if (providedArtifacts.contains(id)) {
191 log.debug(" Excluded {}", id);
192 } else {
193 File file = artifact.getFile();
194 log.debug(" Included {} located at {}", id, file);
195 realm.addURL(file.toURI().toURL());
196 }
197 }
198 return CoreExtensionEntry.discoverFrom(
199 realm,
200 Collections.singleton(artifacts.get(0).getFile()),
201 extension.getGroupId() + ":" + extension.getArtifactId(),
202 extension.getConfiguration());
203 }
204
205 private List<Artifact> resolveExtension(
206 CoreExtension extension,
207 RepositorySystemSession repoSession,
208 List<RemoteRepository> repositories,
209 DependencyFilter dependencyFilter,
210 Interpolator interpolator)
211 throws ExtensionResolutionException {
212 try {
213
214
215
216
217 Plugin plugin = Plugin.newBuilder()
218 .groupId(interpolator.interpolate(extension.getGroupId()))
219 .artifactId(interpolator.interpolate(extension.getArtifactId()))
220 .version(interpolator.interpolate(extension.getVersion()))
221 .build();
222
223 DependencyResult result = pluginDependenciesResolver.resolveCoreExtension(
224 new org.apache.maven.model.Plugin(plugin), dependencyFilter, repositories, repoSession);
225 return result.getArtifactResults().stream()
226 .filter(ArtifactResult::isResolved)
227 .map(ArtifactResult::getArtifact)
228 .collect(Collectors.toList());
229 } catch (PluginResolutionException e) {
230 throw new ExtensionResolutionException(extension, e.getCause());
231 } catch (InterpolationException e) {
232 throw new ExtensionResolutionException(extension, e);
233 }
234 }
235
236 private static Interpolator createInterpolator(MavenExecutionRequest request) {
237 StringSearchInterpolator interpolator = new StringSearchInterpolator();
238 interpolator.addValueSource(new MapBasedValueSource(request.getUserProperties()));
239 interpolator.addValueSource(new MapBasedValueSource(request.getSystemProperties()));
240 return interpolator;
241 }
242
243 static class SimpleSession extends DefaultSession {
244 SimpleSession(
245 MavenSession session,
246 RepositorySystem repositorySystem,
247 List<org.apache.maven.api.RemoteRepository> repositories) {
248 super(session, repositorySystem, repositories, null, null, null);
249 }
250
251 @Override
252 protected Session newSession(
253 MavenSession mavenSession, List<org.apache.maven.api.RemoteRepository> repositories) {
254 return new SimpleSession(mavenSession, getRepositorySystem(), repositories);
255 }
256
257 @Override
258 public <T extends Service> T getService(Class<T> clazz) throws NoSuchElementException {
259 if (clazz == ArtifactCoordinateFactory.class) {
260 return (T) new DefaultArtifactCoordinateFactory();
261 } else if (clazz == VersionParser.class) {
262 return (T) new DefaultVersionParser(new DefaultModelVersionParser(new GenericVersionScheme()));
263 } else if (clazz == VersionRangeResolver.class) {
264 return (T) new DefaultVersionRangeResolver(repositorySystem);
265 } else if (clazz == ArtifactResolver.class) {
266 return (T) new DefaultArtifactResolver();
267 } else if (clazz == ArtifactManager.class) {
268 return (T) new DefaultArtifactManager(this);
269 } else if (clazz == RepositoryFactory.class) {
270 return (T) new DefaultRepositoryFactory(new DefaultRemoteRepositoryManager(
271 new DefaultUpdatePolicyAnalyzer(), new DefaultChecksumPolicyProvider()));
272
273
274 }
275 throw new NoSuchElementException("No service for " + clazz.getName());
276 }
277 }
278 }