imported cnr-notifications-common and cnr-blackboard-common

This commit is contained in:
Claudio Atzori 2019-05-31 11:45:07 +02:00
parent 11c100f627
commit ec964c0c47
43 changed files with 2556 additions and 1 deletions

View File

@ -161,6 +161,10 @@
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context-support</artifactId>
</dependency>
<!-- CXF -->

View File

@ -0,0 +1,71 @@
package eu.dnetlib.enabling.actions;
import javax.annotation.PostConstruct;
import org.springframework.beans.factory.annotation.Required;
import eu.dnetlib.enabling.actions.SubscriptionAction;
import eu.dnetlib.enabling.tools.Enableable;
/**
* Common subscription action.
*
* @author marko
*
*/
public abstract class AbstractSubscriptionAction implements SubscriptionAction, Enableable {
/**
* topic expression.
*/
private String topicExpression;
/**
* topic prefix.
*/
private String topicPrefix;
/**
* true if enabled.
*/
private boolean enabled = true;
/**
* init the default topic prefix.
*/
@PostConstruct
public void init() {
if (topicPrefix == null)
topicPrefix = getTopicExpression().replace("/", ".");
}
@Override
public String getTopicExpression() {
return topicExpression;
}
@Required
public void setTopicExpression(final String topicExpression) {
this.topicExpression = topicExpression;
}
@Override
public String getTopicPrefix() {
return topicPrefix;
}
public void setTopicPrefix(final String topicPrefix) {
this.topicPrefix = topicPrefix;
}
@Override
public boolean isEnabled() {
return enabled;
}
@Override
public void setEnabled(boolean enabled) {
this.enabled = enabled;
}
}

View File

@ -0,0 +1,25 @@
package eu.dnetlib.enabling.actions;
import eu.dnetlib.enabling.tools.blackboard.NotificationHandler;
/**
* Manager subscription action.
*
* @author marko
*
*/
public interface SubscriptionAction extends NotificationHandler {
/**
* topic expression associated with this subscription action.
*
* @return SN topic expression
*/
String getTopicExpression();
/**
* react on this topic prefix. Might be shorter than the topic expression.
*
* @return SN topic prefix
*/
String getTopicPrefix();
}

View File

@ -0,0 +1,191 @@
package eu.dnetlib.enabling.tools;
import java.text.ParseException;
import javax.annotation.PostConstruct;
import javax.annotation.Resource;
import org.quartz.*;
import org.quartz.Trigger.TriggerState;
import org.quartz.impl.JobDetailImpl;
import org.quartz.impl.triggers.CronTriggerImpl;
import org.springframework.beans.factory.BeanNameAware;
import org.springframework.beans.factory.annotation.Required;
/**
* Common implementation for schedulable beans.
*
* @author claudio
*
*/
public abstract class AbstractSchedulable implements Schedulable, Job, BeanNameAware {
private static final String THIS = "this";
private final static String GROUP = "schedulableJobs";
private boolean enabled;
private CronExpression cronExpression;
private String beanName;
@Resource(name = "dnetJobScheduler")
private Scheduler jobScheduler;
@PostConstruct
protected void init() {
try {
JobDataMap jobDataMap = new JobDataMap();
jobDataMap.put(THIS, this);
JobDetailImpl jd = new JobDetailImpl();
jd.setName(getBeanName());
jd.setGroup(GROUP);
jd.setJobDataMap(jobDataMap);
jd.setJobClass(this.getClass());
jobScheduler.scheduleJob(jd, createTrigger());
} catch (SchedulerException e) {
throw new RuntimeException(e);
}
}
private Trigger createTrigger() {
try {
CronTriggerImpl trigger = new CronTriggerImpl(getBeanName(), GROUP, getCronExpression());
trigger.setMisfireInstruction(Trigger.MISFIRE_INSTRUCTION_IGNORE_MISFIRE_POLICY);
trigger.setJobGroup(GROUP);
trigger.setJobName(getBeanName());
return trigger;
} catch (ParseException e) {
throw new IllegalArgumentException("invalid cron expression: " + cronExpression, e);
}
}
protected abstract void doExecute();
@Override
public void execute() {
// bean represents the quartz instance of this object
if (isEnabled()) {
doExecute();
}
}
@Override
public void execute(JobExecutionContext context) throws JobExecutionException {
AbstractSchedulable bean = (AbstractSchedulable) context.getJobDetail().getJobDataMap().get(THIS);
// bean represents the quartz instance of this object
if (bean.isEnabled()) {
bean.doExecute();
}
}
@Override
public boolean isEnabled() {
return enabled;
}
@Override
public void setEnabled(boolean enabled) {
this.enabled = enabled;
}
@Override
public String getCronExpression() {
return cronExpression.getCronExpression();
}
@Required
public void setCronExpression(String cronExpression) {
try {
this.cronExpression = new CronExpression(cronExpression);
} catch (ParseException e) {
throw new IllegalArgumentException("invalid cron expression: " + cronExpression, e);
}
}
@Override
public void updateCronExpression(String cronExpression) {
if (!cronExpression.equals(getCronExpression())) {
setCronExpression(cronExpression);
try {
final TriggerKey tk = new TriggerKey(getBeanName());
jobScheduler.rescheduleJob(tk, createTrigger());
} catch (SchedulerException e) {
throw new RuntimeException("unable to reschedule trigger", e);
}
}
}
@Override
public String getNextFireTime() {
try {
if (isPaused()) {
return "";
}
if (!isEnabled()) {
return "";
}
final TriggerKey tk = new TriggerKey(getBeanName());
final Trigger t = jobScheduler.getTrigger(tk);
return t != null ? t.getNextFireTime().toString() : "";
} catch (SchedulerException e) {
throw new RuntimeException("unable to get trigger", e);
}
}
@Override
public boolean isPaused() {
try {
final TriggerKey tk = new TriggerKey(getBeanName());
final TriggerState state = jobScheduler.getTriggerState(tk);
switch (state) {
case PAUSED:
case NONE:
case ERROR:
return true;
default:
return false;
}
} catch (SchedulerException e) {
throw new RuntimeException("unable to get trigger", e);
}
}
@Override
public void pause() {
try {
final TriggerKey tk = new TriggerKey(getBeanName());
jobScheduler.pauseTrigger(tk);
} catch (SchedulerException e) {
throw new RuntimeException("unable to pause trigger", e);
}
}
@Override
public void resume() {
try {
final TriggerKey tk = new TriggerKey(getBeanName());
jobScheduler.resumeTrigger(tk);
} catch (SchedulerException e) {
throw new RuntimeException("unable to resume trigger", e);
}
}
public String getBeanName() {
return beanName;
}
@Override
@Required
public void setBeanName(String beanName) {
this.beanName = beanName;
}
}

View File

@ -0,0 +1,25 @@
package eu.dnetlib.enabling.tools;
/**
* Simple interface for classes, which's objects can be enabled or disabled.
*
*
* @author marko
*
*/
public interface Enableable {
/**
* True if this object enabled.
*
* @return if this object is currently enabled.
*/
boolean isEnabled();
/**
* Enables of disables this object.
*
* @param enabled
* enabled
*/
void setEnabled(boolean enabled);
}

View File

