!7 cmodule PhilosophyBot2 extends DynPrintLog { transient L unmappedLines = ll("good", "early morning"); transient L unmappedQuestions = ll("what is the time", "how do you feel"); transient SS mappings = ciMap(); transient S program = [[ start start => proc { run (proc { while (there are unmapped questions): find (an unmapped question) x find (an unmapped line) y map x to y } ) with backtracking } ]]; transient new PhilosophyBot1 bot; start-thread { bot.program = program; bot.addNativePredicate("there are unmapped questions", () -> nempty(unmappedQuestions)); bot.addNativePredicate("find (an unmapped question) x", (map, env) -> { print("find unmapped question x"); ret randomChoiceFromChangingList('x, unmappedQuestions, env); }); bot.addNativePredicate("find (an unmapped line) z", (map, env) -> { print("find unmapped line z"); ret randomChoiceFromChangingList('z, unmappedLines, env); }); bot.addNativePredicate("map x to y", (map, env) -> { S x = map.get("x"), y = map.get("y"); unmappedQuestions.remove(x); unmappedLines.remove(y); print("Remaining unmapped questions: " + l(unmappedQuestions)); mappings.put(x, y); print("Mapped question " + quote(x) + " to answer " + quote(y)); ret !env.wantAlternatives() ? true : new PhilosophyBot1.WithAlternative(() -> { print("Undoing mapping of " + quote(x) + " to answer " + quote(y)); unmappedQuestions.add(x); unmappedLines.add(y); false; }, true); }); bot.addNativePredicate("run x with backtracking", (SS map) -> { S x = map.get("x"); print("Run x with backtracking: " + x); bot.runParsedProcedure(bot.parseProcedure(x)); // TODO print("Ran x with backtracking: " + x); true; }); bot.run(); } O randomChoiceFromChangingList(S var, Cl list, PhilosophyBot1.Env env) { if (empty(unmappedLines)) false; if (!env.wantAlternatives()) ret litcimap(var, random(list)); ret lazyMap(cloneList(list), z -> litcimap(var, z)); } }