!747 !multi-line strings m { static new L rules; static new L statements; p { rules.add([["$1 is the singular of $2." <=> "$2 is the plural of $1."]]); statements.add("house is the singular of houses."); process(); print(fromLines(statements)); } static void process() { for (S rule : rules) { new Matches m; if (match3("* <=> *", rule, m)) { S l = m.unq(0), r = m.unq(1); for (Map map : matchStatements(l)) addStatement(convert(r, map)); } } } static L> matchStatements(S pat) { L> l = new ArrayList>(); for (S s : statements) { Map m = match4(pat, s); if (m != null) l.add(m); } ret l; } static Map match4(S pat, S s) { L tok1 = parse3(pat), tok2 = parse3(s); if (tok1.size() != tok2.size()) ret null; new Map m; for (int i = 1; i < tok1.size(); i += 2) { S t1 = tok1.get(i), t2 = tok2.get(i); if (isVar(t1)) m.put(t1, t2); else if (!t1.equalsIgnoreCase(t2)) ret null; } ret m; } static boolean isVar(S s) { ret s.startsWith("$") && s.length() > 1; } static S convert(S r, Map matches) { L tok = javaTokPlusPeriod(r); for (int i = 1; i < tok.size(); i += 2) { if (tok.get(i).startsWith("$")) { S m = matches.get(tok.get(i)); if (m != null) tok.set(i, m); } } ret join(tok); } static void addStatement(S s) { for (S st : statements) if (match3(st, s)) ret; statements.add(s); } !include #2000515 // unquote !include #1001497 // parseBoolean !include #1000709 // formatSnippetID }