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.framework;
017    
018    import static junit.framework.Assert.assertEquals;
019    import static junit.framework.Assert.assertTrue;
020    import static junit.framework.Assert.fail;
021    
022    import java.util.Arrays;
023    import java.util.Collections;
024    import java.util.HashMap;
025    import java.util.HashSet;
026    import java.util.LinkedList;
027    import java.util.List;
028    import java.util.Map;
029    import java.util.Set;
030    
031    import org.apache.commons.lang.StringUtils;
032    import org.junit.Before;
033    import org.junit.Test;
034    import org.kuali.rice.krms.api.engine.Term;
035    import org.kuali.rice.krms.api.engine.TermResolutionEngine;
036    import org.kuali.rice.krms.api.engine.TermResolutionException;
037    import org.kuali.rice.krms.api.engine.TermResolver;
038    import org.kuali.rice.krms.framework.engine.TermResolutionEngineImpl;
039    import org.springframework.util.CollectionUtils;
040    
041    
042    public class TermResolutionEngineTest {
043            
044            private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(TermResolutionEngineTest.class);
045    
046            private TermResolutionEngine termResolutionEngine = null;
047            
048            @Before
049            public void setUp() {
050                    termResolutionEngine = new TermResolutionEngineImpl();
051            }
052            
053            @Test
054            public void testNoResolution() {
055                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
056                    
057                    // GIVENS:
058                    testHelper.addGivens("A");
059    
060                    testHelper.logScenarioDescription();
061                    
062                    testHelper.assertSuccess("A");
063            }
064            
065            @Test
066            public void testSimpleResolution() {
067                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
068                    
069                    // GIVENS:
070                    testHelper.addGivens("A");
071                    
072                    // RESOLVERS:
073                    testHelper.addResolver("B", /* <-- */ "A");
074                    
075                    testHelper.logScenarioDescription();
076                    
077                    testHelper.assertSuccess("B");
078            }
079            
080            @Test
081            public void testTwoStepResolution() {
082                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
083                    
084                    // GIVENS:
085                    testHelper.addGivens("A");
086                    
087                    // RESOLVERS:
088                    testHelper.addResolver("B", /* <-- */ "A");
089                    testHelper.addResolver("C", /* <-- */ "B");
090                    
091                    testHelper.logScenarioDescription();
092                    
093                    testHelper.assertSuccess("C");
094            }
095    
096            @Test
097            public void testForkingResolution() {
098                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
099    
100                    // GIVENS:
101                    testHelper.addGivens("A", "Z");
102                    
103                    // RESOLVERS:
104                    testHelper.addResolver("D", /* <-- */ "B","C");
105                    testHelper.addResolver("C", /* <-- */ "Z");
106                    testHelper.addResolver("B", /* <-- */ "A");
107                    
108                    testHelper.logScenarioDescription();
109                    
110                    testHelper.assertSuccess("D");
111            }
112            
113            @Test
114            public void testMultipleValidPaths() {
115                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
116                    
117                    // GIVENS:
118                    testHelper.addGivens("A", "Z");
119                    
120                    // RESOLVERS:
121                    testHelper.addResolver("D", /* <-- */ "B","C");
122                    testHelper.addResolver("C", /* <-- */ "Z");
123                    testHelper.addResolver("B", /* <-- */ "A");
124                    testHelper.addResolver("D", /* <-- */ "A");
125                    
126                    testHelper.logScenarioDescription();
127                    
128                    testHelper.assertSuccess("D");
129            }
130            
131            @Test
132            public void testDiamond() {
133                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
134    
135                    // GIVENS:
136                    testHelper.addGivens("A");
137                    
138                    // RESOLVERS:
139                    testHelper.addResolver("D", /* <-- */ "B","C");
140                    testHelper.addResolver("C", /* <-- */ "A");
141                    testHelper.addResolver("B", /* <-- */ "A");
142                    
143                    testHelper.logScenarioDescription();
144                    
145                    testHelper.assertSuccess("D");
146                    
147            }
148    
149            @Test
150            public void testComplexPath() {
151                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
152    
153                    // GIVENS:
154                    testHelper.addGivens("Q","R","S");
155                    
156                    // RESOLVERS:
157                    testHelper.addResolver("A", /* <-- */ "B","F");
158                    testHelper.addResolver("A", /* <-- */ "Z");
159                    testHelper.addResolver("B", /* <-- */ "D");
160                    testHelper.addResolver("B", /* <-- */ "C");
161                    testHelper.addResolver("C", /* <-- */ "S");
162                    testHelper.addResolver("D", /* <-- */ "E");
163                    testHelper.addResolver("E", /* <-- */ "S");
164                    testHelper.addResolver("F", /* <-- */ "G","Q");
165                    testHelper.addResolver("G", /* <-- */ "R");
166                    
167                    testHelper.logScenarioDescription();
168                    
169                    testHelper.assertSuccess("A");
170                    
171            }
172    
173            @Test
174            public void testCycle() {
175                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
176    
177                    // GIVENS:
178                    // none
179                    
180                    // RESOLVERS:
181                    testHelper.addResolver("D", /* <-- */ "C");
182                    testHelper.addResolver("C", /* <-- */ "D");
183                    
184                    testHelper.logScenarioDescription();
185                    
186                    testHelper.assertException("D");
187                    
188            }
189            
190    
191            @Test
192            public void testUnreachableTerm() {
193                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
194    
195                    // GIVENS:
196                    // none
197                    
198                    // RESOLVERS:
199                    testHelper.addResolver("D", /* <-- */ "C");
200                    testHelper.addResolver("C", /* <-- */ "B");
201                    
202                    testHelper.logScenarioDescription();
203                    
204                    testHelper.assertException("D");
205                    
206            }
207            
208            @Test
209            public void testRedHerringPath() {
210                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
211    
212                    // GIVENS:
213                    testHelper.addGivens("Q");
214                    
215                    // RESOLVERS:
216                    testHelper.addResolver("A", /* <-- */ "B");
217                    // this is the shortest path, but C can't be resolved
218                    testHelper.addResolver("B", /* <-- */ "C");
219                    
220                    testHelper.addResolver("B", /* <-- */ "D");
221                    testHelper.addResolver("D", /* <-- */ "Q");
222                    
223                    testHelper.logScenarioDescription();
224                    
225                    testHelper.assertSuccess("A");
226            }
227            
228            @Test
229            public void testResolveParamaterizedTerm() {
230                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
231    
232                    // GIVENS:
233                    testHelper.addGivens("Q");
234                    
235                    // RESOLVERS:
236                    testHelper.addResolver(1, "A", new String[] {"param"}, /* <-- */ "B");
237                    testHelper.addResolver("B", /* <-- */ "Q");
238                    
239                    testHelper.logScenarioDescription();
240                    
241                    Map<String,String> params = new HashMap();
242                    params.put("param", "value");
243                    testHelper.assertSuccess(testHelper.getTerm("A", params));
244    
245                    // termName A needs parameters, so this shouldn't work
246                    testHelper.assertException(new Term("A"));
247    
248            }
249            
250            @Test
251            public void testIntermediateParamaterizedTerm() {
252                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
253    
254                    // GIVENS:
255                    testHelper.addGivens("Q");
256                    
257                    // RESOLVERS:
258                    testHelper.addResolver(1, "A", new String[] {"foo"}, /* <-- */ "B");
259                    testHelper.addResolver(1, "B", new String[] {"bar"},  /* <-- */ "Q");
260                    
261                    testHelper.logScenarioDescription();
262                    
263                    Map<String,String> params = new HashMap();
264                    params.put("foo", "foovalue");
265    
266                    // Resolver for termName b requires parameters, so this shouldn't work
267                    testHelper.assertException(testHelper.getTerm("A", params));
268            }
269            
270            private static class WhiteBoxTermResolutionEngineImpl extends TermResolutionEngineImpl {
271                    
272                    // expose this for testing purposes
273                    @Override
274                    public List<TermResolverKey> buildTermResolutionPlan(String termName) {
275                            return super.buildTermResolutionPlan(termName);
276                    }
277            }
278            
279            
280            @Test
281            public void testShortestPath() {
282                    
283                    WhiteBoxTermResolutionEngineImpl whiteBoxTermResolutionService = new WhiteBoxTermResolutionEngineImpl();
284                    TestScenarioHelper testHelper = new TestScenarioHelper(whiteBoxTermResolutionService);
285    
286                    // GIVENS:
287                    testHelper.addGivens("Q");
288                    
289                    // RESOLVERS:
290                    // this one costs 3 (instead of default cost of 1)
291                    testHelper.addResolver(3, "A", /* <-- */ "Q");
292    
293                    // the steps for the alternate path each cost 1, but total length is 4
294                    testHelper.addResolver("A", /* <-- */ "B");
295                    testHelper.addResolver("B", /* <-- */ "C");
296                    testHelper.addResolver("C", /* <-- */ "D");
297                    testHelper.addResolver("D", /* <-- */ "Q");
298                    
299                    testHelper.logScenarioDescription();
300                    
301                    List<?> plan = whiteBoxTermResolutionService.buildTermResolutionPlan("A");
302                    LOG.info("resolutionPlan: " + StringUtils.join(plan, ", ") + " <-- should be length 1!");
303                    assertTrue("didn't choose the shortest resolution path (of length 1)", plan.size() == 1);
304            }
305            
306            /*
307             *  TODO: test exception variants:
308             *  - TermResolver throws TermResolutionException
309             *  - TermResolutionEngine is passed a null termName
310             */
311            
312            // TODO: what should the TermResolutionEngine do if a resolver throws a RuntimeException?
313            /*
314            @Test
315            public void testExplodingResolver() {
316                    TestScenarioHelper testHelper = new TestScenarioHelper(termResolutionEngine);
317    
318                    // GIVENS:
319                    testHelper.addGivens("Q");
320                    
321                    // RESOLVERS:
322                    TermResolverMock exploder = new TermResolverMock(testHelper.getTerm("A"), testHelper.getResult("A"), testHelper.getTerm("Q"));
323                    exploder.setIsExploder(true);
324                    termResolutionEngine.addTermResolver(exploder);
325                    
326                    testHelper.logScenarioDescription();
327                    
328                    testHelper.assertException("A");
329            }
330            */
331            
332            
333            private static class TermResolverMock<T> implements TermResolver<T> {
334                    private String output;
335                    private Set<String> params;
336                    private T result;
337                    private Set<String> prereqs;
338                    private int cost;
339                    private boolean isExploder = false;
340    
341                    public TermResolverMock(String output, T result, int cost, String ... prereqs) {
342                            this(output, null, result, cost, prereqs);
343                    }
344                    
345                    public TermResolverMock(String output, Set<String> params, T result, int cost, String ... prereqs) {
346                            this.output = output;
347                            this.params = Collections.unmodifiableSet(params);
348                            this.result = result;
349                            this.prereqs = new HashSet<String>(Arrays.asList(prereqs));
350                            this.cost = cost;
351                    }
352                    
353                    @Override
354                    public int getCost() {
355                            return cost;
356                    }
357                    
358                    @Override
359                    public String getOutput() {
360                            return output;
361                    }
362                    
363                    @Override
364                    public Set<String> getPrerequisites() {
365                            return prereqs;
366                    }
367                    
368                    @Override
369                    public Set<String> getParameterNames() {
370                            return params;
371                    }
372                    
373                    public void setIsExploder(boolean isExploder) {
374                            this.isExploder = isExploder;
375                    }
376                    
377                    @Override
378                    public T resolve(Map<String, Object> resolvedPrereqs, Map<String, String> parameters) {
379                            
380                            if (isExploder) {
381                                    throw new RuntimeException("I'm the exploder, coo coo catchoo");
382                            }
383                            
384                            // get all prereqs first
385                            for (String prereq : prereqs) {
386                                    Object result = resolvedPrereqs.get(prereq);
387                                    if (result == null) fail("got back null for prereq " + prereq);
388                            }
389    
390                            LOG.info("resolving " + output);
391                            return result;
392                    }
393                    
394                    @Override
395                    public String toString() {
396                            String paramStr = "";
397                            if (!CollectionUtils.isEmpty(params)) {
398                                    paramStr = "+(" + StringUtils.join(params, ",") + ")";
399                            }
400                            return getClass().getSimpleName()+"[ "+output+paramStr+ " <- " + StringUtils.join(prereqs.iterator(), ",") + " ]";
401                    }
402    
403            }
404            
405    
406            private static class TestScenarioHelper {
407                    
408                    private final TermResolutionEngine ars;
409                    
410                    private List<String> givens = new LinkedList<String>();
411                    private List<String> resolvers = new LinkedList<String>();
412    
413                    public TestScenarioHelper(TermResolutionEngine tre) {
414                            this.ars = tre;
415                    }
416                    
417                    public void addGivens(String ... names) {
418                            for (String name : names) {
419                                    ars.addTermValue(new Term(name, null), getResult(name));
420                                    givens.add(name);
421                            }
422                    }
423                    
424                    public String getResult(String name) {
425                            return getResult(new Term(name, null));
426                    }
427                    
428                    public String getResult(Term term) {
429                            return term.getName()  +"-result";
430                    }
431                    
432                    public Term getTerm(String name, Map<String,String> params) {
433                            return new Term(name, params);
434                    }
435                    
436                    public void addResolver(String out, String ... prereqs) {
437                            addResolver(1, out, prereqs);
438                    }
439    
440                    public void addResolver(int cost, String out, String ... prereqs) {
441                            addResolver(1, out, null, prereqs);
442                    }
443                    
444                    public void addResolver(int cost, String out, String[] params, String ... prereqs) {
445                            String [] prereqTerms = new String [prereqs.length];
446                            
447                            for (int i=0; i<prereqs.length; i++) prereqTerms[i] = prereqs[i];
448                            
449                            Set<String> paramSet = Collections.emptySet();
450                            if (params != null) paramSet = new HashSet<String>(Arrays.asList(params));
451                            
452                            ars.addTermResolver(new TermResolverMock<Object>(out, paramSet, getResult(out), cost, prereqTerms));
453                            resolvers.add("(" + out + " <- " + StringUtils.join(prereqs, ",") + ")");
454                    }
455                    
456                    public void logScenarioDescription() {
457                            StringBuilder sb = new StringBuilder();
458                            
459                            sb.append("givens: " + StringUtils.join(givens.iterator(), ", ") + "\n\n");
460                            sb.append("resolvers:\n----------------------\n");
461                            if (resolvers == null || resolvers.size() == 0) {
462                                    sb.append("none");
463                            } else { 
464                                    sb.append(StringUtils.join(resolvers.iterator(), "\n"));
465                            }
466                            sb.append("\n");
467    
468                            LOG.info("Test Scenario:\n\n" + sb.toString());
469                    }
470                    
471                    public void assertSuccess(String toResolve) {
472                            assertSuccess(new Term(toResolve, null));
473                    }
474                    
475                    public void assertSuccess(Term toResolve) {
476                            LOG.info("Testing resolution of " + toResolve);
477                            try {
478                                    assertEquals(getResult(toResolve), ars.resolveTerm(toResolve));
479                                    LOG.info("Success!");
480                            } catch (TermResolutionException e) {
481                                    fail("Should resolve the termName w/o exceptions");
482                            }
483                    }
484    
485                    public void assertException(String toResolve) {
486                            assertException(new Term(toResolve, null));
487                    }
488                            
489                    public void assertException(Term toResolve) {
490                            LOG.info("Testing resolution of " + toResolve);
491                            try {
492                                    ars.resolveTerm(toResolve);
493                                    fail("Should throw TermResolutionException");
494                            } catch (TermResolutionException e) {
495                                    LOG.info("Success! threw " + e);
496                                    // Good, this is what we expect
497                            }
498                    }
499            }
500            
501    }