/**
* Copyright 2005 JBoss Inc
*
* 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 org.drools.workflow.core.node;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import org.drools.definition.process.Connection;
import org.drools.workflow.core.Constraint;
import org.drools.workflow.core.impl.ConnectionRef;
import org.drools.workflow.core.impl.NodeImpl;
/**
* Default implementation of a split node.
*
* @author <a href="mailto:kris_verlaenen@hotmail.com">Kris Verlaenen</a>
*/
public class Split extends NodeImpl implements Constrainable {
public static final int TYPE_UNDEFINED = 0;
/**
* All outgoing connections of a split of this type are triggered
* when its incoming connection has been triggered. A split of this
* type should have no constraints linked to any of its outgoing
* connections.
*/
public static final int TYPE_AND = 1;
/**
* Exactly one outgoing connection of a split of this type is triggered
* when its incoming connection has been triggered. Which connection
* is based on the constraints associated with each of the connections:
* the connection with the highest priority whose constraint is satisfied
* is triggered.
*/
public static final int TYPE_XOR = 2;
/**
* One or multiple outgoing connections of a split of this type are
* triggered when its incoming connection has been triggered. Which
* connections is based on the constraints associated with each of the
* connections: all connections whose constraint is satisfied are
* triggered.
*/
public static final int TYPE_OR = 3;
public static final int TYPE_XAND = 4;
private static final long serialVersionUID = 510l;
private int type;
private Map<ConnectionRef, Constraint> constraints = new HashMap<ConnectionRef, Constraint>();
public Split() {
this.type = TYPE_UNDEFINED;
}
public Split(final int type) {
this.type = type;
}
public void setType(final int type) {
this.type = type;
}
public int getType() {
return this.type;
}
public Constraint getConstraint(final Connection connection) {
if ( connection == null ) {
throw new IllegalArgumentException( "connection is null" );
}
if ( this.type == TYPE_OR || this.type == TYPE_XOR ) {
ConnectionRef ref = new ConnectionRef(connection.getTo().getId(), connection.getToType());
return this.constraints.get(ref);
}
throw new UnsupportedOperationException( "Constraints are " +
"only supported with XOR or OR split types, not with: " + getType() );
}
public Constraint internalGetConstraint(final ConnectionRef ref) {
return this.constraints.get(ref);
}
public void setConstraint(final Connection connection,
final Constraint constraint) {
if ( this.type == TYPE_OR || this.type == TYPE_XOR ) {
if ( connection == null ) {
throw new IllegalArgumentException( "connection is null" );
}
if (!getDefaultOutgoingConnections().contains(connection)) {
throw new IllegalArgumentException("connection is unknown:" + connection);
}
addConstraint(
new ConnectionRef(connection.getTo().getId(), connection.getToType()),
constraint);
} else {
throw new UnsupportedOperationException( "Constraints are " +
"only supported with XOR or OR split types, not with type:" + getType() );
}
}
public void addConstraint(ConnectionRef connectionRef, Constraint constraint) {
if (connectionRef == null) {
throw new IllegalArgumentException(
"A split node only accepts constraints linked to a connection");
}
this.constraints.put(connectionRef, constraint);
}
public Map<ConnectionRef, Constraint> getConstraints() {
return Collections.unmodifiableMap( this.constraints );
}
public void validateAddIncomingConnection(final String type, final Connection connection) {
super.validateAddIncomingConnection(type, connection);
if (!org.drools.workflow.core.Node.CONNECTION_DEFAULT_TYPE.equals(type)) {
throw new IllegalArgumentException(
"This type of node only accepts default incoming connection type!");
}
if (!getIncomingConnections(org.drools.workflow.core.Node.CONNECTION_DEFAULT_TYPE).isEmpty()) {
throw new IllegalArgumentException(
"This type of node cannot have more than one incoming connection!");
}
}
public void validateAddOutgoingConnection(final String type, final Connection connection) {
super.validateAddOutgoingConnection(type, connection);
if (!org.drools.workflow.core.Node.CONNECTION_DEFAULT_TYPE.equals(type)) {
throw new IllegalArgumentException(
"This type of node only accepts default outgoing connection type!");
}
}
public void removeOutgoingConnection(final String type, final Connection connection) {
super.removeOutgoingConnection(type, connection);
removeConstraint(connection);
}
public void removeConstraint(Connection connection) {
ConnectionRef ref = new ConnectionRef(connection.getTo().getId(), connection.getToType());
internalRemoveConstraint(ref);
}
public void internalRemoveConstraint(ConnectionRef ref) {
this.constraints.remove(ref);
}
}