/**
* (c) Copyright 2012 WibiData, Inc.
*
* See the NOTICE file distributed with this work for additional
* information regarding copyright ownership.
*
* Licensed 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.kiji.mapreduce.util;
import java.io.IOException;
import java.net.URL;
import java.net.URLDecoder;
import java.util.Enumeration;
import org.kiji.annotations.ApiAudience;
/** Utility class for dealing with Java Jar files and their contained classes. */
@ApiAudience.Private
public final class Jars {
/** No constructor for this utility class. */
private Jars() {
}
/**
* Finds the file path to the jar that contains a particular class.
* Method mostly cloned from o.a.h.mapred.JobConf.findContainingJar().
*
* @param classObj The class of interest.
* @return The path to the jar that contains <code>classObj</code>.
* @throws ClassNotFoundException If the class cannot be found in a jar.
* @throws IOException If there is a problem reading jars from the file system.
*/
public static String getJarPathForClass(Class<? extends Object> classObj)
throws ClassNotFoundException, IOException {
ClassLoader loader = classObj.getClassLoader();
String classFile = classObj.getName().replaceAll("\\.", "/") + ".class";
for (Enumeration<URL> itr = loader.getResources(classFile); itr.hasMoreElements();) {
URL url = itr.nextElement();
if ("jar".equals(url.getProtocol())) {
String toReturn = url.getPath();
if (toReturn.startsWith("file:")) {
toReturn = toReturn.substring("file:".length());
}
// URLDecoder is a misnamed class, since it actually decodes
// x-www-form-urlencoded MIME type rather than actual
// URL encoding (which the file path has). Therefore it would
// decode +s to ' 's which is incorrect (spaces are actually
// either unencoded or encoded as "%20"). Replace +s first, so
// that they are kept sacred during the decoding process.
toReturn = toReturn.replaceAll("\\+", "%2B");
toReturn = URLDecoder.decode(toReturn, "UTF-8");
return toReturn.replaceAll("!.*$", "");
}
}
throw new ClassNotFoundException(
"Unable to find containing jar for class " + classObj.getName());
}
}