001/* 002 * Licensed to the Apache Software Foundation (ASF) under one 003 * or more contributor license agreements. See the NOTICE file 004 * distributed with this work for additional information 005 * regarding copyright ownership. The ASF licenses this file 006 * to you under the Apache License, Version 2.0 (the 007 * "License"); you may not use this file except in compliance 008 * with the License. You may obtain a copy of the License at 009 * 010 * http://www.apache.org/licenses/LICENSE-2.0 011 * 012 * Unless required by applicable law or agreed to in writing, 013 * software distributed under the License is distributed on an 014 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 015 * KIND, either express or implied. See the License for the 016 * specific language governing permissions and limitations 017 * under the License. 018 */ 019package org.eclipse.aether.artifact; 020 021import java.io.File; 022import java.nio.file.Path; 023import java.util.Collections; 024import java.util.HashMap; 025import java.util.Map; 026import java.util.regex.Matcher; 027import java.util.regex.Pattern; 028 029/** 030 * A simple artifact. <em>Note:</em> Instances of this class are immutable and the exposed mutators return new objects 031 * rather than changing the current instance. 032 */ 033public final class DefaultArtifact extends AbstractArtifact { 034 private static final Pattern COORDINATE_PATTERN = 035 Pattern.compile("([^: ]+):([^: ]+)(:([^: ]*)(:([^: ]+))?)?:([^: ]+)"); 036 037 private final String groupId; 038 039 private final String artifactId; 040 041 private final String version; 042 043 private final String classifier; 044 045 private final String extension; 046 047 private final Path path; 048 049 private final Map<String, String> properties; 050 051 /** 052 * Creates a new artifact with the specified coordinates. If not specified in the artifact coordinates, the 053 * artifact's extension defaults to {@code jar} and classifier to an empty string. 054 * 055 * @param coords The artifact coordinates in the format 056 * {@code <groupId>:<artifactId>[:<extension>[:<classifier>]]:<version>}, must not be {@code null}. 057 * 058 * @throws IllegalArgumentException If the artifact coordinates found in {@code coords} do not match the expected 059 * format. 060 */ 061 public DefaultArtifact(String coords) { 062 this(coords, null, null); 063 } 064 065 /** 066 * Creates a new artifact with the specified coordinates and properties. If not specified in the artifact 067 * coordinates, the artifact's extension defaults to {@code jar} and classifier to an empty string. 068 * 069 * @param coords The artifact coordinates in the format 070 * {@code <groupId>:<artifactId>[:<extension>[:<classifier>]]:<version>}, must not be {@code null}. 071 * @param properties The artifact properties, may be {@code null}. 072 * 073 * @throws IllegalArgumentException If the artifact coordinates found in {@code coords} do not match the expected 074 * format. 075 */ 076 public DefaultArtifact(String coords, Map<String, String> properties) { 077 this(coords, properties, null); 078 } 079 080 /** 081 * Creates a new artifact with the specified coordinates and type. If not specified in the artifact coordinates, 082 * the artifact's extension defaults to type extension (or "jar" if type is {@code null}) and 083 * classifier to type extension (or "" if type is {@code null}). 084 * 085 * @param coords The artifact coordinates in the format 086 * {@code <groupId>:<artifactId>[:<extension>[:<classifier>]]:<version>}, must not be {@code null}. 087 * @param type The artifact type, may be {@code null}. 088 * 089 * @throws IllegalArgumentException If the artifact coordinates found in {@code coords} do not match the expected 090 * format. 091 */ 092 public DefaultArtifact(String coords, ArtifactType type) { 093 this(coords, null, type); 094 } 095 096 /** 097 * Creates a new artifact with the specified coordinates, properties and type. If not specified in the artifact 098 * coordinates, the artifact's extension defaults to type extension (or "jar" if type is {@code null}) and 099 * classifier to type extension (or "" if type is {@code null}). 100 * 101 * @param coords The artifact coordinates in the format 102 * {@code <groupId>:<artifactId>[:<extension>[:<classifier>]]:<version>}, must not be {@code null}. 103 * @param properties The artifact properties, may be {@code null}. 104 * @param type The artifact type, may be {@code null}. 105 * 106 * @throws IllegalArgumentException If the artifact coordinates found in {@code coords} do not match the expected 107 * format. 108 */ 109 public DefaultArtifact(String coords, Map<String, String> properties, ArtifactType type) { 110 Matcher m = COORDINATE_PATTERN.matcher(coords); 111 if (!m.matches()) { 112 throw new IllegalArgumentException("Bad artifact coordinates " + coords 113 + ", expected format is <groupId>:<artifactId>[:<extension>[:<classifier>]]:<version>"); 114 } 115 groupId = m.group(1); 116 artifactId = m.group(2); 117 extension = get(m.group(4), type == null ? "jar" : type.getExtension()); 118 classifier = get(m.group(6), type == null ? "" : type.getClassifier()); 119 this.version = emptify(m.group(7)); 120 this.path = null; 121 this.properties = mergeArtifactProperties(properties, (type != null) ? type.getProperties() : null); 122 } 123 124 private static String get(String value, String defaultValue) { 125 return (value == null || value.isEmpty()) ? defaultValue : value; 126 } 127 128 /** 129 * Creates a new artifact with the specified coordinates and no classifier. Passing {@code null} for any of the 130 * coordinates is equivalent to specifying an empty string. 131 * 132 * @param groupId The group identifier of the artifact, may be {@code null}. 133 * @param artifactId The artifact identifier of the artifact, may be {@code null}. 134 * @param extension The file extension of the artifact, may be {@code null}. 135 * @param version The version of the artifact, may be {@code null}. 136 */ 137 public DefaultArtifact(String groupId, String artifactId, String extension, String version) { 138 this(groupId, artifactId, "", extension, version); 139 } 140 141 /** 142 * Creates a new artifact with the specified coordinates. Passing {@code null} for any of the coordinates is 143 * equivalent to specifying an empty string. 144 * 145 * @param groupId The group identifier of the artifact, may be {@code null}. 146 * @param artifactId The artifact identifier of the artifact, may be {@code null}. 147 * @param classifier The classifier of the artifact, may be {@code null}. 148 * @param extension The file extension of the artifact, may be {@code null}. 149 * @param version The version of the artifact, may be {@code null}. 150 */ 151 public DefaultArtifact(String groupId, String artifactId, String classifier, String extension, String version) { 152 this(groupId, artifactId, classifier, extension, version, null, (File) null); 153 } 154 155 /** 156 * Creates a new artifact with the specified coordinates. Passing {@code null} for any of the coordinates is 157 * equivalent to specifying an empty string. The optional artifact type provided to this constructor will be used to 158 * determine the artifact's classifier and file extension if the corresponding arguments for this constructor are 159 * {@code null}. 160 * 161 * @param groupId The group identifier of the artifact, may be {@code null}. 162 * @param artifactId The artifact identifier of the artifact, may be {@code null}. 163 * @param classifier The classifier of the artifact, may be {@code null}. 164 * @param extension The file extension of the artifact, may be {@code null}. 165 * @param version The version of the artifact, may be {@code null}. 166 * @param type The artifact type from which to query classifier, file extension and properties, may be {@code null}. 167 */ 168 public DefaultArtifact( 169 String groupId, String artifactId, String classifier, String extension, String version, ArtifactType type) { 170 this(groupId, artifactId, classifier, extension, version, null, type); 171 } 172 173 /** 174 * Creates a new artifact with the specified coordinates and properties. Passing {@code null} for any of the 175 * coordinates is equivalent to specifying an empty string. The optional artifact type provided to this constructor 176 * will be used to determine the artifact's classifier and file extension if the corresponding arguments for this 177 * constructor are {@code null}. If the artifact type specifies properties, those will get merged with the 178 * properties passed directly into the constructor, with the latter properties taking precedence. 179 * 180 * @param groupId The group identifier of the artifact, may be {@code null}. 181 * @param artifactId The artifact identifier of the artifact, may be {@code null}. 182 * @param classifier The classifier of the artifact, may be {@code null}. 183 * @param extension The file extension of the artifact, may be {@code null}. 184 * @param version The version of the artifact, may be {@code null}. 185 * @param properties The properties of the artifact, may be {@code null} if none. 186 * @param type The artifact type from which to query classifier, file extension and properties, may be {@code null}. 187 */ 188 public DefaultArtifact( 189 String groupId, 190 String artifactId, 191 String classifier, 192 String extension, 193 String version, 194 Map<String, String> properties, 195 ArtifactType type) { 196 this.groupId = emptify(groupId); 197 this.artifactId = emptify(artifactId); 198 if (classifier != null || type == null) { 199 this.classifier = emptify(classifier); 200 } else { 201 this.classifier = emptify(type.getClassifier()); 202 } 203 if (extension != null || type == null) { 204 this.extension = emptify(extension); 205 } else { 206 this.extension = emptify(type.getExtension()); 207 } 208 this.version = emptify(version); 209 this.path = null; 210 this.properties = mergeArtifactProperties(properties, (type != null) ? type.getProperties() : null); 211 } 212 213 private static Map<String, String> mergeArtifactProperties( 214 Map<String, String> artifactProperties, Map<String, String> typeDefaultProperties) { 215 Map<String, String> properties; 216 217 if (artifactProperties == null || artifactProperties.isEmpty()) { 218 if (typeDefaultProperties == null || typeDefaultProperties.isEmpty()) { 219 properties = Collections.emptyMap(); 220 } else { 221 // type default properties are already unmodifiable 222 return typeDefaultProperties; 223 } 224 } else { 225 properties = new HashMap<>(); 226 if (typeDefaultProperties != null) { 227 properties.putAll(typeDefaultProperties); 228 } 229 if (artifactProperties != null) { 230 properties.putAll(artifactProperties); 231 } 232 properties = Collections.unmodifiableMap(properties); 233 } 234 235 return properties; 236 } 237 238 /** 239 * Creates a new artifact with the specified coordinates, properties and file. Passing {@code null} for any of the 240 * coordinates is equivalent to specifying an empty string. 241 * 242 * @param groupId The group identifier of the artifact, may be {@code null}. 243 * @param artifactId The artifact identifier of the artifact, may be {@code null}. 244 * @param classifier The classifier of the artifact, may be {@code null}. 245 * @param extension The file extension of the artifact, may be {@code null}. 246 * @param version The version of the artifact, may be {@code null}. 247 * @param properties The properties of the artifact, may be {@code null} if none. 248 * @param file The resolved file of the artifact, may be {@code null}. 249 */ 250 public DefaultArtifact( 251 String groupId, 252 String artifactId, 253 String classifier, 254 String extension, 255 String version, 256 Map<String, String> properties, 257 File file) { 258 this.groupId = emptify(groupId); 259 this.artifactId = emptify(artifactId); 260 this.classifier = emptify(classifier); 261 this.extension = emptify(extension); 262 this.version = emptify(version); 263 this.path = file != null ? file.toPath() : null; 264 this.properties = copyProperties(properties); 265 } 266 267 /** 268 * Creates a new artifact with the specified coordinates, properties and file. Passing {@code null} for any of the 269 * coordinates is equivalent to specifying an empty string. 270 * 271 * @param groupId The group identifier of the artifact, may be {@code null}. 272 * @param artifactId The artifact identifier of the artifact, may be {@code null}. 273 * @param classifier The classifier of the artifact, may be {@code null}. 274 * @param extension The file extension of the artifact, may be {@code null}. 275 * @param version The version of the artifact, may be {@code null}. 276 * @param properties The properties of the artifact, may be {@code null} if none. 277 * @param path The resolved file of the artifact, may be {@code null}. 278 */ 279 public DefaultArtifact( 280 String groupId, 281 String artifactId, 282 String classifier, 283 String extension, 284 String version, 285 Map<String, String> properties, 286 Path path) { 287 this.groupId = emptify(groupId); 288 this.artifactId = emptify(artifactId); 289 this.classifier = emptify(classifier); 290 this.extension = emptify(extension); 291 this.version = emptify(version); 292 this.path = path; 293 this.properties = copyProperties(properties); 294 } 295 296 DefaultArtifact( 297 String groupId, 298 String artifactId, 299 String classifier, 300 String extension, 301 String version, 302 Path path, 303 Map<String, String> properties) { 304 // NOTE: This constructor assumes immutability of the provided properties, for internal use only 305 this.groupId = emptify(groupId); 306 this.artifactId = emptify(artifactId); 307 this.classifier = emptify(classifier); 308 this.extension = emptify(extension); 309 this.version = emptify(version); 310 this.path = path; 311 this.properties = properties; 312 } 313 314 private static String emptify(String str) { 315 return (str == null) ? "" : str; 316 } 317 318 public String getGroupId() { 319 return groupId; 320 } 321 322 public String getArtifactId() { 323 return artifactId; 324 } 325 326 public String getVersion() { 327 return version; 328 } 329 330 public String getClassifier() { 331 return classifier; 332 } 333 334 public String getExtension() { 335 return extension; 336 } 337 338 @Deprecated 339 public File getFile() { 340 return path != null ? path.toFile() : null; 341 } 342 343 public Path getPath() { 344 return path; 345 } 346 347 public Map<String, String> getProperties() { 348 return properties; 349 } 350}