!7 /* This pattern maker takes a fixed list of examples and counterexamples and searches for PhraseCache patterns (also called theories or solutions) which distinguish positive from negative examples. Each pattern has a "complexity class" which is the number of words its literal form contains. Ultimately, we are looking for the best pattern (highest score) with the lowest complexity class. These two requirements can contradict each other, so we usually output one best pattern for each complexity class. Let's call the number of examples a pattern solves its "score" and the actual set of examples solved the "exact score". During the computation, we usually keep additional patterns as partial solutions which can then eventually be combined or extended to form a final solution. Thus two patterns which solve distinct sets of examples are good to keep around for possibly combining them. Solutions with no advantage over another known solution, such as a longer pattern but with the same complexity and exact score, can probably be discarded right away. There should also be procedures that simplify patterns without changing their operation. */ concept Example { S text; bool pos; } concept Theory { S pattern; transient simplyCached Cl solvedExamples() { ret filter(list(_concepts, Example), e -> e.pos == mmo2_match(pattern, e.text)); } transient simplyCached Cl unsolvedExamples() { ret setMinusSet(list(_concepts, Example), solvedExamples()); } int score() { ret l(solvedExamples()); } transient simplyCached int complexityClass() { ret numberOfWords2(pattern); } } cmodule PatternMaker { switchable bool enabled = true; switchable S caseID = aGlobalID(); S comment, examplesText, counterexamplesText; transient JTable theoriesTable, resultsTable; transient Concepts cc; transient Set examples, counterexamples; // collectors transient new TreeMap bestByComplexity; transient Map> allByComplexity = autoTreeMap(() -> treeSetWithDuplicatesOverCalculatedField theoryScore()); start { //cc = dm_handleCaseIDField(); //for (Theory t : list(cc, Theory)) addToCollectors(t); if (enabled) thread { think(); } } S _modifyStructForDuplication(S struct) { ret jreplace_first(struct, "caseID=*", "caseID=" + quote(aGlobalID())); } visualize { JComponent c = withCenteredButtons( northAndCenterWithMargin(dm_fieldWithLabel comment(), jvsplit( jhgrid( jCenteredSection("Examples", dm_textArea examplesText()), jCenteredSection("Counterexamples", dm_textArea counterexamplesText())), jhsplit( jCenteredSection("Results", jtabs( "Best", resultsTable = sexyTable(), "All theories", theoriesTable = sexyTable())), dm_printLogComponent())), jThreadedButton("Think", rEnter think), dm_checkBox enabled()); thread { resultsToTables(); } ret c; } void think enter { cc = new Concepts; indexConceptFieldCI(cc, Example, 'text); indexConceptFieldCI(cc, Theory, 'pattern); examples = asLinkedHashSet(tlftj(examplesText)); counterexamples = asLinkedHashSet(tlftj(counterexamplesText)); Set intersection = setIntersection(examples, counterexamples); if (nempty(intersection)) ret with infoBox("Error: Examples appear in both lists, e.g. " + first(intersection)); deleteConcepts(cc, Example); for (S s : examples) uniqCI(cc, Example, text := s, pos := true); for (S s : counterexamples) uniqCI(cc, Example, text := s, pos := false); int nExamples = countConcepts(cc, Example); print("Have " + nExamples(nExamples)); makeTheories(); print("Have " + nTheories(countConcepts(cc, Theory))); if (dm_vis() != null) resultsToTables(); } void makeTheories { // positive examples to pattern for (S s : examples) addPatterns(ai_inputExampleToPossibleMMOPatterns1(s)); // combine some pattern pairs from complexity classes 1-2 twice { for (Theory a : allByComplexity.get(1)) for (int n = 1; n <= 2; n++) for (Theory b : allByComplexity.get(n)) addPattern(mmo2_combineWithOr(a.pattern, b.pattern)); } } Map theoryToMap(Theory t) { ret t == null ? null : litorderedmap( "Pattern" := t.pattern, "Complexity class" := t.complexityClass(), "Solved examples" := l(t.solvedExamples()) + " of " + countConcepts(cc, Example), "Unsolved" := joinWithComma(quoteAll(collect text(t.unsolvedExamples())))); } void resultsToTables enter { dataToTable_uneditable_ifHasTable(resultsTable, mapValuesToList theoryToMap(bestByComplexity)); updateTabNameWithTableCount(resultsTable); dataToTable_uneditable_ifHasTable(theoriesTable, lambdaMap theoryToMap(list(cc, Theory))); updateTabNameWithTableCount(theoriesTable); } void addPatterns(Iterable l) { fOr (S s : l) addPattern(s); } void addPattern(S pattern) { addToCollectors(uniqCI_returnIfNew(cc, Theory, +pattern)); } int theoryScore(Theory t) { ret t == null ? 0 : t.score(); } void addToCollectors(Theory t) { if (t == null) ret; putIfHigherByCalculatedField theoryScore(bestByComplexity, t.complexityClass(), t); allByComplexity.get(t.complexityClass()).add(t); } }