Coverage Report - org.kuali.rice.krms.api.repository.proposition.PropositionDefinition
 
Classes in this File Line Coverage Branch Coverage Complexity
PropositionDefinition
94%
37/39
100%
6/6
1.75
PropositionDefinition$1
N/A
N/A
1.75
PropositionDefinition$Builder
87%
69/79
73%
25/34
1.75
PropositionDefinition$Constants
0%
0/1
N/A
1.75
PropositionDefinition$Elements
0%
0/1
N/A
1.75
 
 1  
 package org.kuali.rice.krms.api.repository.proposition;
 2  
 
 3  
 import java.io.Serializable;
 4  
 import java.util.ArrayList;
 5  
 import java.util.Collection;
 6  
 import java.util.Collections;
 7  
 import java.util.List;
 8  
 
 9  
 import javax.xml.bind.annotation.XmlAccessType;
 10  
 import javax.xml.bind.annotation.XmlAccessorType;
 11  
 import javax.xml.bind.annotation.XmlAnyElement;
 12  
 import javax.xml.bind.annotation.XmlElement;
 13  
 import javax.xml.bind.annotation.XmlElementWrapper;
 14  
 import javax.xml.bind.annotation.XmlRootElement;
 15  
 import javax.xml.bind.annotation.XmlType;
 16  
 
 17  
 import org.apache.commons.lang.StringUtils;
 18  
 import org.kuali.rice.core.api.CoreConstants;
 19  
 import org.kuali.rice.core.api.mo.AbstractDataTransferObject;
 20  
 import org.kuali.rice.core.api.mo.ModelBuilder;
 21  
 import org.kuali.rice.krms.api.repository.LogicalOperator;
 22  
 import org.kuali.rice.krms.api.repository.rule.RuleDefinition;
 23  
 
 24  
 /**
 25  
  * Concrete model object implementation of KRMS Proposition. 
 26  
  * Immutable. 
 27  
  * Instances of Proposition can be (un)marshalled to and from XML.
 28  
  *
 29  
  * There are three main types of Propositions:
 30  
  *    1. Compound Propositions - a proposition consisting of other propositions
 31  
  *             and a boolean algebra operator (AND, OR) defining how to evaluate those propositions.
 32  
  *    2. Parameterized Propositions - a proposition which is parameterized by some set of values, 
 33  
  *             evaluation logic is implemented by hand and returns true or false
 34  
  *    3. Simple Propositions - a proposition of the form lhs op rhs where 
 35  
  *            lhs=left-hand side, rhs=right-hand side, and op=operator
 36  
  * Propositions are reference by a rule or another proposition (in the case of compound propositions).
 37  
  * Propositions are never re-used across multiple rules.
 38  
  * Each proposition can have zero or more parameters. The proposition parameter is the primary 
 39  
  * data element used to define the proposition.  (@see PropositionParameter)
 40  
  * 
 41  
  * @see PropositonContract
 42  
  * @see PropositionParameterContract
 43  
  */
 44  
 @XmlRootElement(name = PropositionDefinition.Constants.ROOT_ELEMENT_NAME)
 45  
 @XmlAccessorType(XmlAccessType.NONE)
 46  
 @XmlType(name = PropositionDefinition.Constants.TYPE_NAME, propOrder = {
 47  
                 PropositionDefinition.Elements.ID,
 48  
                 PropositionDefinition.Elements.DESC,
 49  
         PropositionDefinition.Elements.RULE_ID,
 50  
         PropositionDefinition.Elements.TYPE_ID,
 51  
                 PropositionDefinition.Elements.PROP_TYPE_CODE,
 52  
                 PropositionDefinition.Elements.PARAMETERS,                                                                        // xml element name differs from class property name
 53  
                 PropositionDefinition.Elements.CMPND_OP_CODE,
 54  
                 PropositionDefinition.Elements.CMPND_COMPONENTS,
 55  
         CoreConstants.CommonElements.VERSION_NUMBER,
 56  
                 CoreConstants.CommonElements.FUTURE_ELEMENTS
 57  
 })
 58  14
 public final class PropositionDefinition extends AbstractDataTransferObject implements PropositionDefinitionContract {
 59  
         private static final long serialVersionUID = 2783959459503209577L;
 60  
 
 61  
         // TODO: change this to field name to id
 62  
         @XmlElement(name = Elements.ID, required=true)
 63  
         private String id;
 64  
         
 65  
         @XmlElement(name = Elements.DESC, required=true)
 66  
         private String description;
 67  
         
 68  
         @XmlElement(name = Elements.TYPE_ID, required=true)
 69  
         private String typeId;
 70  
         
 71  
     @XmlElement(name = Elements.RULE_ID, required=true)
 72  
     private String ruleId;
 73  
     
 74  
     @XmlElement(name = Elements.PROP_TYPE_CODE, required=true)
 75  
         private String propositionTypeCode;
 76  
 
 77  
         @XmlElementWrapper(name = Elements.PARAMETERS)
 78  
         @XmlElement(name = Elements.PARAMETER, required=false)
 79  
         private List<PropositionParameter> parameters;
 80  
         
 81  
         @XmlElement(name = Elements.CMPND_OP_CODE, required=false)
 82  
         private String compoundOpCode;
 83  
         
 84  
         @XmlElementWrapper(name = Elements.CMPND_COMPONENTS, required=false)
 85  
         @XmlElement(name = Elements.CMPND_COMPONENT, required=false)
 86  
         private List<PropositionDefinition> compoundComponents;
 87  
         
 88  
     @XmlElement(name = CoreConstants.CommonElements.VERSION_NUMBER, required = false)
 89  
     private final Long versionNumber;
 90  
             
 91  35
         @SuppressWarnings("unused")
 92  
     @XmlAnyElement
 93  
     private final Collection<org.w3c.dom.Element> _futureElements = null;
 94  
         
 95  
         
 96  
          /** 
 97  
      * This constructor should never be called.  It is only present for use during JAXB unmarshalling. 
 98  
      */
 99  21
     private PropositionDefinition() {
 100  21
             this.id = null;
 101  21
             this.description = null;
 102  21
             this.typeId = null;
 103  21
             this.propositionTypeCode = null;
 104  21
             this.parameters = null;
 105  21
             this.compoundOpCode = null;
 106  21
             this.compoundComponents = null;
 107  21
         this.versionNumber = null;
 108  21
     }
 109  
     
 110  
     /**
 111  
          * Constructs a KRMS Proposition from the given builder.  
 112  
          * This constructor is private and should only ever be invoked from the builder.
 113  
          * 
 114  
          * @param builder the Builder from which to construct the KRMS Proposition
 115  
          */
 116  14
     private PropositionDefinition(Builder builder) {
 117  14
         this.id = builder.getId();
 118  14
         this.description = builder.getDescription();
 119  14
         this.ruleId = builder.getRuleId();
 120  14
         this.typeId = builder.getTypeId();
 121  14
         this.propositionTypeCode = builder.getPropositionTypeCode();
 122  
         
 123  
         // Build parameter list
 124  14
         List<PropositionParameter> paramList = new ArrayList<PropositionParameter>();
 125  14
         for (PropositionParameter.Builder b : builder.parameters){
 126  36
                 paramList.add(b.build());
 127  
         }
 128  14
         this.parameters = Collections.unmodifiableList(paramList);
 129  
         
 130  
         // Build Compound Proposition properties
 131  14
         this.compoundOpCode = builder.getCompoundOpCode();
 132  14
         List <PropositionDefinition> componentList = new ArrayList<PropositionDefinition>();
 133  14
         if (builder.compoundComponents != null){
 134  8
                 for (PropositionDefinition.Builder b : builder.compoundComponents){
 135  4
                         componentList.add(b.build());
 136  
                 }
 137  8
             this.compoundComponents = Collections.unmodifiableList(componentList);
 138  
         }
 139  14
         this.versionNumber = builder.getVersionNumber();
 140  14
     }
 141  
     
 142  
         @Override
 143  
         public String getId() {
 144  6
                 return this.id;
 145  
         }
 146  
 
 147  
         @Override
 148  
         public String getDescription() {
 149  9
                 return this.description;
 150  
         }
 151  
 
 152  
     /**
 153  
      * @return the ruleId
 154  
      */
 155  
     public String getRuleId() {
 156  0
         return this.ruleId;
 157  
     }
 158  
 
 159  
         @Override
 160  
         public String getTypeId() {
 161  6
                 return this.typeId;
 162  
         }
 163  
 
 164  
         @Override
 165  
         public String getPropositionTypeCode() {
 166  6
                 return this.propositionTypeCode; 
 167  
         }
 168  
 
 169  
         @Override
 170  
         public List<PropositionParameter> getParameters() {
 171  5
                 return this.parameters; 
 172  
         }
 173  
 
 174  
         @Override
 175  
         public String getCompoundOpCode() {
 176  1
                 return this.compoundOpCode; 
 177  
         }
 178  
 
 179  
         @Override
 180  
         public List<PropositionDefinition> getCompoundComponents() {
 181  11
                 return this.compoundComponents; 
 182  
         }
 183  
 
 184  
     @Override
 185  
     public Long getVersionNumber() {
 186  0
         return versionNumber;
 187  
     }
 188  
         
 189  
         /**
 190  
      * This builder is used to construct instances of KRMS Proposition.  It enforces the constraints of the {@link PropositionDefinitionContract}.
 191  
      */
 192  42
     public static class Builder implements PropositionDefinitionContract, ModelBuilder, Serializable {
 193  
             private static final long serialVersionUID = -6889320709850568900L;
 194  
                 
 195  
         private String id;
 196  
         private String description;
 197  
         private String ruleId;
 198  
         private String typeId;
 199  
         private String propositionTypeCode;
 200  
         private List<PropositionParameter.Builder> parameters;
 201  
         private String compoundOpCode;
 202  
         private List<PropositionDefinition.Builder> compoundComponents;
 203  
         private RuleDefinition.Builder rule;
 204  
         private Long versionNumber;
 205  
 
 206  
                 /**
 207  
                  * Private constructor for creating a builder with all of it's required attributes.
 208  
                  * @param typeId TODO
 209  
                  */
 210  27
         private Builder(String propId, String propTypeCode, String ruleId, String typeId, List<PropositionParameter.Builder> parameters) {
 211  27
                 setId(propId);
 212  25
                         setPropositionTypeCode(propTypeCode);
 213  20
                         setRuleId(ruleId);
 214  20
                         setTypeId(typeId);
 215  20
                         setParameters(parameters);
 216  20
         }
 217  
         
 218  
         public Builder compoundOpCode(String opCode){
 219  0
                 setCompoundOpCode(opCode);
 220  0
                 return this;
 221  
         }
 222  
         
 223  
         public Builder compoundComponents (List<PropositionDefinition.Builder> components){
 224  0
                 setCompoundComponents(components);
 225  0
                 return this;
 226  
         }
 227  
  
 228  
         public static Builder create(String propId, String propTypeCode, String ruleId, String typeId, List<PropositionParameter.Builder> parameters){
 229  16
                 return new Builder(propId, propTypeCode, ruleId, typeId, parameters);
 230  
         }
 231  
         
 232  
         /**
 233  
          * Creates a builder by populating it with data from the given {@link PropositionDefinitionContract}.
 234  
          * 
 235  
          * @param contract the contract from which to populate this builder
 236  
          * @return an instance of the builder populated with data from the contract
 237  
          */
 238  
         public static Builder create(PropositionDefinitionContract contract) {
 239  11
                 if (contract == null) {
 240  0
                 throw new IllegalArgumentException("contract is null");
 241  
             }
 242  11
                 List <PropositionParameter.Builder> paramBuilderList = new ArrayList<PropositionParameter.Builder>();
 243  11
                 if (contract.getParameters() != null){
 244  11
                         for (PropositionParameterContract paramContract : contract.getParameters()){
 245  24
                                 PropositionParameter.Builder myBuilder = PropositionParameter.Builder.create(paramContract);
 246  24
                                 paramBuilderList.add(myBuilder);
 247  24
                         }
 248  
                 }
 249  11
             Builder builder =  new Builder(contract.getId(), contract.getPropositionTypeCode(), contract.getRuleId(), contract.getTypeId(), paramBuilderList);
 250  
             
 251  11
                 List <PropositionDefinition.Builder> componentBuilderList = new ArrayList<PropositionDefinition.Builder>();
 252  11
                 if (contract.getCompoundComponents() != null) {
 253  11
                         for (PropositionDefinitionContract cContract : contract.getCompoundComponents()){
 254  6
                                 PropositionDefinition.Builder pBuilder = PropositionDefinition.Builder.create(cContract);
 255  6
                                 componentBuilderList.add(pBuilder);
 256  6
                         }
 257  11
                 builder.setCompoundComponents(componentBuilderList);
 258  
                 }
 259  11
                 builder.setCompoundOpCode(contract.getCompoundOpCode());
 260  11
             builder.setDescription(contract.getDescription());
 261  11
             builder.setVersionNumber(contract.getVersionNumber());
 262  11
             return builder;
 263  
         }
 264  
 
 265  
                 /**
 266  
                  * Sets the value of the id on this builder to the given value.
 267  
                  * 
 268  
                  * @param id the id value to set
 269  
                  */
 270  
 
 271  
         public void setId(String propId) {
 272  27
             if (propId != null && StringUtils.isBlank(propId)) {
 273  2
                 throw new IllegalArgumentException("proposition id must not be blank");                    
 274  
             }
 275  25
                         this.id = propId;
 276  25
                 }
 277  
 
 278  
                 public void setDescription(String desc) {
 279  16
                         this.description = desc;
 280  16
                 }
 281  
                 
 282  
         public void setTypeId(String typeId) {
 283  21
             this.typeId = typeId;
 284  21
         }
 285  
         
 286  
         public void setRuleId(String ruleId) {
 287  20
             this.ruleId = ruleId;
 288  20
         }
 289  
         
 290  
         public void setRule(RuleDefinition.Builder rule) {
 291  0
             if (rule != null && !StringUtils.isBlank(rule.getId())) {
 292  0
                 setRuleId(rule.getId());
 293  
             }
 294  0
             this.rule = rule;
 295  0
         }
 296  
         
 297  
                 public void setPropositionTypeCode(String propTypeCode) {
 298  25
                         if (StringUtils.isBlank(propTypeCode)) {
 299  4
                 throw new IllegalArgumentException("proposition type code is blank");
 300  
                         }
 301  21
                         if (!PropositionType.VALID_TYPE_CODES.contains(propTypeCode)) {
 302  1
                 throw new IllegalArgumentException("invalid proposition type code");
 303  
                         }
 304  20
                         this.propositionTypeCode = propTypeCode;
 305  20
                 }
 306  
                 
 307  
                 public void setParameters(List<PropositionParameter.Builder> parameters) {
 308  
                         // compound propositions have empty parameter lists
 309  
                         // Simple propositions must have a non-empty parameter list
 310  20
                         if (parameters == null || parameters.isEmpty()){
 311  5
                                 this.parameters = Collections.unmodifiableList(new ArrayList<PropositionParameter.Builder>());
 312  
                         } else {
 313  15
                             this.parameters = Collections.unmodifiableList(parameters);
 314  
                         }
 315  20
                 }
 316  
                 
 317  
                 public void setCompoundOpCode(String opCode){
 318  11
                         if (StringUtils.isBlank(opCode)){ return; }
 319  3
                         if (!LogicalOperator.OP_CODES.contains(opCode)){
 320  0
                                 throw new IllegalArgumentException("invalid opCode value");
 321  
                         }
 322  3
                         this.compoundOpCode = opCode;
 323  3
                 }
 324  
 
 325  
                 public void setCompoundComponents(List<PropositionDefinition.Builder> components){
 326  11
                         if (components == null || components.isEmpty()){
 327  8
                                 this.compoundComponents = Collections.unmodifiableList(new ArrayList<PropositionDefinition.Builder>());
 328  8
                                 return;
 329  
                         }
 330  3
                         this.compoundComponents = Collections.unmodifiableList(components);
 331  3
                 }
 332  
                 
 333  
         public void setVersionNumber(Long versionNumber){
 334  12
             this.versionNumber = versionNumber;
 335  12
         }
 336  
         
 337  
                 @Override
 338  
                 public String getId() {
 339  24
                         return id;
 340  
                 }
 341  
 
 342  
                 @Override
 343  
                 public String getDescription() {
 344  20
                         return description;
 345  
                 }
 346  
                 
 347  
                 @Override
 348  
                 public String getRuleId() {
 349  20
                     return ruleId;
 350  
                 }
 351  
 
 352  
                 @Override
 353  
                 public String getTypeId() {
 354  20
                         return typeId;
 355  
                 }
 356  
 
 357  
                 @Override
 358  
                 public String getPropositionTypeCode() {
 359  20
                         return propositionTypeCode;
 360  
                 }
 361  
                 
 362  
                 @Override
 363  
                 public List<PropositionParameter.Builder> getParameters() {
 364  12
                         return parameters;
 365  
                 }
 366  
 
 367  
                 @Override
 368  
                 public String getCompoundOpCode() {
 369  20
                         return compoundOpCode;
 370  
                 }
 371  
                 
 372  
                 @Override
 373  
                 public List<PropositionDefinition.Builder> getCompoundComponents() {
 374  12
                         return compoundComponents;
 375  
                 }
 376  
 
 377  
         @Override
 378  
         public Long getVersionNumber() {
 379  20
             return versionNumber;
 380  
         }
 381  
 
 382  
                 /**
 383  
                  * Builds an instance of a Proposition based on the current state of the builder.
 384  
                  * 
 385  
                  * @return the fully-constructed Proposition
 386  
                  */
 387  
         @Override
 388  
         public PropositionDefinition build() {
 389  14
             return new PropositionDefinition(this);
 390  
         }
 391  
                 
 392  
     }
 393  
         
 394  
         /**
 395  
          * Defines some internal constants used on this class.
 396  
          */
 397  0
         static class Constants {
 398  
                 final static String ROOT_ELEMENT_NAME = "proposition";
 399  
                 final static String TYPE_NAME = "PropositionType";
 400  
         }
 401  
         
 402  
         /**
 403  
          * A private class which exposes constants which define the XML element names to use
 404  
          * when this object is marshalled to XML.
 405  
          */
 406  0
         public static class Elements {
 407  
                 final static String ID = "id";
 408  
                 final static String DESC = "description";
 409  
         final static String RULE_ID = "ruleId";
 410  
                 final static String TYPE_ID = "typeId";
 411  
                 final static String PROP_TYPE_CODE = "propositionTypeCode";
 412  
                 final static String PARAMETER = "parameter";
 413  
                 final static String PARAMETERS = "parameters";
 414  
                 final static String CMPND_OP_CODE = "compoundOpCode";
 415  
                 final static String CMPND_COMPONENTS = "compoundComponents";
 416  
                 final static String CMPND_COMPONENT = "proposition";
 417  
         }
 418  
 
 419  
         
 420  
 }