1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19 package org.apache.maven.repository.metadata;
20
21 import javax.inject.Inject;
22 import javax.inject.Named;
23 import javax.inject.Singleton;
24
25 import java.util.ArrayList;
26 import java.util.List;
27 import java.util.TreeSet;
28
29 import org.apache.maven.artifact.ArtifactScopeEnum;
30
31
32
33
34
35
36 @Named
37 @Singleton
38 public class DefaultGraphConflictResolver implements GraphConflictResolver {
39
40
41
42 @Inject
43 protected GraphConflictResolutionPolicy policy;
44
45
46
47 public MetadataGraph resolveConflicts(MetadataGraph graph, ArtifactScopeEnum scope)
48 throws GraphConflictResolutionException {
49 if (policy == null) {
50 throw new GraphConflictResolutionException("no GraphConflictResolutionPolicy injected");
51 }
52
53 if (graph == null) {
54 return null;
55 }
56
57 final MetadataGraphVertex entry = graph.getEntry();
58 if (entry == null) {
59 return null;
60 }
61
62 if (graph.isEmpty()) {
63 throw new GraphConflictResolutionException("graph with an entry, but not vertices do not exist");
64 }
65
66 if (graph.isEmptyEdges()) {
67 return null;
68 }
69
70 final TreeSet<MetadataGraphVertex> vertices = graph.getVertices();
71
72 try {
73
74 if (vertices.size() == 1) {
75 return new MetadataGraph(entry);
76 }
77
78 final ArtifactScopeEnum requestedScope = ArtifactScopeEnum.checkScope(scope);
79
80 MetadataGraph res = new MetadataGraph(vertices.size());
81 res.setVersionedVertices(false);
82 res.setScopedVertices(false);
83
84 MetadataGraphVertex resEntry = res.addVertex(entry.getMd());
85 res.setEntry(resEntry);
86
87 res.setScope(requestedScope);
88
89 for (MetadataGraphVertex v : vertices) {
90 final List<MetadataGraphEdge> ins = graph.getIncidentEdges(v);
91 final MetadataGraphEdge edge = cleanEdges(v, ins, requestedScope);
92
93 if (edge == null) {
94 if (entry.equals(v)) {
95
96 res.getEntry().getMd().setWhy("This is a graph entry point. No links.");
97 } else {
98
99
100
101
102
103 }
104 } else {
105
106
107 ArtifactMetadata md = v.getMd();
108 ArtifactMetadata newMd = new ArtifactMetadata(
109 md.getGroupId(),
110 md.getArtifactId(),
111 edge.getVersion(),
112 md.getType(),
113 md.getScopeAsEnum(),
114 md.getClassifier(),
115 edge.getArtifactUri(),
116 edge.getSource() == null
117 ? ""
118 : edge.getSource().getMd().toString(),
119 edge.isResolved(),
120 edge.getTarget() == null
121 ? null
122 : edge.getTarget().getMd().getError());
123 MetadataGraphVertex newV = res.addVertex(newMd);
124 MetadataGraphVertex sourceV = res.addVertex(edge.getSource().getMd());
125
126 res.addEdge(sourceV, newV, edge);
127 }
128 }
129
130
131
132
133 return findLinkedSubgraph(res);
134 } catch (MetadataResolutionException e) {
135 throw new GraphConflictResolutionException(e);
136 }
137 }
138
139
140 private MetadataGraph findLinkedSubgraph(MetadataGraph g) {
141 if (g.getVertices().size() == 1) {
142 return g;
143 }
144
145 List<MetadataGraphVertex> visited = new ArrayList<>(g.getVertices().size());
146 visit(g.getEntry(), visited, g);
147
148 List<MetadataGraphVertex> dropList = new ArrayList<>(g.getVertices().size());
149
150
151 for (MetadataGraphVertex v : g.getVertices()) {
152 if (!visited.contains(v)) {
153 dropList.add(v);
154 }
155 }
156
157 if (dropList.size() < 1) {
158 return g;
159 }
160
161
162 TreeSet<MetadataGraphVertex> vertices = g.getVertices();
163 for (MetadataGraphVertex v : dropList) {
164 vertices.remove(v);
165 }
166
167 return g;
168 }
169
170
171 private void visit(MetadataGraphVertex from, List<MetadataGraphVertex> visited, MetadataGraph graph) {
172 if (visited.contains(from)) {
173 return;
174 }
175
176 visited.add(from);
177
178 List<MetadataGraphEdge> exitList = graph.getExcidentEdges(from);
179
180 if (exitList != null && exitList.size() > 0) {
181 for (MetadataGraphEdge e : graph.getExcidentEdges(from)) {
182 visit(e.getTarget(), visited, graph);
183 }
184 }
185 }
186
187
188 private MetadataGraphEdge cleanEdges(
189 MetadataGraphVertex v, List<MetadataGraphEdge> edges, ArtifactScopeEnum scope) {
190 if (edges == null || edges.isEmpty()) {
191 return null;
192 }
193
194 if (edges.size() == 1) {
195 MetadataGraphEdge e = edges.get(0);
196 if (scope.encloses(e.getScope())) {
197 return e;
198 }
199
200 return null;
201 }
202
203 MetadataGraphEdge res = null;
204
205 for (MetadataGraphEdge e : edges) {
206 if (!scope.encloses(e.getScope())) {
207 continue;
208 }
209
210 if (res == null) {
211 res = e;
212 } else {
213 res = policy.apply(e, res);
214 }
215 }
216
217 return res;
218 }
219
220
221 }