/*
* Copyright (c) 2009-2015
* IT-Consulting Stephan Schloepke (http://www.schloepke.de/)
* klemm software consulting Mirko Klemm (http://www.klemm-scs.com/)
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package org.jbasics.versionmanager;
import org.jbasics.checker.ContractCheck;
import org.jbasics.enviroment.JVMEnviroment;
import org.jbasics.exception.DelegatedException;
import org.jbasics.pattern.resolver.Resolver;
import org.jbasics.types.builders.MapBuilder;
import org.jbasics.types.factories.MapFactory;
import org.jbasics.types.tuples.Pair;
import org.jbasics.utilities.DataUtilities;
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.util.Collections;
import java.util.Map;
import java.util.Properties;
import java.util.logging.Level;
import java.util.logging.Logger;
public class VersionsResourceResolver implements Resolver<VersionInformation, VersionIdentifier> {
public static final String DEFAULT_RESOURCE_NAME = "version-info.properties"; //$NON-NLS-1$
public static final String GROUP_PROPERTY = "groupId"; //$NON-NLS-1$
public static final String ARTIFACT_PROPERTY = "artifactId"; //$NON-NLS-1$
public static final String VERSION_PROPERTY = "version"; //$NON-NLS-1$
private static final Logger LOGGER = Logger.getLogger(VersionsResourceResolver.class.getName());
private final String resourceName;
private final Map<VersionIdentifier, VersionInformation> versions;
public VersionsResourceResolver() {
this(null);
}
public VersionsResourceResolver(final String resourceName) {
this.resourceName = DataUtilities.coalesce(resourceName, VersionsResourceResolver.DEFAULT_RESOURCE_NAME);
this.versions = VersionsResourceResolver.scanVersions(this.resourceName);
}
public static Map<VersionIdentifier, VersionInformation> scanVersions(final String resourceName) {
if (VersionsResourceResolver.LOGGER.isLoggable(Level.FINEST)) {
VersionsResourceResolver.LOGGER.log(Level.FINEST, "Scanning version information under resource name {0}", resourceName); //$NON-NLS-1$
}
try {
return VersionsResourceResolver.scanVersions(Collections.list(JVMEnviroment.getContextClassLoader().getResources(
ContractCheck.mustNotBeNullOrTrimmedEmpty(resourceName, "resourceName"))));
} catch (final IOException e) {
throw DelegatedException.delegate(e);
}
}
public static Map<VersionIdentifier, VersionInformation> scanVersions(final Iterable<URL> resourceURLs) {
final MapBuilder<VersionIdentifier, VersionInformation> builder = new MapBuilder<>(MapFactory.<VersionIdentifier, VersionInformation> orderedMapFactory()).immutable();
for (final URL resourceURL : resourceURLs) {
final Pair<VersionIdentifier, VersionInformation> temp = VersionsResourceResolver.loadVersionInformation(resourceURL);
if (VersionsResourceResolver.LOGGER.isLoggable(Level.FINEST)) {
if (temp == null) {
VersionsResourceResolver.LOGGER.log(Level.FINEST, "Scanned {0} without result", resourceURL); //$NON-NLS-1$
} else {
VersionsResourceResolver.LOGGER.log(Level.FINEST,
"Scanned {0} and found {1}={2}", new Object[]{resourceURL, temp.first(), temp.second().getVersion()}); //$NON-NLS-1$
}
}
builder.putConditional(temp != null, temp);
}
return builder.build();
}
@SuppressWarnings("resource")
public static Pair<VersionIdentifier, VersionInformation> loadVersionInformation(final URL resourceURL) {
final Properties props = new Properties();
InputStream inp = null;
try {
inp = ContractCheck.mustNotBeNull(resourceURL, "resourceURL").openStream(); //$NON-NLS-1$
props.load(inp);
final String group = props.getProperty(VersionsResourceResolver.GROUP_PROPERTY);
final String artifact = props.getProperty(VersionsResourceResolver.ARTIFACT_PROPERTY);
final String version = props.getProperty(MavenVersionResolver.MAVEN_VERSION_PROPERTY);
if (group == null || artifact == null || version == null) {
return null;
}
final VersionIdentifier id = new VersionIdentifier(group, artifact);
return new Pair<VersionIdentifier, VersionInformation>(id, new VersionInformation(id, version));
} catch (final IOException e) {
return null;
} finally {
if (inp != null) {
try {
inp.close();
} catch (final IOException e) {
// silently ignore close problems
// we want to log here what happened
}
}
}
}
public String getResourceName() {
return this.resourceName;
}
@Override
public VersionInformation resolve(final VersionIdentifier request, final VersionInformation defaultResult) {
if (VersionsResourceResolver.LOGGER.isLoggable(Level.FINE)) {
VersionsResourceResolver.LOGGER.log(Level.FINE, "Trying to resolve version information for {0}", request); //$NON-NLS-1$
}
final VersionInformation result = this.versions.get(request);
if (result == null) {
if (VersionsResourceResolver.LOGGER.isLoggable(Level.FINE)) {
VersionsResourceResolver.LOGGER.log(Level.FINE, "Could not resolve any version information for {0}", request); //$NON-NLS-1$
}
return defaultResult;
}
if (VersionsResourceResolver.LOGGER.isLoggable(Level.FINE)) {
VersionsResourceResolver.LOGGER.log(Level.FINE, "Found version information for {0} with version", //$NON-NLS-1$
new Object[]{request, result.getVersion()});
}
return result;
}
}