@ -0,0 +1,57 @@
package eu.dnetlib.enabling.tools;
import java.util.Map;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.BeanFactoryAware;
import org.springframework.beans.factory.ListableBeanFactory;
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
/**
* This bean has to live in the same bean factory where all enableables live. Thus it cannot be put inside the EnableablController, which
* lives in the webContext.
*
* @author marko
*
*/
public class EnableableEnumerator implements BeanFactoryAware {
/**
* bean factory.
*/
private ListableBeanFactory beanFactory;
/**
* Get all beans implementing the Enableable interface.
*
* @return
*/
public Map<String, Enableable> getAllEnableables() {
return beanFactory.getBeansOfType(Enableable.class);
}
@Override
public void setBeanFactory(final BeanFactory beanFactory) throws BeansException {
this.beanFactory = (ListableBeanFactory) beanFactory;
}
public ListableBeanFactory getBeanFactory() {
return beanFactory;
}
/**
* Get given enableable or null.
*
* @param name
* @return
*/
public Enableable getEnableable(final String name) {
try {
return beanFactory.getBean(name, Enableable.class);
} catch (final NoSuchBeanDefinitionException e) {
return null;
}
}
}

View File

@ -0,0 +1,52 @@
package eu.dnetlib.enabling.tools;
/**
* Simple interface for classes, which's objects actions can be scheduled.
*
* @author claudio
*/
public interface Schedulable extends Enableable {
/**
* Updates the cron expression.
*
* @param cronExpression
*/
void updateCronExpression(String cronExpression);
/**
* Getter for the cron expression.
*
* @return the String representing the cron expression.
*/
String getCronExpression();
/**
* Computes the next time the doExecute method will be invoked.
*
* @return the String representing the next fire time.
*/
String getNextFireTime();
/**
* True if the object has been paused.
*
* @return if the bean has been paused.
*/
boolean isPaused();
/**
* Prevents the bean for being scheduled for future invocations.
*/
void pause();
/**
* Resumes the bean for being scheduled for future invocations.
*/
void resume();
/**
* Must be implemented to perform the real action.
*/
void execute();
}

View File

@ -0,0 +1,57 @@
package eu.dnetlib.enabling.tools;
import java.util.Map;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.BeanFactoryAware;
import org.springframework.beans.factory.ListableBeanFactory;
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
/**
* This bean has to live in the same bean factory where all schedulables live. Thus it cannot be put inside the EnableablController, which
* lives in the webContext.
*
* @author claudio
*
*/
public class SchedulableEnumerator implements BeanFactoryAware {
/**
* bean factory.
*/
private ListableBeanFactory beanFactory;
/**
* Get all beans implementing the Schedulable interface.
*
* @return
*/
public Map<String, Schedulable> getAllSchedulables() {
return beanFactory.getBeansOfType(Schedulable.class);
}
@Override
public void setBeanFactory(final BeanFactory beanFactory) throws BeansException {
this.beanFactory = (ListableBeanFactory) beanFactory;
}
public ListableBeanFactory getBeanFactory() {
return beanFactory;
}
/**
* Get given schedulable or null.
*
* @param name
* @return
*/
public Schedulable getSchedulable(final String name) {
try {
return beanFactory.getBean(name, Schedulable.class);
} catch (final NoSuchBeanDefinitionException e) {
return null;
}
}
}

View File

@ -0,0 +1,48 @@
package eu.dnetlib.enabling.tools;
import javax.annotation.Resource;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.factory.annotation.Required;
import eu.dnetlib.enabling.is.registry.rmi.ISRegistryException;
import eu.dnetlib.enabling.is.registry.rmi.ISRegistryService;
import eu.dnetlib.enabling.locators.UniqueServiceLocator;
/**
* Schedulable implementation of an xUpdate.
*
* @author claudio
*
*/
public class SchedulableProfileUpdater extends AbstractSchedulable {
private static final Log log = LogFactory.getLog(SchedulableProfileUpdater.class); // NOPMD by marko on 11/24/08 5:02 PM
private String xUpdate;
@Resource
private UniqueServiceLocator serviceLocator;
@Override
protected void doExecute() {
try {
log.info("triggering scheduled reindex: " + getxUpdate());
serviceLocator.getService(ISRegistryService.class).executeXUpdate(getxUpdate());
} catch (ISRegistryException e) {
throw new RuntimeException(e);
}
}
public String getxUpdate() {
return xUpdate;
}
@Required
public void setxUpdate(String xUpdate) {
this.xUpdate = xUpdate;
}
}

View File

@ -0,0 +1,62 @@
package eu.dnetlib.enabling.tools.blackboard;
/**
* Utility abstract class which dispatches to commodity onSomething() methods, one for each interesting
* blackboard job state.
*
* @author marko
*
*/
public abstract class AbstractBlackboardJobListener implements BlackboardJobListener {
/**
* {@inheritDoc}
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardJobListener#processJob(eu.dnetlib.enabling.tools.blackboard.BlackboardJob)
*/
@Override
public void processJob(final BlackboardJob job) {
if (job.getActionStatus() == ActionStatus.DONE)
onDone(job);
else if (job.getActionStatus() == ActionStatus.FAILED)
onFailed(job);
else if (job.getActionStatus() == ActionStatus.ASSIGNED)
onAssigned(job);
else if (job.getActionStatus() == ActionStatus.ONGOING)
onOngoing(job);
}
/**
* Called when the job enters the ASSIGNED state.
*
* @param job job
*/
protected void onAssigned(final BlackboardJob job) { // NOPMD
// default no operation
// TODO: increase job expiry time
}
/**
* Called when the job enters the ONGOING state.
*
* @param job job
*/
protected void onOngoing(final BlackboardJob job) { // NOPMD
// default no operation
// TODO: increase job expiry time
}
/**
* Called when the job finishes in the FAILED state.
*
* @param job job
*/
protected abstract void onFailed(BlackboardJob job);
/**
* Called when the job finishes in the DONE state.
*
* @param job job
*/
protected abstract void onDone(BlackboardJob job);
}

View File

@ -0,0 +1,96 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.io.IOException;
import java.util.concurrent.Executor;
import java.util.concurrent.Executors;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.xpath.XPathExpressionException;
import org.springframework.beans.factory.annotation.Required;
import org.xml.sax.SAXException;
import eu.dnetlib.enabling.tools.Enableable;
import eu.dnetlib.enabling.tools.StringOpaqueResource;
/**
* Common blackboard notification handler. This notification handler processes only message with ...BODY.BLACKBOARD.LAST* as topic.
*
* @param <T>
* type of blackboard handler used to extract the blackboard message (client or server)
* @author marko
*
*/
public abstract class AbstractBlackboardNotificationHandler<T extends BlackboardHandler> implements NotificationHandler, Enableable {
/**
* blackboard handler.
*/
private T blackboardHandler;
/**
* true if enabled.
*/
private boolean enabled = true;
/**
* Executor handles the notified request in a dedicated thread and allows to return immediately.
*/
private Executor executor = Executors.newCachedThreadPool();
/**
* {@inheritDoc}
*
* @see eu.dnetlib.data.mdstore.NotificationHandler#notified(java.lang.String, java.lang.String, java.lang.String, java.lang.String)
*/
@Override
public void notified(final String subscrId, final String topic, final String rsId, final String profile) {
if (!topic.contains("BODY.BLACKBOARD.LAST")) return;
executor.execute(new Runnable() {
@Override
public void run() {
try {
processJob(blackboardHandler.getJob(new StringOpaqueResource(profile)));
} catch (final XPathExpressionException e) {
throw new IllegalStateException(e);
} catch (final SAXException e) {
throw new IllegalStateException(e);
} catch (final IOException e) {
throw new IllegalStateException(e);
} catch (final ParserConfigurationException e) {
throw new IllegalStateException(e);
}
}
});
}
/**
* Subclassess override this to process incoming blackboard jobs.
*
* @param job
* blackboard job
*/
protected abstract void processJob(BlackboardJob job);
public T getBlackboardHandler() {
return blackboardHandler;
}
@Required
public void setBlackboardHandler(final T blackboardHandler) {
this.blackboardHandler = blackboardHandler;
}
@Override
public boolean isEnabled() {
return enabled;
}
@Override
public void setEnabled(final boolean enabled) {
this.enabled = enabled;
}
}

