JBoss6VFS.java
/*
* Copyright 2009-2024 the original author or authors.
*
* 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
*
* https://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.apache.ibatis.io;
import java.io.IOException;
import java.lang.reflect.Method;
import java.net.URL;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.locks.ReentrantLock;
import org.apache.ibatis.logging.Log;
import org.apache.ibatis.logging.LogFactory;
/**
* A {@link VFS} implementation that works with the VFS API provided by JBoss 6.
*
* @author Ben Gunter
*/
public class JBoss6VFS extends VFS {
private static final Log log = LogFactory.getLog(JBoss6VFS.class);
private static final ReentrantLock lock = new ReentrantLock();
/** A class that mimics a tiny subset of the JBoss VirtualFile class. */
static class VirtualFile {
static Class<?> VirtualFile;
static Method getPathNameRelativeTo;
static Method getChildrenRecursively;
Object virtualFile;
VirtualFile(Object virtualFile) {
this.virtualFile = virtualFile;
}
String getPathNameRelativeTo(VirtualFile parent) {
try {
return invoke(getPathNameRelativeTo, virtualFile, parent.virtualFile);
} catch (IOException e) {
// This exception is not thrown by the called method
log.error("This should not be possible. VirtualFile.getPathNameRelativeTo() threw IOException.");
return null;
}
}
List<VirtualFile> getChildren() throws IOException {
List<?> objects = invoke(getChildrenRecursively, virtualFile);
List<VirtualFile> children = new ArrayList<>(objects.size());
for (Object object : objects) {
children.add(new VirtualFile(object));
}
return children;
}
}
/** A class that mimics a tiny subset of the JBoss VFS class. */
static class VFS {
static Class<?> VFS;
static Method getChild;
private VFS() {
// Prevent Instantiation
}
static VirtualFile getChild(URL url) throws IOException {
Object o = invoke(getChild, VFS, url);
return o == null ? null : new VirtualFile(o);
}
}
/** Flag that indicates if this VFS is valid for the current environment. */
private static Boolean valid;
/** Find all the classes and methods that are required to access the JBoss 6 VFS. */
protected static void initialize() {
lock.lock();
try {
if (valid == null) {
// Assume valid. It will get flipped later if something goes wrong.
valid = Boolean.TRUE;
// Look up and verify required classes
VFS.VFS = checkNotNull(getClass("org.jboss.vfs.VFS"));
VirtualFile.VirtualFile = checkNotNull(getClass("org.jboss.vfs.VirtualFile"));
// Look up and verify required methods
VFS.getChild = checkNotNull(getMethod(VFS.VFS, "getChild", URL.class));
VirtualFile.getChildrenRecursively = checkNotNull(getMethod(VirtualFile.VirtualFile, "getChildrenRecursively"));
VirtualFile.getPathNameRelativeTo = checkNotNull(
getMethod(VirtualFile.VirtualFile, "getPathNameRelativeTo", VirtualFile.VirtualFile));
// Verify that the API has not changed
checkReturnType(VFS.getChild, VirtualFile.VirtualFile);
checkReturnType(VirtualFile.getChildrenRecursively, List.class);
checkReturnType(VirtualFile.getPathNameRelativeTo, String.class);
}
} finally {
lock.unlock();
}
}
/**
* Verifies that the provided object reference is null. If it is null, then this VFS is marked as invalid for the
* current environment.
*
* @param <T>
* the generic type
* @param object
* The object reference to check for null.
*
* @return the t
*/
protected static <T> T checkNotNull(T object) {
if (object == null) {
setInvalid();
}
return object;
}
/**
* Verifies that the return type of method is what it is expected to be. If it is not, then this VFS is marked as
* invalid for the current environment.
*
* @param method
* The method whose return type is to be checked.
* @param expected
* A type to which the method's return type must be assignable.
*
* @see Class#isAssignableFrom(Class)
*/
protected static void checkReturnType(Method method, Class<?> expected) {
if (method != null && !expected.isAssignableFrom(method.getReturnType())) {
log.error("Method " + method.getClass().getName() + "." + method.getName() + "(..) should return "
+ expected.getName() + " but returns " + method.getReturnType().getName() + " instead.");
setInvalid();
}
}
/**
* Mark this {@link VFS} as invalid for the current environment.
*/
protected static void setInvalid() {
if (JBoss6VFS.valid.booleanValue()) {
log.debug("JBoss 6 VFS API is not available in this environment.");
JBoss6VFS.valid = Boolean.FALSE;
}
}
static {
initialize();
}
@Override
public boolean isValid() {
return valid;
}
@Override
public List<String> list(URL url, String path) throws IOException {
VirtualFile directory;
directory = VFS.getChild(url);
if (directory == null) {
return Collections.emptyList();
}
if (!path.endsWith("/")) {
path += "/";
}
List<VirtualFile> children = directory.getChildren();
List<String> names = new ArrayList<>(children.size());
for (VirtualFile vf : children) {
names.add(path + vf.getPathNameRelativeTo(directory));
}
return names;
}
}