001 /** 002 * Copyright 2005-2014 The Kuali Foundation 003 * 004 * Licensed under the Educational Community License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.opensource.org/licenses/ecl2.php 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 package org.kuali.rice.krms.api.repository.proposition; 017 018 import java.io.Serializable; 019 import java.util.ArrayList; 020 import java.util.Collection; 021 import java.util.Collections; 022 import java.util.List; 023 024 import javax.xml.bind.annotation.XmlAccessType; 025 import javax.xml.bind.annotation.XmlAccessorType; 026 import javax.xml.bind.annotation.XmlAnyElement; 027 import javax.xml.bind.annotation.XmlElement; 028 import javax.xml.bind.annotation.XmlElementWrapper; 029 import javax.xml.bind.annotation.XmlRootElement; 030 import javax.xml.bind.annotation.XmlType; 031 import org.kuali.rice.krms.api.repository.term.TermParameterDefinition; 032 033 import org.apache.commons.lang.StringUtils; 034 import org.kuali.rice.core.api.CoreConstants; 035 import org.kuali.rice.core.api.mo.AbstractDataTransferObject; 036 import org.kuali.rice.core.api.mo.ModelBuilder; 037 import org.kuali.rice.krms.api.KrmsConstants; 038 import org.kuali.rice.krms.api.repository.LogicalOperator; 039 import org.kuali.rice.krms.api.repository.rule.RuleDefinition; 040 041 /** 042 * Concrete model object implementation of KRMS Proposition. 043 * Immutable. 044 * Instances of Proposition can be (un)marshalled to and from XML. 045 * 046 * There are three main types of Propositions: 047 * 1. Compound Propositions - a proposition consisting of other propositions 048 * and a boolean algebra operator (AND, OR) defining how to evaluate those propositions. 049 * 2. Parameterized Propositions - a proposition which is parameterized by some set of values, 050 * evaluation logic is implemented by hand and returns true or false 051 * 3. Simple Propositions - a proposition of the form lhs op rhs where 052 * lhs=left-hand side, rhs=right-hand side, and op=operator 053 * Propositions are reference by a rule or another proposition (in the case of compound propositions). 054 * Propositions are never re-used across multiple rules. 055 * Each proposition can have zero or more parameters. The proposition parameter is the primary 056 * data element used to define the proposition. (@see PropositionParameter) 057 * 058 * @see PropositonContract 059 * @see PropositionParameterContract 060 */ 061 @XmlRootElement(name = PropositionDefinition.Constants.ROOT_ELEMENT_NAME) 062 @XmlAccessorType(XmlAccessType.NONE) 063 @XmlType(name = PropositionDefinition.Constants.TYPE_NAME, propOrder = { 064 PropositionDefinition.Elements.ID, 065 PropositionDefinition.Elements.DESC, 066 PropositionDefinition.Elements.RULE_ID, 067 PropositionDefinition.Elements.TYPE_ID, 068 PropositionDefinition.Elements.PROP_TYPE_CODE, 069 PropositionDefinition.Elements.PARAMETERS, // xml element name differs from class property name 070 PropositionDefinition.Elements.CMPND_OP_CODE, 071 PropositionDefinition.Elements.CMPND_COMPONENTS, 072 CoreConstants.CommonElements.VERSION_NUMBER, 073 PropositionDefinition.Elements.CMPND_SEQ_NO, 074 CoreConstants.CommonElements.FUTURE_ELEMENTS 075 }) 076 public final class PropositionDefinition extends AbstractDataTransferObject implements PropositionDefinitionContract { 077 private static final long serialVersionUID = 2783959459503209577L; 078 079 // TODO: change this to field name to id 080 @XmlElement(name = Elements.ID, required=true) 081 private String id; 082 083 @XmlElement(name = Elements.DESC, required=true) 084 private String description; 085 086 @XmlElement(name = Elements.TYPE_ID, required=true) 087 private String typeId; 088 089 @XmlElement(name = Elements.RULE_ID, required=true) 090 private String ruleId; 091 092 @XmlElement(name = Elements.PROP_TYPE_CODE, required=true) 093 private String propositionTypeCode; 094 095 @XmlElementWrapper(name = Elements.PARAMETERS) 096 @XmlElement(name = Elements.PARAMETER, required=false) 097 private List<PropositionParameter> parameters; 098 099 @XmlElement(name = Elements.CMPND_OP_CODE, required=false) 100 private String compoundOpCode; 101 102 @XmlElementWrapper(name = Elements.CMPND_COMPONENTS, required=false) 103 @XmlElement(name = Elements.CMPND_COMPONENT, required=false) 104 private List<PropositionDefinition> compoundComponents; 105 106 @XmlElement(name = CoreConstants.CommonElements.VERSION_NUMBER, required = false) 107 private final Long versionNumber; 108 109 @XmlElement(name = Elements.CMPND_SEQ_NO, required=false) 110 private Integer compoundSequenceNumber; 111 112 @SuppressWarnings("unused") 113 @XmlAnyElement 114 private final Collection<org.w3c.dom.Element> _futureElements = null; 115 116 117 /** 118 * This constructor should never be called. It is only present for use during JAXB unmarshalling. 119 */ 120 private PropositionDefinition() { 121 this.id = null; 122 this.description = null; 123 this.typeId = null; 124 this.propositionTypeCode = null; 125 this.parameters = null; 126 this.compoundOpCode = null; 127 this.compoundSequenceNumber = null; 128 this.compoundComponents = null; 129 this.versionNumber = null; 130 } 131 132 /** 133 * Constructs a KRMS Proposition from the given builder. 134 * This constructor is private and should only ever be invoked from the builder. 135 * 136 * @param builder the Builder from which to construct the KRMS Proposition 137 */ 138 private PropositionDefinition(Builder builder) { 139 this.id = builder.getId(); 140 this.description = builder.getDescription(); 141 this.ruleId = builder.getRuleId(); 142 this.typeId = builder.getTypeId(); 143 this.propositionTypeCode = builder.getPropositionTypeCode(); 144 145 // Build parameter list 146 List<PropositionParameter> paramList = new ArrayList<PropositionParameter>(); 147 for (PropositionParameter.Builder b : builder.parameters){ 148 b.setProposition(builder); 149 paramList.add(b.build()); 150 } 151 this.parameters = Collections.unmodifiableList(paramList); 152 153 // Build Compound Proposition properties 154 this.compoundOpCode = builder.getCompoundOpCode(); 155 this.compoundSequenceNumber = builder.getCompoundSequenceNumber(); 156 List <PropositionDefinition> componentList = new ArrayList<PropositionDefinition>(); 157 if (builder.compoundComponents != null){ 158 for (PropositionDefinition.Builder b : builder.compoundComponents){ 159 componentList.add(b.build()); 160 } 161 this.compoundComponents = Collections.unmodifiableList(componentList); 162 } 163 this.versionNumber = builder.getVersionNumber(); 164 } 165 166 @Override 167 public String getId() { 168 return this.id; 169 } 170 171 @Override 172 public String getDescription() { 173 return this.description; 174 } 175 176 /** 177 * @return the ruleId 178 */ 179 @Override 180 public String getRuleId() { 181 return this.ruleId; 182 } 183 184 @Override 185 public String getTypeId() { 186 return this.typeId; 187 } 188 189 @Override 190 public String getPropositionTypeCode() { 191 return this.propositionTypeCode; 192 } 193 194 @Override 195 public List<PropositionParameter> getParameters() { 196 return this.parameters; 197 } 198 199 @Override 200 public String getCompoundOpCode() { 201 return this.compoundOpCode; 202 } 203 204 @Override 205 public List<PropositionDefinition> getCompoundComponents() { 206 return this.compoundComponents; 207 } 208 209 @Override 210 public Long getVersionNumber() { 211 return versionNumber; 212 } 213 214 @Override 215 public Integer getCompoundSequenceNumber() { 216 return this.compoundSequenceNumber; 217 } 218 219 /** 220 * This builder is used to construct instances of KRMS Proposition. It enforces the constraints of the {@link PropositionDefinitionContract}. 221 */ 222 public static class Builder implements PropositionDefinitionContract, ModelBuilder, Serializable { 223 private static final long serialVersionUID = -6889320709850568900L; 224 225 private String id; 226 private String description; 227 private String ruleId; 228 private String typeId; 229 private String propositionTypeCode; 230 private List<PropositionParameter.Builder> parameters; 231 private String compoundOpCode; 232 private Integer compoundSequenceNumber; 233 private List<PropositionDefinition.Builder> compoundComponents; 234 private RuleDefinition.Builder rule; 235 private Long versionNumber; 236 237 /** 238 * Private constructor for creating a builder with all of it's required attributes. 239 * @param propId the propId value to set 240 * @param propTypeCode the propTypeCode value to set 241 * @param ruleId the ruleId value to set 242 * @param typeId the typeId value to set 243 * @param parameters the parameters value to set 244 */ 245 private Builder(String propId, String propTypeCode, String ruleId, String typeId, List<PropositionParameter.Builder> parameters) { 246 setId(propId); 247 setPropositionTypeCode(propTypeCode); 248 setRuleId(ruleId); 249 setTypeId(typeId); 250 setParameters(parameters); 251 } 252 253 /** 254 * Set the value of the opCode to the given value. 255 * @param opCode the opCode value to set 256 * @return Builder an instance of the builder populated with given parameters 257 */ 258 public Builder compoundOpCode(String opCode){ 259 setCompoundOpCode(opCode); 260 return this; 261 } 262 263 /** 264 * Set the value of the components to the given value. 265 * @param components the components value to set 266 * @return Builder 267 */ 268 public Builder compoundComponents (List<PropositionDefinition.Builder> components){ 269 setCompoundComponents(components); 270 return this; 271 } 272 273 /** 274 * Create a Builder with the given values 275 * @param propId the propId value to set 276 * @param propTypeCode the propTypeCode value to set 277 * @param ruleId the ruleId value to set 278 * @param typeId the typeId value to set 279 * @param parameters the parameters value to set 280 * @return Builder an instance of the builder populated with given parameters 281 */ 282 public static Builder create(String propId, String propTypeCode, String ruleId, String typeId, List<PropositionParameter.Builder> parameters){ 283 return new Builder(propId, propTypeCode, ruleId, typeId, parameters); 284 } 285 286 /** 287 * Creates a builder by populating it with data from the given {@link PropositionDefinitionContract}. 288 * 289 * @param contract the contract from which to populate this builder 290 * @return an instance of the builder populated with data from the contract 291 */ 292 public static Builder create(PropositionDefinitionContract contract) { 293 if (contract == null) { 294 throw new IllegalArgumentException("contract is null"); 295 } 296 List <PropositionParameter.Builder> paramBuilderList = new ArrayList<PropositionParameter.Builder>(); 297 if (contract.getParameters() != null){ 298 for (PropositionParameterContract paramContract : contract.getParameters()){ 299 PropositionParameter.Builder myBuilder = PropositionParameter.Builder.create(paramContract); 300 paramBuilderList.add(myBuilder); 301 } 302 } 303 Builder builder = new Builder(contract.getId(), contract.getPropositionTypeCode(), contract.getRuleId(), contract.getTypeId(), paramBuilderList); 304 305 List <PropositionDefinition.Builder> componentBuilderList = new ArrayList<PropositionDefinition.Builder>(); 306 if (contract.getCompoundComponents() != null) { 307 for (PropositionDefinitionContract cContract : contract.getCompoundComponents()){ 308 PropositionDefinition.Builder pBuilder = PropositionDefinition.Builder.create(cContract); 309 componentBuilderList.add(pBuilder); 310 } 311 builder.setCompoundComponents(componentBuilderList); 312 } 313 builder.setCompoundOpCode(contract.getCompoundOpCode()); 314 builder.setCompoundSequenceNumber(contract.getCompoundSequenceNumber()); 315 builder.setDescription(contract.getDescription()); 316 builder.setVersionNumber(contract.getVersionNumber()); 317 return builder; 318 } 319 320 /** 321 * Sets the value of the propId on this builder to the given value. 322 * 323 * @param propId the propId value to set 324 * @throws IllegalArgumentException if the propId is null or blank 325 */ 326 public void setId(String propId) { 327 if (propId != null && StringUtils.isBlank(propId)) { 328 throw new IllegalArgumentException("proposition id must not be blank"); 329 } 330 this.id = propId; 331 } 332 333 /** 334 * Sets the value of the description on this builder to the given value. 335 * 336 * @param description the description value to set 337 */ 338 public void setDescription(String description) { 339 this.description = description; 340 } 341 342 /** 343 * Sets the value of the typeId on this builder to the given value. 344 * 345 * @param typeId the typeId value to set 346 */ 347 public void setTypeId(String typeId) { 348 this.typeId = typeId; 349 } 350 351 /** 352 * Sets the value of the ruleId on this builder to the given value. 353 * 354 * @param ruleId the ruleId value to set 355 */ 356 public void setRuleId(String ruleId) { 357 this.ruleId = ruleId; 358 } 359 360 /** 361 * Sets the value of the rule on this builder to the given value. 362 * 363 * @param rule the rule value to set 364 */ 365 public void setRule(RuleDefinition.Builder rule) { 366 if (rule != null && !StringUtils.isBlank(rule.getId())) { 367 setRuleId(rule.getId()); 368 } 369 this.rule = rule; 370 } 371 372 /** 373 * Sets the value of the propTypeCode on this builder to the given value. 374 * 375 * @param propTypeCode the propTypeCode value to set 376 * @throws IllegalArgumentException if the propTypeCode is null, blank or invalid 377 */ 378 public void setPropositionTypeCode(String propTypeCode) { 379 if (StringUtils.isBlank(propTypeCode)) { 380 throw new IllegalArgumentException("proposition type code is blank"); 381 } 382 if (!PropositionType.VALID_TYPE_CODES.contains(propTypeCode)) { 383 throw new IllegalArgumentException("invalid proposition type code"); 384 } 385 this.propositionTypeCode = propTypeCode; 386 } 387 388 /** 389 * Sets the value of the parameters on this builder to the given value. 390 * 391 * @param parameters the parameters value to set 392 */ 393 public void setParameters(List<PropositionParameter.Builder> parameters) { 394 // compound propositions have empty parameter lists 395 // Simple propositions must have a non-empty parameter list 396 if (parameters == null || parameters.isEmpty()){ 397 this.parameters = Collections.unmodifiableList(new ArrayList<PropositionParameter.Builder>()); 398 } else { 399 this.parameters = Collections.unmodifiableList(parameters); 400 } 401 } 402 403 /** 404 * Sets the value of the opCode on this builder to the given value. 405 * 406 * @param opCode the opCode value to set 407 * @throws IllegalArgumentException if the opCode invalid 408 */ 409 public void setCompoundOpCode(String opCode){ 410 if (StringUtils.isBlank(opCode)){ return; } 411 if (!LogicalOperator.OP_CODES.contains(opCode)){ 412 throw new IllegalArgumentException("invalid opCode value"); 413 } 414 this.compoundOpCode = opCode; 415 } 416 417 /** 418 * Sets the value of the compound sequence no on this builder to the 419 * given value. 420 * 421 * @param seqNo the sequence number for this compound prop 422 * @throws IllegalArgumentException if the seqNo invalid 423 */ 424 public void setCompoundSequenceNumber(Integer seqNo) { 425 this.compoundSequenceNumber = seqNo; 426 } 427 428 /** 429 * Sets the value of the components on this builder to the given value. 430 * 431 * @param components the components value to set 432 */ 433 public void setCompoundComponents(List<PropositionDefinition.Builder> components){ 434 if (components == null || components.isEmpty()){ 435 this.compoundComponents = new ArrayList<PropositionDefinition.Builder>(); 436 return; 437 } 438 this.compoundComponents = new ArrayList<PropositionDefinition.Builder>(components); 439 } 440 441 /** 442 * Sets the value of the versionNumber on this builder to the given value. 443 * 444 * @param versionNumber the versionNumber value to set 445 */ 446 public void setVersionNumber(Long versionNumber){ 447 this.versionNumber = versionNumber; 448 } 449 450 @Override 451 public String getId() { 452 return id; 453 } 454 455 @Override 456 public String getDescription() { 457 return description; 458 } 459 460 @Override 461 public String getRuleId() { 462 return ruleId; 463 } 464 465 @Override 466 public String getTypeId() { 467 return typeId; 468 } 469 470 @Override 471 public String getPropositionTypeCode() { 472 return propositionTypeCode; 473 } 474 475 @Override 476 public List<PropositionParameter.Builder> getParameters() { 477 return parameters; 478 } 479 480 @Override 481 public String getCompoundOpCode() { 482 return compoundOpCode; 483 } 484 485 @Override 486 public Integer getCompoundSequenceNumber() { 487 return compoundSequenceNumber; 488 } 489 490 @Override 491 public List<PropositionDefinition.Builder> getCompoundComponents() { 492 return compoundComponents; 493 } 494 495 @Override 496 public Long getVersionNumber() { 497 return versionNumber; 498 } 499 500 /** 501 * Builds an instance of a Proposition based on the current state of the builder. 502 * 503 * @return the fully-constructed Proposition 504 */ 505 @Override 506 public PropositionDefinition build() { 507 return new PropositionDefinition(this); 508 } 509 510 } 511 512 /** 513 * Defines some internal constants used on this class. 514 */ 515 static class Constants { 516 final static String ROOT_ELEMENT_NAME = "proposition"; 517 final static String TYPE_NAME = "PropositionType"; 518 } 519 520 /** 521 * A private class which exposes constants which define the XML element names to use 522 * when this object is marshalled to XML. 523 */ 524 public static class Elements { 525 final static String ID = "id"; 526 final static String DESC = "description"; 527 final static String RULE_ID = "ruleId"; 528 final static String TYPE_ID = "typeId"; 529 final static String PROP_TYPE_CODE = "propositionTypeCode"; 530 final static String PARAMETER = "parameter"; 531 final static String PARAMETERS = "parameters"; 532 final static String CMPND_OP_CODE = "compoundOpCode"; 533 final static String CMPND_SEQ_NO = "compoundSequenceNumber"; 534 final static String CMPND_COMPONENTS = "compoundComponents"; 535 final static String CMPND_COMPONENT = "proposition"; 536 } 537 538 public static class Cache { 539 public static final String NAME = KrmsConstants.Namespaces.KRMS_NAMESPACE_2_0 + "/" + PropositionDefinition.Constants.TYPE_NAME; 540 } 541 542 }