View File

@ -0,0 +1,26 @@
package eu.dnetlib.enabling.tools.blackboard;
/**
* Blackboard action status.
*
* @author marko
*
*/
public enum ActionStatus {
/**
* The job/action is assigned but not jet taken into started execution.
*/
ASSIGNED,
/**
* The job/action is ongoing.
*/
ONGOING,
/**
* The job/action is completed successfully.
*/
DONE,
/**
* The job/action is completed with failure.
*/
FAILED
}

View File

@ -0,0 +1,32 @@
package eu.dnetlib.enabling.tools.blackboard;
/**
* Helpers for the blackboard protocol client.
*
* @author marko
*
*/
public interface BlackboardClientHandler extends BlackboardHandler {
/**
* Create a new job.
*
* @param serviceId service identifier
* @return newly created blackboard job
*/
BlackboardJob newJob(String serviceId);
/**
* Assign a blackboard job to a service.
*
* @param job blackboard job to send
*/
void assign(BlackboardJob job);
/**
* The client can delete the job after it has reached a final state
* or the job timeout has expired.
*
* @param job blackboard job to delete.
*/
void delete(BlackboardJob job);
}

View File

@ -0,0 +1,155 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.util.Map;
import javax.xml.bind.JAXBException;
import javax.xml.transform.dom.DOMSource;
import javax.xml.xpath.XPath;
import javax.xml.xpath.XPathConstants;
import javax.xml.xpath.XPathExpressionException;
import javax.xml.xpath.XPathFactory;
import org.springframework.beans.factory.annotation.Required;
import org.w3c.dom.Element;
import eu.dnetlib.enabling.is.registry.rmi.ISRegistryException;
import eu.dnetlib.enabling.is.registry.rmi.ISRegistryService;
import eu.dnetlib.enabling.locators.UniqueServiceLocator;
import eu.dnetlib.enabling.tools.OpaqueResource;
import eu.dnetlib.enabling.tools.UniqueIdentifierGenerator;
import eu.dnetlib.enabling.tools.UniqueIdentifierGeneratorImpl;
import eu.dnetlib.miscutils.jaxb.JaxbFactory;
/**
* Blackboard client.
*
* @author marko
*
*/
public class BlackboardClientHandlerImpl implements BlackboardClientHandler {
/**
* blackboard message factory.
*/
private JaxbFactory<BlackboardMessage> messageFactory;
/**
* service locator.
*/
private UniqueServiceLocator serviceLocator;
/**
* generates blackboard message identifiers for new jobs.
*/
private UniqueIdentifierGenerator uuidGenerator = new UniqueIdentifierGeneratorImpl("bb-");
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardClientHandler#assign(eu.dnetlib.enabling.tools.blackboard.BlackboardJob)
*/
@Override
public void assign(final BlackboardJob job) {
checkJob(job);
try {
serviceLocator.getService(ISRegistryService.class).addBlackBoardMessage(job.getServiceId(), job.getId(),
messageFactory.serialize(job.getMessage()));
} catch (final ISRegistryException e) {
throw new IllegalStateException("cannot register blackboard message", e);
} catch (final JAXBException e) {
throw new IllegalArgumentException("cannot serialize blackboard message", e);
}
}
/**
* Check that the job has sane values.
*
* @param job
*/
protected void checkJob(final BlackboardJob job) {
for (Map.Entry<String, String> param : job.getParameters().entrySet()) {
if (param.getValue() == null) { throw new IllegalStateException("job parameter value cannot be null: " + param.getKey()); }
}
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardClientHandler#delete(eu.dnetlib.enabling.tools.blackboard.BlackboardJob)
*/
@Override
public void delete(final BlackboardJob job) {
try {
serviceLocator.getService(ISRegistryService.class).deleteBlackBoardMessage(job.getServiceId(), job.getId());
} catch (final ISRegistryException e) {
throw new IllegalStateException("cannot delete blackboard message", e);
}
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardClientHandler#newJob()
*/
@Override
public BlackboardJob newJob(final String serviceId) {
final BlackboardJob job = new BlackboardJobImpl(serviceId, messageFactory.newInstance());
job.setActionStatus(ActionStatus.ASSIGNED);
job.getParameters().put("id", "");
job.getParameters().put("error", "");
job.setId(uuidGenerator.generateIdentifier());
return job;
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardHandler#getJob(eu.dnetlib.enabling.tools.OpaqueResource)
*/
@Override
public BlackboardJob getJob(final OpaqueResource profile) {
final XPath xpa = XPathFactory.newInstance().newXPath();
try {
final Element source = (Element) xpa.evaluate("/RESOURCE_PROFILE/BODY/BLACKBOARD/MESSAGE[@id = /RESOURCE_PROFILE/BODY/BLACKBOARD/LAST_RESPONSE]",
profile.asDom(), XPathConstants.NODE);
if (source == null) { throw new IllegalStateException("cannot find last blackboard message in the service profile"); }
return new BlackboardJobImpl(profile.getResourceId(), messageFactory.parse(new DOMSource(source)));
} catch (final JAXBException e) {
throw new IllegalStateException("cannot parse blackboard message", e);
} catch (final XPathExpressionException e) {
throw new IllegalStateException("cannot find last blackboard message in the service profile", e);
}
}
public JaxbFactory<BlackboardMessage> getMessageFactory() {
return messageFactory;
}
public void setMessageFactory(final JaxbFactory<BlackboardMessage> messageFactory) {
this.messageFactory = messageFactory;
}
public UniqueIdentifierGenerator getUuidGenerator() {
return uuidGenerator;
}
public void setUuidGenerator(final UniqueIdentifierGenerator uuidGenerator) {
this.uuidGenerator = uuidGenerator;
}
public UniqueServiceLocator getServiceLocator() {
return serviceLocator;
}
@Required
public void setServiceLocator(final UniqueServiceLocator serviceLocator) {
this.serviceLocator = serviceLocator;
}
}

View File

@ -0,0 +1,20 @@
package eu.dnetlib.enabling.tools.blackboard;
import eu.dnetlib.enabling.tools.OpaqueResource;
/**
* Basic blackboard handler.
*
* @author marko
*
*/
public interface BlackboardHandler {
/**
* Get the current job from, as notified in the service profile.
*
* @param profile
* service profile
* @return notified blackboard job
*/
BlackboardJob getJob(OpaqueResource profile);
}

View File

@ -0,0 +1,135 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.io.PrintWriter;
import java.io.StringWriter;
import javax.xml.bind.JAXBException;
import javax.xml.transform.dom.DOMSource;
import javax.xml.xpath.XPath;
import javax.xml.xpath.XPathConstants;
import javax.xml.xpath.XPathExpressionException;
import javax.xml.xpath.XPathFactory;
import org.springframework.beans.factory.annotation.Required;
import org.w3c.dom.Element;
import eu.dnetlib.enabling.is.registry.rmi.ISRegistryException;
import eu.dnetlib.enabling.is.registry.rmi.ISRegistryService;
import eu.dnetlib.enabling.locators.UniqueServiceLocator;
import eu.dnetlib.enabling.tools.OpaqueResource;
import eu.dnetlib.miscutils.jaxb.JaxbFactory;
/**
* Blackboard handler implementation.
*
* @author marko
*
*/
public class BlackboardHandlerImpl implements BlackboardServerHandler {
/**
* blackboard message factory.
*/
private JaxbFactory<BlackboardMessage> messageFactory;
/**
* service locator.
*/
private UniqueServiceLocator serviceLocator;
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardHandler#getJob(eu.dnetlib.enabling.tools.OpaqueResource)
*/
@Override
public BlackboardJob getJob(final OpaqueResource profile) {
final XPath xpa = XPathFactory.newInstance().newXPath();
try {
final Element source = (Element) xpa.evaluate("/RESOURCE_PROFILE/BODY/BLACKBOARD/MESSAGE[@id = /RESOURCE_PROFILE/BODY/BLACKBOARD/LAST_REQUEST]",
profile.asDom(), XPathConstants.NODE);
if (source == null) { throw new IllegalStateException("cannot find last blackboard message in the service profile"); }
return new BlackboardJobImpl(profile.getResourceId(), messageFactory.parse(new DOMSource(source)));
} catch (final JAXBException e) {
throw new IllegalStateException("cannot parse blackboard message", e);
} catch (final XPathExpressionException e) {
throw new IllegalStateException("cannot find last blackboard message in the service profile", e);
}
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardHandler#done(eu.dnetlib.enabling.tools.blackboard.BlackboardJob)
*/
@Override
public void done(final BlackboardJob job) {
job.setActionStatus(ActionStatus.DONE);
replyJob(job);
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardHandler#failed(eu.dnetlib.enabling.tools.blackboard.BlackboardJob,
* java.lang.Exception)
*/
@Override
public void failed(final BlackboardJob job, final Throwable exception) {
job.setActionStatus(ActionStatus.FAILED);
final StringWriter stackTrace = new StringWriter();
exception.printStackTrace(new PrintWriter(stackTrace));
job.getParameters().put("error", exception.toString());
job.getParameters().put("errorDetails", stackTrace.toString());
replyJob(job);
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardHandler#ongoing(eu.dnetlib.enabling.tools.blackboard.BlackboardJob)
*/
@Override
public void ongoing(final BlackboardJob job) {
job.setActionStatus(ActionStatus.ONGOING);
replyJob(job);
}
/**
* Internal helper method which replies a blackboard job.
*
* @param job
* blackboard job
*/
protected void replyJob(final BlackboardJob job) {
try {
serviceLocator.getService(ISRegistryService.class).replyBlackBoardMessage(job.getServiceId(), messageFactory.serialize(job.getMessage()));
} catch (final ISRegistryException e) {
throw new IllegalStateException("cannot reply the blackboard message", e);
} catch (final JAXBException e) {
throw new IllegalArgumentException("cannot serialize blackboard message", e);
}
}
public JaxbFactory<BlackboardMessage> getMessageFactory() {
return messageFactory;
}
public void setMessageFactory(final JaxbFactory<BlackboardMessage> messageFactory) {
this.messageFactory = messageFactory;
}
public UniqueServiceLocator getServiceLocator() {
return serviceLocator;
}
@Required
public void setServiceLocator(final UniqueServiceLocator serviceLocator) {
this.serviceLocator = serviceLocator;
}
}

View File

@ -0,0 +1,118 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.util.Map;
/**
* High level representation of a blackboard job.
*
* @author marko
*
*/
public interface BlackboardJob {
/**
* Get the message id.
*
* @return id
*/
String getId();
/**
* Set message id.
*
* @param identifier
* id
*/
void setId(String identifier);
/**
* Get the action name.
*
* @return action name
*/
String getAction();
/**
* Set the action name.
*
* @param action
* action name
*/
void setAction(String action);
/**
* Get the action status.
*
* @return action status
*/
ActionStatus getActionStatus();
/**
* Set the action status.
*
* @param actionStatus
* action status
*/
void setActionStatus(ActionStatus actionStatus);
/**
* Get the message date.
*
* @return date
*/
String getDate();
/**
* Set the message date.
*
* @param date
* date
*/
void setDate(String date);
/**
* obtains a mutable parameter map (key/value).
*
* @return mutable parameter map
*/
Map<String, String> getParameters();
/**
* Get the error message, if the actionStatus is FAILED. Usually the error message is also present in the parameter map but it's
* preferred to use this method.
*
* @return error message
*/
String getError();
/**
* Set the error message, if the actionStatus is FAILED. Usually the error message is also present in the parameter map but it's
* preferred to use this method.
*
* @param error
* error message
*/
void setError(String error);
/**
* Get the setviceId.
*
* @return the serviceId
*/
String getServiceId();
/**
* True if the action status is one of the termination statuses (e.g. done and failed).
*
* @return true if completed
*/
boolean isCompleted();
/**
* Get the bb message.
*
* @return the bb message
*/
BlackboardMessage getMessage();
}

View File

@ -0,0 +1,163 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.util.HashMap;
import java.util.Map;
import java.util.Map.Entry;
/**
* High level representation of a blackboard job.
*
* @author marko
*
*/
public class BlackboardJobImpl implements BlackboardJob {
/**
* underlying low level blackboard message.
*/
private BlackboardMessage message;
/**
* service identifier.
*/
private String serviceId;
/**
* parameters.
*/
private final transient Map<String, String> parameters = new HashMap<String, String>();
/**
* Construct a new blackboard job from a blackboard message.
*
* @param serviceId
* service identifier
* @param message
* underlying low-level blackboard message
*/
public BlackboardJobImpl(final String serviceId, final BlackboardMessage message) {
super();
this.message = message;
this.serviceId = serviceId;
for (final BlackboardParameter param : message.getParameters()) {
parameters.put(param.getName(), param.getValue());
}
}
@Override
public String getId() {
return message.getId();
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardJob#setId(java.lang.String)
*/
@Override
public void setId(final String identifier) {
message.setId(identifier);
}
@Override
public String getAction() {
return message.getAction();
}
@Override
public ActionStatus getActionStatus() {
return message.getActionStatus();
}
@Override
public String getDate() {
return message.getDate();
}
@Override
public String getError() {
return getParameters().get("error");
}
@Override
public Map<String, String> getParameters() {
return parameters;
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardJob#setAction(java.lang.String)
*/
@Override
public void setAction(final String action) {
message.setAction(action);
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardJob#setActionStatus(eu.dnetlib.enabling.tools.blackboard.ActionStatus)
*/
@Override
public void setActionStatus(final ActionStatus actionStatus) {
message.setActionStatus(actionStatus);
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardJob#setDate(java.lang.String)
*/
@Override
public void setDate(final String date) {
message.setDate(date);
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardJob#setError(java.lang.String)
*/
@Override
public void setError(final String error) {
getParameters().put("error", error);
}
/**
* returns the blackboard message, potentially modified.
*
* @return underlying blackboard message
*/
@Override
public BlackboardMessage getMessage() {
message.getParameters().clear();
for (final Entry<String, String> entry : getParameters().entrySet()) {
final BlackboardParameterImpl param = new BlackboardParameterImpl(); // NOPMD
param.setName(entry.getKey());
param.setValue(entry.getValue());
message.getParameters().add(param);
}
return message;
}
public void setMessage(final BlackboardMessage message) {
this.message = message;
}
@Override
public String getServiceId() {
return serviceId;
}
public void setServiceId(final String serviceId) {
this.serviceId = serviceId;
}
@Override
public boolean isCompleted() {
return getActionStatus() == ActionStatus.DONE || getActionStatus() == ActionStatus.FAILED;
}
}

View File

@ -0,0 +1,18 @@
package eu.dnetlib.enabling.tools.blackboard;
/**
* Implement this interface in order to receive notifications for specific blackboard messages.
*
* @author marko
*
*/
public interface BlackboardJobListener {
/**
* process the given job.
*
* @param job job to process
*/
void processJob(BlackboardJob job);
}

View File

@ -0,0 +1,19 @@
package eu.dnetlib.enabling.tools.blackboard;
/**
* Registers BlackboardJobListeners to be called when related incoming blackboard job notifications arrive.
* @author marko
*
*/
public interface BlackboardJobRegistry {
/**
* Registers a new job listener. It will be notified when the job changes state. The listener will be unregistered
* when the job arrives to a final state (DONE, or FAILED).
*
* @param job
* job
* @param listener
* job listener
*/
void registerJobListener(BlackboardJob job, BlackboardJobListener listener);
}

View File

@ -0,0 +1,82 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.util.List;
/**
* Implementor of this interface represent blackboard messages nodes as the appear in the service profile blackboard.
*
* This is the low-level representation of the blackboard message. The BlackboardJob is a higher level representation of a blackboard message,
* and a BlackboardHandler helps converting between the two.
*
* Implementors of this interface should simply provide a serialization so that the message can be stored in the service profile.
*
* @author marko
*
*/
public interface BlackboardMessage {
/**
* blackboard message date.
*
* @return date
*/
String getDate();
/**
* sets the blackboard message date/timestamp.
*
* @param date
* date
*/
void setDate(String date);
/**
* Blackboard message identifier.
*
* @return identifier
*/
String getId();
/**
* sets the blackboard message identifier.
*
* @param id identifier
*/
void setId(String id); // NOPMD
/**
* Blackboard message action name.
*
* @return action
*/
String getAction();
/**
* Sets the blackboard message action name.
*
* @param action
* action
*/
void setAction(String action);
/**
* Blackboard message parameter list.
*
* @return parameter list
*/
List<BlackboardParameter> getParameters();
/**
* Status of the action associated with this message.
*
* @return action status
*/
ActionStatus getActionStatus();
/**
* sets the action status.
*
* @param actionStatus
* action status
*/
void setActionStatus(ActionStatus actionStatus);
}

View File

@ -0,0 +1,142 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.util.ArrayList;
import java.util.List;
import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;
import org.apache.commons.lang3.builder.EqualsBuilder;
import org.apache.commons.lang3.builder.HashCodeBuilder;
/**
* standard serialization of the blackboard message.
*
* @author marko
*
*/
@XmlRootElement(namespace = "", name = "MESSAGE")
@XmlAccessorType(XmlAccessType.NONE)
public class BlackboardMessageImpl implements BlackboardMessage {
/**
* hash seed.
*/
private static final int HASH_SEED_2 = 63;
/**
* hash seed.
*/
private static final int HASH_SEED = 13;
/**
* blackboard message timestamp.
*/
@XmlAttribute
private String date;
/**
* blackboard message identifier.
*/
@XmlAttribute
private String id; // NOPMD
/**
* blackboard message action name.
*/
@XmlElement(name = "ACTION", required = true)
private String action;
/**
* blackboard message parameters (key/value).
*/
@XmlElement(name = "PARAMETER", type = BlackboardParameterImpl.class)
private List<BlackboardParameter> parameters = new ArrayList<BlackboardParameter>();
/**
* the status of the action described by this message.
*/
@XmlElement(name = "ACTION_STATUS", required = true)
private ActionStatus actionStatus;
/**
* {@inheritDoc}
*
* @see java.lang.Object#equals(java.lang.Object)
*/
@Override
public boolean equals(final Object obj) {
if (!(obj instanceof BlackboardMessage))
return false;
if (this == obj)
return true;
final BlackboardMessage rhs = (BlackboardMessage) obj;
return new EqualsBuilder().append(id, rhs.getId()).append(date, rhs.getDate()).append(action, rhs.getAction()).append(actionStatus,
rhs.getActionStatus()).append(parameters, rhs.getParameters()).isEquals();
}
/**
* {@inheritDoc}
*
* @see java.lang.Object#hashCode()
*/
@Override
public int hashCode() {
return new HashCodeBuilder(HASH_SEED, HASH_SEED_2).append(id).append(date).append(action).append(actionStatus).append(parameters).toHashCode();
}
@Override
public String getDate() {
return date;
}
@Override
public void setDate(final String date) {
this.date = date;
}
@Override
public String getId() {
return id;
}
@Override
public void setId(final String id) { // NOPMD
this.id = id;
}
@Override
public String getAction() {
return action;
}
@Override
public void setAction(final String action) {
this.action = action;
}
@Override
public List<BlackboardParameter> getParameters() {
return parameters;
}
public void setParameters(final List<BlackboardParameter> parameters) {
this.parameters = parameters;
}
@Override
public ActionStatus getActionStatus() {
return actionStatus;
}
@Override
public void setActionStatus(final ActionStatus actionStatus) {
this.actionStatus = actionStatus;
}
}

View File

@ -0,0 +1,59 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.util.HashMap;
import java.util.Map;
/**
* This notification handler dispatches incoming notifications to registered blackboard message listeners. Usually used
* by blackboard notification clients, since they are usually interested only in the messages they have sent and not in
* messages sent by other BB clients. Moreover it's an useful hook for dispatching orchestration.
*
* @param <T>
* blackboard handler (client or server)
* @author marko
*
*/
public class BlackboardNotificationHandler<T extends BlackboardHandler> extends AbstractBlackboardNotificationHandler<T> implements BlackboardJobRegistry {
/**
* blackboard message listeners.
*/
private Map<String, BlackboardJobListener> listeners = new HashMap<String, BlackboardJobListener>();
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.AbstractBlackboardNotificationHandler#processJob(eu.dnetlib.enabling.tools.blackboard.BlackboardJob)
*/
@Override
protected void processJob(final BlackboardJob job) {
final BlackboardJobListener listener = listeners.get(job.getId());
if (listener != null) {
if (job.isCompleted()) {
listeners.remove(job.getId());
}
listener.processJob(job);
}
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardJobRegistry#registerJobListener(eu.dnetlib.enabling.tools.blackboard.BlackboardJob,
* eu.dnetlib.enabling.tools.blackboard.BlackboardJobListener)
*/
@Override
public void registerJobListener(final BlackboardJob job, final BlackboardJobListener listener) {
listeners.put(job.getId(), listener);
}
public Map<String, BlackboardJobListener> getListeners() {
return listeners;
}
public void setListeners(final Map<String, BlackboardJobListener> listeners) {
this.listeners = listeners;
}
}

View File

@ -0,0 +1,36 @@
package eu.dnetlib.enabling.tools.blackboard;
/**
* Blackboard parameter used in a BlackboardMessage.
*
* @author marko
*
*/
public interface BlackboardParameter {
/**
* parameter name (key).
* @return name
*/
String getName();
/**
* setter.
*
* @param name name
*/
void setName(String name);
/**
* parameter value.
* @return value
*/
String getValue();
/**
* setter.
*
* @param value value
*/
void setValue(String value);
}

View File

@ -0,0 +1,87 @@
package eu.dnetlib.enabling.tools.blackboard;
import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.XmlRootElement;
import org.apache.commons.lang3.builder.EqualsBuilder;
import org.apache.commons.lang3.builder.HashCodeBuilder;
/**
* Blackboard parameter.
*
* @author marko
*
*/
@XmlRootElement(name = "PARAMETER")
public class BlackboardParameterImpl implements BlackboardParameter {
/**
* hash seed.
*/
private static final int HASH_SEED_2 = 59;
/**
* hash seed.
*/
private static final int HASH_SEED = 35;
/**
* parameter name.
*/
private String name;
/**
* parameter value.
*/
private String value;
/**
* {@inheritDoc}
*
* @see java.lang.Object#equals(java.lang.Object)
*/
@Override
public boolean equals(final Object obj) {
if (!(obj instanceof BlackboardParameter))
return false;
if (this == obj)
return true;
final BlackboardParameter rhs = (BlackboardParameter) obj;
return new EqualsBuilder().append(name, rhs.getName()).append(value, rhs.getValue()).isEquals();
}
/**
* {@inheritDoc}
*
* @see java.lang.Object#hashCode()
*/
@Override
public int hashCode() {
return new HashCodeBuilder(HASH_SEED, HASH_SEED_2).append(name).append(value).toHashCode();
}
@Override
@XmlAttribute
public String getName() {
return name;
}
@Override
public void setName(final String name) {
this.name = name;
}
@Override
@XmlAttribute
public String getValue() {
return value;
}
@Override
public void setValue(final String value) {
this.value = value;
}
}

View File

@ -0,0 +1,24 @@
package eu.dnetlib.enabling.tools.blackboard;
import eu.dnetlib.enabling.tools.blackboard.BlackboardJob;
import eu.dnetlib.enabling.tools.blackboard.BlackboardServerHandler;
/**
* Callback interface for serverside blackboard actions.
*
* <p>
* Actions should use the 'handler' methods to set the 'ongoing' or 'done' status.
* </p>
*
* <p>
* failed status is automatically set upon catching an exception thrown by this interface, so feel free to simply throw
* whatever you want
* </p>
*
* @author marko
*
* @param <X>
*/
public interface BlackboardServerAction<X extends Enum<?>> {
void execute(BlackboardServerHandler handler, BlackboardJob job) throws Exception;
}

View File

@ -0,0 +1,119 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.util.Map;
import javax.annotation.PostConstruct;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.factory.annotation.Required;
/**
* A BlackboardServerActionExecutor dispatches the execution of server side blackboard actions.
*
* <p>
* The primary responsibility of this class is to organize the actions, dispatch the action when called, and transform
* exceptions into blackboard 'failures'.
* </p>
*
* <p>
* The action executor is synchronous. It's responsibility of the actual action implementations to put the jobs in
* background, if needed.
* </p>
*
* @author marko
*
* @param <X>
*/
public class BlackboardServerActionExecutor<X extends Enum<X>> {
private static final Log log = LogFactory.getLog(BlackboardServerActionExecutor.class); // NOPMD by marko on 11/24/08 5:02 PM
/**
* set by spring, so key has to be string
*/
private Map<String, BlackboardServerAction<X>> actionMap;
/**
* used to set bb msg status as done, failed etc.
*/
private BlackboardServerHandler blackboardHandler;
/**
* used to exploit static enum declaration for runtime checks of allowed types.
*/
private Class<X> actionType;
/**
* if true, the executor will allow for an incomplete implementation of actions. If false, it an exception will be
* thrown if not all actions are implemented, according to the actionType enum.
*/
private boolean incomplete = false;
@PostConstruct
public void check() {
for (String key : actionMap.keySet())
Enum.valueOf(actionType, key);
if (!incomplete)
for (X en : actionType.getEnumConstants())
if (actionMap.get(en.toString()) == null)
throw new IllegalArgumentException("action " + en + " not declared in action map. Action is mandatory");
}
/**
* Executes a blackboard job, watching for
*
* @param job
*/
public void execute(final BlackboardJob job) {
try {
final X label = Enum.valueOf(actionType, job.getAction());
final BlackboardServerAction<X> action = actionMap.get(label.toString());
if (action != null) {
action.execute(blackboardHandler, job);
} else {
log.warn("Cannot find action handler for blackboard action: " + label);
throw new IllegalArgumentException("Cannot find action handler for blackboard action: " + label);
}
} catch (final Throwable e) {
log.warn("got some exception during blackboard handler execution", e);
blackboardHandler.failed(job, e);
}
}
public BlackboardServerHandler getBlackboardHandler() {
return blackboardHandler;
}
@Required
public void setBlackboardHandler(final BlackboardServerHandler blackboardHandler) {
this.blackboardHandler = blackboardHandler;
}
public Class<X> getActionType() {
return actionType;
}
@Required
public void setActionType(final Class<X> actionType) {
this.actionType = actionType;
}
public Map<String, BlackboardServerAction<X>> getActionMap() {
return actionMap;
}
public void setActionMap(Map<String, BlackboardServerAction<X>> actionMap) {
this.actionMap = actionMap;
}
public boolean isIncomplete() {
return incomplete;
}
public void setIncomplete(boolean incomplete) {
this.incomplete = incomplete;
}
}

View File

@ -0,0 +1,52 @@
package eu.dnetlib.enabling.tools.blackboard;
import javax.annotation.PostConstruct;
/**
* In most cases a service just dispatches every incoming blackboard execution request to the blackboard server
* executor, so that the right action call back will be called.
*
* <p>
* This bean will wrap a BlackboardServerExecutor as a BlackboardNotificationHandler so that it can be installed as a
* notification handler of a server.</p>
*
* @author marko
*
* @param <T>
*/
public class BlackboardServerExecutorNotificationHandler<T extends Enum<T>> extends AbstractBlackboardNotificationHandler<BlackboardServerHandler> {
private BlackboardServerActionExecutor<T> blackboardExecutor;
@PostConstruct
public void init() {
setBlackboardHandler(blackboardExecutor.getBlackboardHandler());
}
@Override
protected void processJob(final BlackboardJob job) {
blackboardExecutor.execute(job);
}
public BlackboardServerActionExecutor<T> getBlackboardExecutor() {
return blackboardExecutor;
}
public void setBlackboardExecutor(BlackboardServerActionExecutor<T> blackboardExecutor) {
this.blackboardExecutor = blackboardExecutor;
}
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.AbstractBlackboardNotificationHandler#setBlackboardHandler(eu.dnetlib.enabling.tools.blackboard.BlackboardHandler)
*
* Redefined here to avoid inheriting the @Required property. It simplifies the spring config for an
* unnecessarry property, since we can get it from the blackboardExecutor.
*/
@Override
public void setBlackboardHandler(BlackboardServerHandler handler) {
super.setBlackboardHandler(handler);
}
}

View File

@ -0,0 +1,37 @@
package eu.dnetlib.enabling.tools.blackboard;
/**
* Helper component used to bridge the high level blackboard job handling from the low level blackboard protocol based
* on notifications and blackboard messages.
*
* @author marko
*
*/
public interface BlackboardServerHandler extends BlackboardHandler {
/**
* Sets the ongoing action status to the given job, publishing this new state.
*
* @param job
* blackboard job
*/
void ongoing(BlackboardJob job);
/**
* Sets the "failed" action status to the given job, publishing this new state along with the error message obtained
* from the exception.
*
* @param job
* blackboard job
* @param exception
* exception which caused the failure
*/
void failed(BlackboardJob job, Throwable exception);
/**
* Set the "done" action status to the given job, publishing the new state.
*
* @param job
* blackboard job
*/
void done(BlackboardJob job);
}

View File

@ -0,0 +1,70 @@
package eu.dnetlib.enabling.tools.blackboard;
import javax.annotation.Resource;
import javax.xml.bind.JAXBException;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import eu.dnetlib.miscutils.jaxb.JaxbFactory;
/**
* This BB handler deletes completed (successful or unsuccessful) blackboard messages after dispatching. Job should be registered to only
* one "deleting BB handler".
*
* @author marko
*
*/
public class DeletingBlackboardNotificationHandler extends BlackboardNotificationHandler<BlackboardClientHandler> {
/**
* Logger.
*/
private static final Log log = LogFactory.getLog(DeletingBlackboardNotificationHandler.class); // NOPMD by marko on 11/24/08 5:02 PM
/**
* blackboard message factory.
*/
@Resource(name = "blackboardMessageFactory")
private JaxbFactory<BlackboardMessage> messageFactory;
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.BlackboardNotificationHandler#processJob(eu.dnetlib.enabling.tools.blackboard.BlackboardJob)
*/
@Override
protected void processJob(final BlackboardJob job) {
if (getListeners().containsKey(job.getId()) && job.isCompleted()) {
if (log.isDebugEnabled()) {
log.debug(serializeBlackBoardMessage(job));
}
getBlackboardHandler().delete(job);
}
super.processJob(job);
}
/**
* Helper method, serializes a Blackboard message using a blackboardMessageFactory.
*
* @param job
* @return
*/
private String serializeBlackBoardMessage(final BlackboardJob job) {
try {
return getMessageFactory().serialize(job.getMessage());
} catch (JAXBException e) {
return "cannot serialize blackboard message: " + e.getMessage();
}
}
public void setMessageFactory(final JaxbFactory<BlackboardMessage> messageFactory) {
this.messageFactory = messageFactory;
}
public JaxbFactory<BlackboardMessage> getMessageFactory() {
return messageFactory;
}
}

View File

@ -0,0 +1,22 @@
package eu.dnetlib.enabling.tools.blackboard;
/**
* Each service may have a chain of NotificationHandlers, which process incoming notifications.
*
* @author marko
*
*/
public interface NotificationHandler {
/**
* Incoming notification received.
*
* @param subscrId subscriptionId
* @param topic topic
* @param rsId resource id
* @param profile resource profile
*/
void notified(String subscrId, final String topic, final String rsId, final String profile);
}

View File

@ -0,0 +1,14 @@
package eu.dnetlib.enabling.tools.blackboard;
/**
* Invokes a chain of notification handlers for each incoming notification. Normally an instance of this class is
* registered as the notification handler for a given service.
*
* @author marko
*
*/
public interface NotificationHandlerChain extends NotificationHandler {
public void delegateNotification(final String subscrId, final String topic, final String rsId, final String profile, final NotificationHandler handler);
}

View File

@ -0,0 +1,104 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.util.Collection;
import java.util.Map;
import javax.annotation.Resource;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.factory.annotation.Required;
import org.springframework.core.task.TaskExecutor;
import eu.dnetlib.enabling.actions.AbstractSubscriptionAction;
import eu.dnetlib.enabling.tools.Enableable;
import eu.dnetlib.enabling.tools.EnableableEnumerator;
public class NotificationHandlerChainImpl implements NotificationHandlerChain {
/**
* logger.
*/
private static final Log log = LogFactory.getLog(NotificationHandlerChain.class); // NOPMD by marko on 11/24/08 5:02 PM
/**
* notification handler chain.
*/
private Collection<NotificationHandler> handlers;
/**
* task executor used for invoking the handlers.
*/
private TaskExecutor handlerExecutor;
@Resource
private NotificationHistory notificationHistory;
@Resource
private EnableableEnumerator enableableEnumerator;
/**
* {@inheritDoc}
*
* @see eu.dnetlib.enabling.tools.blackboard.NotificationHandler#notified(java.lang.String, java.lang.String,
* java.lang.String, java.lang.String)
*/
@Override
public void notified(final String subscrId, final String topic, final String rsId, final String profile) {
for (final NotificationHandler handler : handlers) {
try {
if (handler instanceof AbstractSubscriptionAction)
if (topic.startsWith(((AbstractSubscriptionAction) handler).getTopicPrefix()))
if (!((Enableable) handler).isEnabled()) {
for (Map.Entry<String, Enableable> entry : enableableEnumerator.getAllEnableables().entrySet()) {
if (entry.getValue() == handler) {
NotificationInfo info = new NotificationInfo();
info.setName(entry.getKey());
info.setProfile(profile);
info.setRsId(rsId);
info.setSubscrId(subscrId);
info.setTopic(topic);
notificationHistory.saveNotification(info);
}
}
continue;
}
delegateNotification(subscrId, topic, rsId, profile, handler);
} catch (final RuntimeException e) {
log.fatal("error processing notification handler " + handler, e);
}
}
}
@Override
public void delegateNotification(final String subscrId, final String topic, final String rsId, final String profile, final NotificationHandler handler) {
handlerExecutor.execute(new Runnable() {
@Override
public void run() {
handler.notified(subscrId, topic, rsId, profile);
}
});
}
public Collection<NotificationHandler> getHandlers() {
return handlers;
}
@Required
public void setHandlers(final Collection<NotificationHandler> handlers) {
this.handlers = handlers;
}
public TaskExecutor getHandlerExecutor() {
return handlerExecutor;
}
@Required
public void setHandlerExecutor(TaskExecutor handlerExecutor) {
this.handlerExecutor = handlerExecutor;
}
}

View File

@ -0,0 +1,37 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.util.Map;
public interface NotificationHistory {
/**
* Return a saved notification associated to a bean and to a rsId.
*
* @param name
* @return
*/
public NotificationInfo obtainNotification(String bean, String rsId);
/**
* Save a notification.
*
* @param info
*/
void saveNotification(NotificationInfo info);
/**
* Return all saved notifications.
*
* @return
*/
Map<String, Map<String, NotificationInfo>> obtainAllNotifications();
/**
* Remove a saved notifications.
*
* @return
*/
public void clearNotification(String bean, String rsId);
}

View File

@ -0,0 +1,39 @@
package eu.dnetlib.enabling.tools.blackboard;
import java.util.HashMap;
import java.util.Map;
public class NotificationHistoryImpl implements NotificationHistory {
private Map<String, Map<String, NotificationInfo>> notifications = new HashMap<String, Map<String, NotificationInfo>>();
@Override
public NotificationInfo obtainNotification(String bean, String rsId) {
if (!notifications.containsKey(bean))
return null;
return notifications.get(bean).get(rsId);
}
@Override
public void saveNotification(NotificationInfo info) {
String bean = info.getName();
String rsId = info.getRsId();
if (!notifications.containsKey(bean))
notifications.put(bean, new HashMap<String, NotificationInfo>());
notifications.get(bean).put(rsId, info);
}
@Override
public Map<String, Map<String, NotificationInfo>> obtainAllNotifications() {
return notifications;
}
@Override
public void clearNotification(String bean, String rsId) {
if (notifications.containsKey(bean))
notifications.get(bean).remove(rsId);
}
}

View File

@ -0,0 +1,74 @@
package eu.dnetlib.enabling.tools.blackboard;
import eu.dnetlib.miscutils.datetime.DateUtils;
public class NotificationInfo {
private String name;
private String subscrId;
private String topic;
private String rsId;
private String profile;
private String date;
public NotificationInfo() {
this.date = DateUtils.now_ISO8601();
}
public NotificationInfo(String name, String subscrId, String topic, String rsId, String profile) {
super();
this.name = name;
this.subscrId = subscrId;
this.topic = topic;
this.rsId = rsId;
this.profile = profile;
this.date = DateUtils.now_ISO8601();
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getSubscrId() {
return subscrId;
}
public void setSubscrId(String subscrId) {
this.subscrId = subscrId;
}
public String getTopic() {
return topic;
}
public void setTopic(String topic) {
this.topic = topic;
}
public String getRsId() {
return rsId;
}
public void setRsId(String rsId) {
this.rsId = rsId;
}
public String getProfile() {
return profile;
}
public void setProfile(String profile) {
this.profile = profile;
}
public String getDate() {
return date;
}
public void setDate(String date) {
this.date = date;
}
}

View File

@ -0,0 +1,12 @@
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">
<bean id="enableableEnumerator"
class="eu.dnetlib.enabling.tools.EnableableEnumerator" />
<bean id="notificationHistory"
class="eu.dnetlib.enabling.tools.blackboard.NotificationHistoryImpl" />
</beans>

View File

@ -0,0 +1,28 @@
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:jaxws="http://cxf.apache.org/jaxws"
xmlns:sec="http://cxf.apache.org/configuration/security" xmlns:wsa="http://cxf.apache.org/ws/addressing"
xmlns:p="http://www.springframework.org/schema/p" xmlns:http="http://cxf.apache.org/transports/http/configuration"
xmlns:t="http://dnetlib.eu/springbeans/t" xmlns:template="http://dnetlib.eu/springbeans/template"
xmlns:util="http://www.springframework.org/schema/util"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://cxf.apache.org/ws/addressing http://cxf.apache.org/schemas/ws-addr-conf.xsd
http://cxf.apache.org/configuration/security http://cxf.apache.org/schemas/configuration/security.xsd
http://cxf.apache.org/transports/http/configuration http://cxf.apache.org/schemas/configuration/http-conf.xsd
http://cxf.apache.org/jaxws http://cxf.apache.org/schemas/jaxws.xsd
http://dnetlib.eu/springbeans/template http://dnetlib.eu/springbeans/template.xsd
http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-2.0.xsd">
<bean id="dnetJobScheduler"
class="org.springframework.scheduling.quartz.SchedulerFactoryBean"
destroy-method="destroy">
<property name="jobFactory">
<bean id="jobSchedulerFactory"
class="org.springframework.scheduling.quartz.SpringBeanJobFactory" />
</property>
</bean>
<bean id="schedulableEnumerator"
class="eu.dnetlib.enabling.tools.SchedulableEnumerator" />
</beans>

View File

@ -0,0 +1,18 @@
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="blackboardMessageFactory" class="eu.dnetlib.miscutils.jaxb.JaxbFactory">
<constructor-arg
value="eu.dnetlib.enabling.tools.blackboard.BlackboardMessageImpl" />
</bean>
<bean id="blackboardHandler"
class="eu.dnetlib.enabling.tools.blackboard.BlackboardHandlerImpl"
p:messageFactory-ref="blackboardMessageFactory" p:serviceLocator-ref="uniqueServiceLocator" />
<bean id="blackboardClientHandler"
class="eu.dnetlib.enabling.tools.blackboard.BlackboardClientHandlerImpl"
p:messageFactory-ref="blackboardMessageFactory" p:serviceLocator-ref="uniqueServiceLocator" />
</beans>

View File

@ -0,0 +1,104 @@
package eu.dnetlib.enabling.tools.blackboard;
import javax.annotation.Resource;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
@RunWith(value = SpringJUnit4ClassRunner.class)
@ContextConfiguration
public class NotificationHandlerChainTest {
public class RecJob implements Runnable {
private int times;
private int num;
public RecJob(int num, int times) {
this.num = num;
this.times = times;
}
@Override
public void run() {
System.out.println("starting " + num);
if(times >= 0)
executor.execute(new RecJob(num + 1, times - 1));
try {
Thread.sleep(4000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println("thread finished - " + num);
}
}
private static final class Job implements Runnable {
private final int value;
public Job(final int i) {
value = i;
}
@Override
public void run() {
System.out.println("thread started - " + value);
try {
Thread.sleep(4000);
} catch (final InterruptedException e) {
//
}
System.out.println("thread finished - " + value);
}
}
@Resource
private transient ThreadPoolTaskExecutor executor;
@Test
public void testDelegateNotification() throws InterruptedException {
System.out.println(executor);
System.out.println("executing");
for (int i = 0; i < 100; i++)
executor.execute(new Job(i));
System.out.println("executed - waiting");
Thread.sleep(2000);
System.out.println("active count: " + executor.getActiveCount());
System.out.println("current pool size: " + executor.getCorePoolSize());
System.out.println("pool size " + executor.getPoolSize());
Thread.sleep(3000);
System.out.println("ok");
}
@Test
public void testRecursive() throws InterruptedException {
for (int i = 0; i < 4; i++)
executor.execute(new RecJob(i * 10, 4));
System.out.println("executed - waiting");
Thread.sleep(2000);
System.out.println("active count: " + executor.getActiveCount());
System.out.println("current pool size: " + executor.getCorePoolSize());
System.out.println("pool size " + executor.getPoolSize());
Thread.sleep(3000);
System.out.println("ok");
}
}

View File

@ -83,7 +83,7 @@ public class IterableXmlParserTest {
parser = new IterableXmlParser(element, stream);
int count = 0;
for (String xml : parser) {
System.out.println(xml);
//System.out.println(xml);
Document doc = reader.read(new StringReader(xml));
assertNotNull(doc);
assertNotNull(doc.selectSingleNode("//" + element));

View File

@ -0,0 +1,16 @@
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:lang="http://www.springframework.org/schema/lang"
xmlns:p="http://www.springframework.org/schema/p"
xmlns:util="http://www.springframework.org/schema/util"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/lang http://www.springframework.org/schema/lang/spring-lang-2.5.xsd
http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-2.5.xsd">
<bean id="msroNotificationExecutor"
class="org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor"
p:corePoolSize="32" p:keepAliveSeconds="3600" p:queueCapacity="0" />
</beans>

View File

@ -269,6 +269,11 @@
<artifactId>spring-tx</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context-support</artifactId>
<version>${spring.version}</version>
</dependency>
<!-- CXF -->