/*
* Copyright 2010 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
*
* 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 com.joshlong.activiti.coordinator;
import com.joshlong.activiti.coordinator.registry.ActivitiStateHandlerRegistration;
import com.joshlong.activiti.coordinator.registry.ActivitiStateHandlerRegistry;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.integration.Message;
import org.springframework.integration.MessageHeaders;
import org.springframework.integration.MessagingException;
import org.springframework.integration.support.MessageBuilder;
import java.lang.reflect.Method;
import java.util.*;
/**
* handles coordinating inbound messages that have come from the {@link CoordinatorGatewayProducer}
*
* @author Josh Long
* @since 1.0
*/
public class CoordinatorGatewayClient implements ApplicationContextAware {
private volatile ActivitiStateHandlerRegistry registry;
private ApplicationContext applicationContext;
public void setRegistry(ActivitiStateHandlerRegistry registry) {
this.registry = registry;
}
public Message<?> processMessage(Message<?> message) throws MessagingException {
try {
MessageHeaders headers = message.getHeaders();
String procName = (String) headers.get(CoordinatorConstants.PROCESS_NAME);
String stateName = (String) headers.get(CoordinatorConstants.STATE_NAME);
ActivitiStateHandlerRegistration registration = registry.findRegistrationForProcessAndState( procName, stateName);
Object bean = applicationContext.getBean(registration.getBeanName());
Method method = registration.getHandlerMethod();
// int size = method.getParameterTypes().length;
ArrayList<Object> argsList = new ArrayList<Object>();
Map<Integer, String> processVariablesMap = registration.getProcessVariablesExpected();
// already has which indexes get which process variables
Map<Integer, Object> variables = new HashMap<Integer, Object>();
for (Integer i : processVariablesMap.keySet())
variables.put(i, headers.get(processVariablesMap.get(i)));
if (registration.requiresProcessId()) {
variables.put(registration.getProcessIdIndex(),
headers.get(CoordinatorConstants.PROC_ID));
}
// System.out.println(variables.toString());
List<Integer> indices = new ArrayList<Integer>(variables.keySet());
Collections.sort(indices);
argsList.clear();
for (Integer idx : indices)
argsList.add(variables.get(idx));
Object[] args = argsList.toArray(new Object[argsList.size()]);
Object result = (args.length == 0) ? method.invoke(bean)
: method.invoke(bean, args);
MessageBuilder builder = MessageBuilder.withPayload(message.getPayload())
.copyHeaders(message.getHeaders());
if (result instanceof Map) {
// then we update the BPM
// by including them in the reply message
builder.copyHeadersIfAbsent( (Map)result);
}
return builder.build();
} catch (Throwable e) {
throw new RuntimeException(e);
}
}
public void setApplicationContext(ApplicationContext applicationContext)
throws BeansException {
this.applicationContext = applicationContext;
}
}