PrologHub

Homoiconic Prolog: Explain yourself!

2019-10-02
Paul Brown

Explaining itself to end-users is a key feature of Expert Systems, in some applications it's absolutely vital. It also sets them apart from the more popular machine learning methods.

Homoiconicity

The reasoning in Prolog is done through rules with sub-goals, these are essentially queries against a knowledge base. Prolog is homoiconic, meaning we can manipulate these rules (i.e. the language itself) as data in Prolog. Turning these rules into something we can reason about is known as reification. That's most of the jargon!

You'll find Prolog has inbuilt clauses for inspecting code, we're most interested in clause/2, which unifies with the head and body of a predicate. A quirk of Tau-Prolog, which we're using in your browser means this only works with "public" predicates, therefore we need to declare the predicate as dynamic for the demo to work. Chances are in your dialect you won't need to do this.

:- dynamic(foo/1).

foo(X) :- bar(Y), baz(Z), X is Y + Z, \+ (X = 2 ; X = 3), something(_).

bar(2).
baz(3).
something(_).
foo(X).
clause(foo(X), Body).

That's some pretty ugly reading, but our contrived example covers a good deal of syntax, such that you shouldn't be feeling too lost when expanding on these ideas to make your own explainer.

There's two parts to being a homoiconic language though, not only do we need to be able to get at the language as data, but we need to be able to manipulate it as such. For this job we'll use a DCG.

DCG

We're not going to do a deep-dive on DCG's in this post, we do have a few DCG related posts on PrologHub. If you're in need of a complete tutorial you might want to start with the Learn Prolog Now! DCG chapter.

To manipulate the rule as data into an explanation we need to choose what each of the operators mean in natural language. We can then pattern-match on the operators for the DCG rules. We can also call the body to ensure any variables that can be ground are ground.

A couple of things to watch out for:

Also, most Prolog's provide phrase/2 or phrase/3 predicates that should be used, Tau-Prolog doesn't (yet!), so we've written a proxy, chances are you won't need this.

:- use_module(library(lists)). % import maplist/2
:- dynamic(foo/1).

foo(X) :- bar(Y), baz(Z), X is Y + Z, \+ (X = 2 ; X = 3), something(_).
bar(2). baz(3). something(_).

% explain(+Rule, -Explanation)
% generate an Explanation for Rule
explain(Rule, Explanation) :-
    nonvar(Rule), var(Explanation), % only one mode supported: +, -
    clause(Rule, Body),
    call(Body), % ground vars to values, ensure succeeds
    once(phrase(explain(:-(Rule, Body)), ExplainList)),
    maplist(var_something, ExplainList), % ground remaining vars
    atomic_list_concat(ExplainList, ' ', Explanation). % for reading

% explain//1 DCG
% match on operators and drill down to explanations
explain(:-(G, B)) --> explain(G), [is, implied, by], explain(B).
explain((A, B)) --> explain(A), [and], explain(B).
explain(;(A, B)) --> explain(A), [or], explain(B).
explain('\\+'(A)) --> ['it\'s', not, the, case, that], explain(A).
explain(is(A, B)) --> explain(B), [=, A].
explain(=(A, B)) --> explain(A), [is, the, same, as], explain(B).
explain(G) --> { G =.. [F, A, B] }, [A, F, B].
explain(G) --> { G =.. [F, A] }, [F, A].
explain(G) -->  [G].

var_something(something) :- !.
var_something(X) :- nonvar(X).

% This is a built-in for many dialects
phrase(DCG, List) :-
    call(DCG, List, []).
explain(foo(X), Explanation).

Conclusion

So there we go! Our Prolog can explain itself! Well, to a degree anyway, you'll notice that most Expert Systems seem to implement their own domain-specific query language that they use to generate explanations and to query their data. As it stands our program fails to capture disjunctive clauses that aren't separated by ;. These query languages can limit the expressiveness of the query to what you can explain via your DCG or in some cases provide more operators such as implication, equivalence, universal and existential qualification. Should you wish to implement such a thing, the DCG here should provide you with a useful inspiration for generating your explanations.

Some useful exercises to expand on this further:


Tags: dcg expert-systems reification