blob: ee6b6bc71384cb4c76855b46e537718f228b39d3 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.apache.sling.commons.fsclassloader.impl;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.ArrayList;
import java.util.List;
import org.apache.felix.scr.annotations.Activate;
import org.apache.felix.scr.annotations.Component;
import org.apache.felix.scr.annotations.Deactivate;
import org.apache.felix.scr.annotations.Property;
import org.apache.felix.scr.annotations.Reference;
import org.apache.felix.scr.annotations.Service;
import org.apache.sling.commons.classloader.ClassLoaderWriter;
import org.apache.sling.commons.classloader.DynamicClassLoaderManager;
import org.osgi.framework.Bundle;
import org.osgi.framework.Constants;
import org.osgi.framework.ServiceReference;
import org.osgi.service.component.ComponentContext;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* The <code>FSClassLoaderProvider</code> is a dynamic class loader provider
* which uses the file system to store and read class files from.
*
*/
@Component
@Service(value={ClassLoaderWriter.class}, serviceFactory = true)
@Property( name=Constants.SERVICE_RANKING, intValue=100)
public class FSClassLoaderProvider
implements ClassLoaderWriter {
/** File root */
private File root;
/** File root URL */
private URL rootURL;
/** Current class loader */
private FSDynamicClassLoader loader;
private final Logger logger = LoggerFactory.getLogger(this.getClass());
@Reference(
referenceInterface = DynamicClassLoaderManager.class,
bind = "bindDynamicClassLoaderManager",
unbind = "unbindDynamicClassLoaderManager")
private ServiceReference dynamicClassLoaderManager;
/** The bundle asking for this service instance */
private Bundle callerBundle;
/**
* Activate this component.
* Create the root directory.
* @param componentContext
* @throws MalformedURLException
*/
@Activate
protected void activate(final ComponentContext componentContext) throws MalformedURLException {
// get the file root
this.root = new File(componentContext.getBundleContext().getDataFile(""), "classes");
this.root.mkdirs();
this.rootURL = this.root.toURI().toURL();
this.callerBundle = componentContext.getUsingBundle();
}
/**
* Deactivate this component.
* Create the root directory.
*/
@Deactivate
protected void deactivate() {
this.root = null;
this.rootURL = null;
this.destroyClassLoader();
}
/**
* Called to handle binding the DynamicClassLoaderManager service
* reference
*/
@SuppressWarnings("unused")
private void bindDynamicClassLoaderManager(final ServiceReference ref) {
this.dynamicClassLoaderManager = ref;
}
/**
* Called to handle unbinding of the DynamicClassLoaderManager service
* reference
*/
@SuppressWarnings("unused")
private void unbindDynamicClassLoaderManager(final ServiceReference ref) {
if (this.dynamicClassLoaderManager == ref) {
this.dynamicClassLoaderManager = null;
}
}
private void destroyClassLoader() {
final ClassLoader rcl = this.loader;
if (rcl != null) {
this.loader = null;
final ServiceReference localDynamicClassLoaderManager = this.dynamicClassLoaderManager;
final Bundle localCallerBundle = this.callerBundle;
if ( localDynamicClassLoaderManager != null && localCallerBundle != null ) {
localCallerBundle.getBundleContext().ungetService(localDynamicClassLoaderManager);
}
}
}
/**
* @see org.apache.sling.commons.classloader.ClassLoaderWriter#getClassLoader()
*/
public ClassLoader getClassLoader() {
synchronized ( this ) {
if ( loader == null || !loader.isLive() ) {
this.destroyClassLoader();
// get the dynamic class loader for the bundle using this
// class loader writer
final DynamicClassLoaderManager dclm = (DynamicClassLoaderManager) this.callerBundle.getBundleContext().getService(
this.dynamicClassLoaderManager);
loader = new FSDynamicClassLoader(new URL[] {this.rootURL}, dclm.getDynamicClassLoader());
}
return this.loader;
}
}
private void checkClassLoader(final String filePath) {
if ( filePath.endsWith(".class") ) {
// remove store directory and .class
final String path = filePath.substring(this.root.getAbsolutePath().length() + 1, filePath.length() - 6);
// convert to a class name
final String className = path.replace(File.separatorChar, '.');
synchronized ( this ) {
final FSDynamicClassLoader currentLoader = this.loader;
if ( currentLoader != null ) {
currentLoader.check(className);
}
}
}
}
//---------- SCR Integration ----------------------------------------------
private boolean deleteRecursive(final File f, final List<String> names) {
if ( f.isDirectory() ) {
for(final File c : f.listFiles()) {
if ( !deleteRecursive(c, names) ) {
return false;
}
}
}
names.add(f.getAbsolutePath());
return f.delete();
}
/**
* @see org.apache.sling.commons.classloader.ClassLoaderWriter#delete(java.lang.String)
*/
public boolean delete(final String name) {
final String path = cleanPath(name);
final File file = new File(path);
if ( file.exists() ) {
final List<String> names = new ArrayList<String>();
final boolean result = deleteRecursive(file, names);
logger.debug("Deleted {} : {}", name, result);
if ( result ) {
for(final String n : names ) {
this.checkClassLoader(n);
}
}
return result;
}
// file does not exist so we return false
return false;
}
/**
* @see org.apache.sling.commons.classloader.ClassLoaderWriter#getOutputStream(java.lang.String)
*/
public OutputStream getOutputStream(final String name) {
logger.debug("Get stream for {}", name);
final String path = cleanPath(name);
final File file = new File(path);
final File parentDir = file.getParentFile();
if ( !parentDir.exists() ) {
parentDir.mkdirs();
}
try {
if ( file.exists() ) {
this.checkClassLoader(path);
}
return new FileOutputStream(path);
} catch (FileNotFoundException e) {
throw new RuntimeException(e);
}
}
/**
* @see org.apache.sling.commons.classloader.ClassLoaderWriter#rename(java.lang.String, java.lang.String)
*/
public boolean rename(final String oldName, final String newName) {
logger.debug("Rename {} to {}", oldName, newName);
final String oldPath = cleanPath(oldName);
final String newPath = cleanPath(newName);
final File old = new File(oldPath);
final boolean result = old.renameTo(new File(newPath));
if ( result ) {
this.checkClassLoader(oldPath);
this.checkClassLoader(newPath);
}
return result;
}
/**
* Clean the path by converting slashes to the correct format
* and prefixing the root directory.
* @param path The path
* @return The file path
*/
private String cleanPath(String path) {
// replace backslash by slash
path = path.replace('\\', '/');
// cut off trailing slash
while (path.endsWith("/")) {
path = path.substring(0, path.length() - 1);
}
if ( File.separatorChar != '/') {
path = path.replace('/', File.separatorChar);
}
return this.root.getAbsolutePath() + path;
}
/**
* @see org.apache.sling.commons.classloader.ClassLoaderWriter#getInputStream(java.lang.String)
*/
public InputStream getInputStream(final String name)
throws IOException {
logger.debug("Get input stream of {}", name);
final String path = cleanPath(name);
final File file = new File(path);
return new FileInputStream(file);
}
/**
* @see org.apache.sling.commons.classloader.ClassLoaderWriter#getLastModified(java.lang.String)
*/
public long getLastModified(final String name) {
logger.debug("Get last modified of {}", name);
final String path = cleanPath(name);
final File file = new File(path);
if ( file.exists() ) {
return file.lastModified();
}
// fallback to "non-existant" in case of problems
return -1;
}
}