001    package edu.sampleu.krms.impl;
002    
003    import org.kuali.rice.core.api.resourceloader.GlobalResourceLoader;
004    import org.kuali.rice.core.api.uif.RemotableAttributeError;
005    import org.kuali.rice.krms.api.KrmsApiServiceLocator;
006    import org.kuali.rice.krms.api.repository.function.FunctionDefinition;
007    import org.kuali.rice.krms.api.repository.function.FunctionRepositoryService;
008    import org.kuali.rice.krms.api.repository.operator.CustomOperator;
009    import org.kuali.rice.krms.api.repository.type.KrmsTypeDefinition;
010    import org.kuali.rice.krms.api.repository.type.KrmsTypeRepositoryService;
011    import org.kuali.rice.krms.framework.engine.Function;
012    import org.kuali.rice.krms.framework.type.FunctionTypeService;
013    import org.kuali.rice.krms.impl.repository.FunctionBoService;
014    
015    import java.text.MessageFormat;
016    import java.util.ArrayList;
017    import java.util.Collection;
018    import java.util.Collections;
019    import java.util.List;
020    
021    /**
022     * An example CustomOperator to demonstrate this functionality in of the KRMS UI and engine.
023     *
024     * <p>This one also does the FunctionTypeService duties to produce the engine executable, though that responsibility
025     * could be extracted to a separate class if desired, though that would require more wiring and database configuration.
026     * </p>
027     *
028     * <p>This implementation cheats a bit for easy configuration by persisting the FunctionDefinition on first access.
029     * The only prerequisite configuration to use it: 1) add a KRMS type with namespace "KR-SAP", name "contains operator",
030     * and the serviceName "sampleAppContainsOperatorService".  2) add a type relation of type 'A' (usage allowed) from
031     * your KRMS type to the context type you want to use it with.</p>
032     *
033     * <p>This service implementation is wired up in Spring and exported to the service bus.</p>
034     *
035     * <!-- NOTE: all ID and FK columns in the example SQL below should be replaced appropriately, but you knew that. -->
036     *
037     * <!-- sample SQL for creating the KRMS type:
038     * insert into krms_typ_t (typ_id, nm, nmspc_cd, srvc_nm, actv, ver_nbr)
039     * values ('OPERATOR-KRMS-TYPE-ID', 'contains operator', 'KR-SAP', 'sampleAppContainsOperatorService', 'Y', 1);
040     * -->
041     *
042     * <!-- sample SQL for creating the type relation:
043     * insert into krms_typ_reln_t (typ_reln_id, from_typ_id, to_typ_id, reln_typ, seq_no)
044     * values ('A-UNIQUE-ID', 'CONTEXT-TYPE-ID', 'OPERATOR-KRMS-TYPE-ID', 'A', 1);
045     * -->
046     *
047     * @author Kuali Rice Team (rice.collab@kuali.org)
048     */
049    public class ContainsOperator implements CustomOperator, FunctionTypeService {
050    
051        /**
052         * Returns the FunctionDefinition for the custom function the executable portion of this CustomOperator will
053         * call.
054         *
055         * <p>If the FunctionDefinition hasn't been persisted yet, this method will persist it and then return it.</p>
056         *
057         * <p>Note that having the KRMS type for this </p>
058         *
059         * @return
060         */
061        @Override
062        public FunctionDefinition getOperatorFunctionDefinition() {
063            FunctionBoService functionBoService =
064                    (FunctionBoService) GlobalResourceLoader.getService("functionRepositoryService");
065    
066            KrmsTypeRepositoryService typeRepository = KrmsApiServiceLocator.getKrmsTypeRepositoryService();
067            KrmsTypeDefinition containsOperatorType = typeRepository.getTypeByName("KR-SAP", "contains operator");
068    
069            if (containsOperatorType == null) {
070                throw new IllegalStateException("There must be a persisted KRMS type with namespace 'KR-SAP', "
071                        + "name 'contains operator', and serviceName 'sampleAppContainsOperatorService'");
072            }
073    
074            FunctionDefinition containsFunction = functionBoService.getFunctionByNameAndNamespace("contains", "KR-SAP");
075    
076            if (containsFunction == null) {
077                FunctionDefinition.Builder functionBuilder =
078                        FunctionDefinition.Builder.create("KR-SAP", "contains", "java.lang.Boolean", containsOperatorType.getId());
079    
080                containsFunction = functionBoService.createFunction(functionBuilder.build());
081            }
082    
083            return containsFunction;
084        }
085    
086        /**
087         * Validate the argument types.  This operator supports (String, String), and (Collection, Object).  For other
088         * classes, errors will be produced.  Note that at the current time (Rice 2.3.1), the rhsClassName will always be
089         * "java.lang.String" since the UI is constrained to set the second arg as a constant value.
090         *
091         * @param lhsClassName the class name for the left hand side operand
092         * @param rhsClassName the class name for the right hand side operand
093         * @return errors
094         */
095        @Override
096        public List<RemotableAttributeError> validateOperandClasses(String lhsClassName, String rhsClassName) {
097            List<RemotableAttributeError> errors = new ArrayList<RemotableAttributeError>();
098    
099            if (String.class.getName().equals(lhsClassName)) {
100                if (!String.class.getName().equals(rhsClassName)) {
101                    RemotableAttributeError.Builder errorBuilder =
102                            RemotableAttributeError.Builder.create("operator", "right hand operand is not a String");
103                    errors.add(errorBuilder.build());
104                }
105            } else {
106                try {
107                    Class<?> clazz = getClass().getClassLoader().loadClass(lhsClassName);
108    
109                    if (!Collection.class.isAssignableFrom(clazz)) {
110                        RemotableAttributeError.Builder errorBuilder =
111                                RemotableAttributeError.Builder.create("operator", "left hand operand is not a Collection");
112                        errors.add(errorBuilder.build());
113                    }
114                } catch (ClassNotFoundException e) {
115                    // it is not a class available to the class loader so we can't determine if it extends Collection or not
116                }
117            }
118    
119            return errors;
120        }
121    
122        /**
123         * Loads the Function object that the KRMS engine can execute during rule evaluation
124         *
125         * @param functionDefinition {@link FunctionDefinition} to create the {@link Function} from.
126         * @return
127         */
128        @Override
129        public Function loadFunction(FunctionDefinition functionDefinition) {
130            if (!"contains".equals(functionDefinition.getName()) || "KR-SAP".equals(functionDefinition.getNamespace())) {
131                throw new IllegalArgumentException("oops, you have the wrong type service, I can't load this function");
132            }
133    
134            // return our KRMS engine executable operator function:
135            return new Function() {
136                @Override
137                public Object invoke(Object... arguments) {
138                    if (arguments.length != 2) {
139                        throw new IllegalArgumentException("contains operator expects 2 arguments");
140                    }
141    
142                    if (arguments[0] instanceof String) {
143                        if (arguments[1] instanceof String) {
144                            return Boolean.valueOf(((String)arguments[0]).contains((String)arguments[1]));
145                        } else {
146                            throw new IllegalArgumentException("if the first argument is a String, the second argument "
147                                    + "must be as well");
148                        }
149                    } else if (arguments[0] instanceof Collection) {
150                        return Boolean.valueOf(((Collection)arguments[0]).contains(arguments[1]));
151                    }
152    
153                    throw new IllegalArgumentException("argument types must be either (String, String) or (Collection, Object)");
154                }
155            };
156        }
157    }