Coverage Report - org.kuali.rice.krms.impl.provider.repository.SimplePropositionTypeService
 
Classes in this File Line Coverage Branch Coverage Complexity
SimplePropositionTypeService
0%
0/57
0%
0/32
5
 
 1  
 /*
 2  
  * Copyright 2011 The Kuali Foundation
 3  
  *
 4  
  * Licensed under the Educational Community License, Version 1.0 (the "License");
 5  
  * you may not use this file except in compliance with the License.
 6  
  * You may obtain a copy of the License at
 7  
  *
 8  
  * http://www.opensource.org/licenses/ecl1.php
 9  
  *
 10  
  * Unless required by applicable law or agreed to in writing, software
 11  
  * distributed under the License is distributed on an "AS IS" BASIS,
 12  
  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 13  
  * See the License for the specific language governing permissions and
 14  
  * limitations under the License.
 15  
  */
 16  
 package org.kuali.rice.krms.impl.provider.repository;
 17  
 
 18  
 import java.util.ArrayList;
 19  
 import java.util.LinkedList;
 20  
 import java.util.List;
 21  
 import java.util.Map;
 22  
 import java.util.Set;
 23  
 import java.util.TreeMap;
 24  
 
 25  
 import org.apache.commons.collections.CollectionUtils;
 26  
 import org.apache.commons.lang.StringUtils;
 27  
 import org.kuali.rice.krms.api.engine.Term;
 28  
 import org.kuali.rice.krms.api.engine.TermResolutionEngine;
 29  
 import org.kuali.rice.krms.api.engine.TermSpecification;
 30  
 import org.kuali.rice.krms.api.repository.RepositoryDataException;
 31  
 import org.kuali.rice.krms.api.repository.function.FunctionDefinition;
 32  
 import org.kuali.rice.krms.api.repository.function.FunctionParameterDefinition;
 33  
 import org.kuali.rice.krms.api.repository.function.FunctionRepositoryService;
 34  
 import org.kuali.rice.krms.api.repository.proposition.PropositionDefinition;
 35  
 import org.kuali.rice.krms.api.repository.proposition.PropositionParameter;
 36  
 import org.kuali.rice.krms.api.repository.proposition.PropositionParameterType;
 37  
 import org.kuali.rice.krms.api.repository.term.TermDefinition;
 38  
 import org.kuali.rice.krms.api.repository.term.TermParameterDefinition;
 39  
 import org.kuali.rice.krms.api.repository.term.TermSpecificationDefinition;
 40  
 import org.kuali.rice.krms.framework.engine.Function;
 41  
 import org.kuali.rice.krms.framework.engine.Proposition;
 42  
 import org.kuali.rice.krms.framework.engine.expression.BinaryOperatorExpression;
 43  
 import org.kuali.rice.krms.framework.engine.expression.BooleanValidatingExpression;
 44  
 import org.kuali.rice.krms.framework.engine.expression.ComparisonOperator;
 45  
 import org.kuali.rice.krms.framework.engine.expression.ConstantExpression;
 46  
 import org.kuali.rice.krms.framework.engine.expression.Expression;
 47  
 import org.kuali.rice.krms.framework.engine.expression.ExpressionBasedProposition;
 48  
 import org.kuali.rice.krms.framework.engine.expression.FunctionExpression;
 49  
 import org.kuali.rice.krms.framework.engine.expression.TermExpression;
 50  
 import org.kuali.rice.krms.framework.type.FunctionTypeService;
 51  
 import org.kuali.rice.krms.framework.type.PropositionTypeService;
 52  
 import org.kuali.rice.krms.impl.repository.TermBoService;
 53  
 import org.kuali.rice.krms.impl.type.KrmsTypeResolver;
 54  
 
 55  
 /**
 56  
  * A default implementation of {@link PropositionTypeService} for propositions
 57  
  * which are composed of terms, operators, and functions.  A simple proposition
 58  
  * is self-contained and has no compound "sub" propositions.  However, it's
 59  
  * behavior is defined by the set of parameters on the {@link PropositionDefinition}.
 60  
  * 
 61  
  * @author Kuali Rice Team (rice.collab@kuali.org)
 62  
  *
 63  
  */
 64  0
 public class SimplePropositionTypeService implements PropositionTypeService {
 65  
 
 66  
         private TermBoService termBoService;
 67  
         private FunctionRepositoryService functionRepositoryService;
 68  
         private KrmsTypeResolver typeResolver;
 69  
         
 70  
         @Override
 71  
         public Proposition loadProposition(PropositionDefinition propositionDefinition) {
 72  0
                 return new ExpressionBasedProposition(translateToExpression(propositionDefinition));
 73  
         }
 74  
 
 75  
         /**
 76  
          * Translates the parameters on the given proposition definition to create an expression for evaluation.
 77  
          * The proposition parameters are defined in a reverse-polish notation so a stack is used for
 78  
          * evaluation purposes.
 79  
          * 
 80  
          * @param propositionDefinition the proposition definition to translate
 81  
          * 
 82  
          * @return the translated expression for the given proposition, this
 83  
          * expression, when evaluated, will return a Boolean.
 84  
          */
 85  
         protected Expression<Boolean> translateToExpression(PropositionDefinition propositionDefinition) {
 86  0
                 LinkedList<Expression<? extends Object>> stack = new LinkedList<Expression<? extends Object>>();
 87  0
                 for (PropositionParameter parameter : propositionDefinition.getParameters()) {
 88  0
                         PropositionParameterType parameterType = PropositionParameterType.fromCode(parameter.getParameterType());
 89  0
                         if (parameterType == PropositionParameterType.CONSTANT) {
 90  
                                 // TODO - need some way to define data type on the prop parameter as well?  Not all constants will actually be String values!!!
 91  0
                                 stack.addFirst(new ConstantExpression<String>(parameter.getValue()));
 92  0
                         } else if (parameterType == PropositionParameterType.FUNCTION) {
 93  0
                                 String functionId = parameter.getValue();
 94  0
                                 FunctionDefinition functionDefinition = functionRepositoryService.getFunction(functionId);
 95  0
                                 if (functionDefinition == null) {
 96  0
                                         throw new RepositoryDataException("Unable to locate function with the given id: " + functionId);
 97  
                                 }
 98  0
                                 FunctionTypeService functionTypeService = typeResolver.getFunctionTypeService(functionDefinition);
 99  0
                                 Function function = functionTypeService.loadFunction(functionDefinition);
 100  
                                 // TODO throw an exception if function is null?
 101  0
                                 List<FunctionParameterDefinition> parameters = functionDefinition.getParameters();
 102  0
                                 if (stack.size() < parameters.size()) {
 103  0
                                         throw new RepositoryDataException("Failed to initialize custom function '" + functionDefinition.getNamespace() + " " + functionDefinition.getName() +
 104  
                                                         "'.  There were only " + stack.size() + " values on the stack but function requires at least " + parameters.size());
 105  
                                 }
 106  0
                                 List<Expression<? extends Object>> arguments = new ArrayList<Expression<? extends Object>>();
 107  
                                 // work backward through the list to match params to the stack
 108  0
                                 for (int index = parameters.size() - 1; index >= 0; index--) {
 109  0
                                         FunctionParameterDefinition parameterDefinition = parameters.get(index);
 110  
                                         // TODO need to check types here? expression object probably needs a getType on it so that we can confirm that the types will be compatible?
 111  0
                                         Expression<? extends Object> argument = stack.removeFirst();
 112  0
                                         arguments.add(argument);
 113  
                                 }
 114  0
                                 stack.addFirst(new FunctionExpression(function, arguments));
 115  0
                         } else if (parameterType == PropositionParameterType.OPERATOR) {
 116  0
                                 ComparisonOperator operator = ComparisonOperator.fromCode(parameter.getValue());
 117  0
                                 if (stack.size() < 2) {
 118  0
                                         throw new RepositoryDataException("Failed to initialize expression for comparison operator " + operator + " because a sufficient number of arguments was not available on the stack.  Current contents of stack: " + stack.toString());
 119  
                                 }
 120  0
                                 Expression<? extends Object> rhs = stack.removeFirst();
 121  0
                                 Expression<? extends Object> lhs = stack.removeFirst();
 122  0
                                 stack.addFirst(new BinaryOperatorExpression(operator, lhs, rhs));
 123  0
                         } else if (parameterType == PropositionParameterType.TERM) {
 124  0
                                 String termId = parameter.getValue();
 125  
 
 126  0
                                 TermDefinition termDefinition = termBoService.getTermById(termId);
 127  0
                                 if (termDefinition == null) { throw new RepositoryDataException("unable to load term with id " + termId);}
 128  0
                                 Term term = translateTermDefinition(termDefinition);
 129  
                                 
 130  0
                                 stack.addFirst(new TermExpression(term));
 131  
                         }
 132  0
                 }
 133  0
                 if (stack.size() != 1) {
 134  0
                         throw new RepositoryDataException("Final contents of expression stack are incorrect, there should only be one entry but was " + stack.size() +".  Current contents of stack: " + stack.toString());
 135  
                 }
 136  0
                 return new BooleanValidatingExpression(stack.removeFirst());
 137  
         }
 138  
         
 139  
         protected Term translateTermDefinition(TermDefinition termDefinition) {
 140  0
                 if (termDefinition == null) {
 141  0
                         throw new RepositoryDataException("Given TermDefinition is null");
 142  
                 }
 143  0
                 TermSpecificationDefinition termSpecificationDefinition = termDefinition.getSpecification();
 144  0
                 if (termSpecificationDefinition == null) { throw new RepositoryDataException("term with id " + termDefinition.getId() + " has a null specification"); } 
 145  
                 
 146  0
                 Set<TermParameterDefinition> params = termDefinition.getParameters();
 147  0
                 Map<String,String> paramsMap = new TreeMap<String,String>();
 148  0
                 if (!CollectionUtils.isEmpty(params)) for (TermParameterDefinition param : params) {
 149  0
                         if (StringUtils.isBlank(param.getName())) { 
 150  0
                                 throw new RepositoryDataException("TermParameterDefinition.name may not be blank"); 
 151  
                         }
 152  0
                         paramsMap.put(param.getName(), param.getValue());
 153  
                 }
 154  
                 
 155  0
                 return new Term(new TermSpecification(termSpecificationDefinition.getName(), termSpecificationDefinition.getType()), paramsMap);
 156  
         }
 157  
 
 158  
         /**
 159  
          * @param termBoService the termBoService to set
 160  
          */
 161  
         public void setTermBoService(TermBoService termBoService) {
 162  0
                 this.termBoService = termBoService;
 163  0
         }
 164  
 
 165  
         public void setFunctionRepositoryService(FunctionRepositoryService functionRepositoryService) {
 166  0
                 this.functionRepositoryService = functionRepositoryService;
 167  0
         }
 168  
         
 169  
         public void setTypeResolver(KrmsTypeResolver typeResolver) {
 170  0
                 this.typeResolver = typeResolver;
 171  0
         }
 172  
         
 173  
 }