ro.isdc.wro.manager.callback.LifecycleCallbackRegistry Maven / Gradle / Ivy
/*
* Copyright (c) 2011. All rights reserved.
*/
package ro.isdc.wro.manager.callback;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import ro.isdc.wro.config.Context;
import ro.isdc.wro.model.resource.Resource;
import ro.isdc.wro.util.Function;
import ro.isdc.wro.util.ObjectFactory;
/**
* Register all available callbacks. The registry acts as a {@link LifecycleCallback} itself whose implementation
* delegate the call to registered callbacks. The registry will handle any runtime exceptions thrown by callbacks, in
* order to allow successful lifecycle execution.
*
* @author Alex Objelean
* @created Created on 8 Dec 2011
* @since 1.4.3
*/
public class LifecycleCallbackRegistry
implements LifecycleCallback {
private static final Logger LOG = LoggerFactory.getLogger(LifecycleCallbackRegistry.class);
/**
* The list of registered callbacks.
*/
private final List> callbackFactoryList = new ArrayList>();
private final Map> map = new ConcurrentHashMap>();
/**
* Register a callback using a factory responsible for callback instantiation.
*
* @param callbackFactory
* the factory used to instantiate callbacks.
*/
public void registerCallback(final ObjectFactory callbackFactory) {
callbackFactoryList.add(callbackFactory);
}
private List getCallbacks() {
final String key = Context.getCorrelationId();
List callbacks = map.get(key);
if (callbacks == null) {
callbacks = initCallbacks();
map.put(key, callbacks);
}
return callbacks;
}
protected List initCallbacks() {
final List callbacks = new ArrayList();
for (final ObjectFactory callbackFactory : callbackFactoryList) {
callbacks.add(callbackFactory.create());
}
return callbacks;
}
/**
* {@inheritDoc}
*/
@Override
public void onBeforeModelCreated() {
forEachCallbackDo(new Function() {
@Override
public Void apply(final LifecycleCallback input)
throws Exception {
input.onBeforeModelCreated();
return null;
}
});
}
/**
* {@inheritDoc}
*/
@Override
public void onAfterModelCreated() {
forEachCallbackDo(new Function() {
@Override
public Void apply(final LifecycleCallback input)
throws Exception {
input.onAfterModelCreated();
return null;
}
});
}
/**
* {@inheritDoc}
*/
@Override
public void onBeforePreProcess() {
forEachCallbackDo(new Function() {
@Override
public Void apply(final LifecycleCallback input)
throws Exception {
input.onBeforePreProcess();
return null;
}
});
}
/**
* {@inheritDoc}
*/
@Override
public void onAfterPreProcess() {
forEachCallbackDo(new Function() {
@Override
public Void apply(final LifecycleCallback input)
throws Exception {
input.onAfterPreProcess();
return null;
}
});
}
/**
* {@inheritDoc}
*/
@Override
public void onBeforePostProcess() {
forEachCallbackDo(new Function() {
@Override
public Void apply(final LifecycleCallback input)
throws Exception {
input.onBeforePostProcess();
return null;
}
});
}
/**
* {@inheritDoc}
*/
@Override
public void onAfterPostProcess() {
forEachCallbackDo(new Function() {
@Override
public Void apply(final LifecycleCallback input)
throws Exception {
input.onAfterPostProcess();
return null;
}
});
}
/**
* {@inheritDoc}
*/
@Override
public void onBeforeMerge() {
forEachCallbackDo(new Function() {
@Override
public Void apply(final LifecycleCallback input)
throws Exception {
input.onBeforeMerge();
return null;
}
});
}
@Override
public void onAfterMerge() {
forEachCallbackDo(new Function() {
@Override
public Void apply(final LifecycleCallback input)
throws Exception {
input.onAfterMerge();
return null;
}
});
}
/**
* {@inheritDoc}
*/
@Override
public void onProcessingComplete() {
forEachCallbackDo(new Function() {
@Override
public Void apply(final LifecycleCallback input)
throws Exception {
input.onProcessingComplete();
return null;
}
});
}
@Override
public void onResourceChanged(final Resource resource) {
forEachCallbackDo(new Function() {
@Override
public Void apply(final LifecycleCallback input)
throws Exception {
input.onResourceChanged(resource);
return null;
}
});
}
private void forEachCallbackDo(final Function func) {
for (final LifecycleCallback callback : getCallbacks()) {
try {
func.apply(callback);
} catch (final Exception e) {
LOG.error("Problem invoking callback", e);
onException(e);
} finally {
map.remove(Context.getCorrelationId());
}
}
}
/**
* Invoked when a callback fails. By default exception is ignored.
*
* @param e
* {@link Exception} thrown by the fallback.
*/
protected void onException(final Exception e) {
}
}