Help with tabling to avoid infinite left-recursion

Hi everyone,

I have an ILP algorithm that’s giving me some trouble with infinite recursion
and I wanted to use tabling for it. My first attempts failed so I thought I’d
ask for help.

It’s probably best if I avoid any descriptions of the algorithm, before
they’re really needed so let’s be lazy instead. At some point in the execution
of the algorithm, (a portion of) the dynamic database look like this:

   ,- m(ancestor, stathis, kostas).
   |  m(ancestor, stefanos, dora).
   |  m(ancestor, kostas, stassa).
   |  m(ancestor, alexandra, kostas).
E+ |  m(ancestor, paraskevi, dora).
   |  m(ancestor, dora, stassa).
   |  m(ancestor, stathis, stassa).
   |  m(ancestor, stefanos, stassa).
   |  m(ancestor, alexandra, stassa).
   '- m(ancestor, paraskevi, stassa).
   ,- m(parent, A, B) :-
   |    p(father, A, B).
   |  m(parent, A, B) :-
   |    p(mother, A, B).
BK |
   |  p(father, kostas, stassa).
   |  p(father, stathis, kostas).
   |  p(father, stefanos, dora).
   |  p(mother, alexandra, kostas).
   |  p(mother, dora, stassa).
   '- p(mother, paraskevi, dora).
   
   ,- m(ancestor, A, C) :-
H  |    m(ancestor, A, B),
   '-   m(ancestor, B, C).

In short, the clauses marked with E+ are positive examples, the clauses marked
BK are background knowledge and the clauses marked with H are the clauses of
the hypothesis that we have learned so far. The target predicate is
ancestor/2. The clauses of m/3 are “encapsulating” the E+, BK and H (for reasons
omitted for now).

You’ll notice that H is left-recursive. What’s really causing me the trouble is that the
procedure that calls H is nondeterministic, so even though there’s the “base case”
of the positive examples, H eventually causes the whole procedure to “go infinite”.
The procedure needs to be nondeterministic to be complete.

I was hoping to use tabling on m/3 to avoid infinite recursion. Clauses of m/3
are added to the dynamic database as they are induced, so I’ve tried this:

:-table m/3 as dynamic.

But my program is still going infinite. Tracing it a bit, I can see that it’s
really m/3 that’s going infinite.

You can try it yourself. Write the E+, BK and H to a file and load it, then
try a query like m(ancestor,X,Y). After a few solutions (all the pairs of X-Y
ancestor-descendant), it will enter an infinite recursion (that’s where I abort,
below):

?- m(ancestor,A,B).
A = stathis,
B = kostas ;
A = stefanos,
B = dora ;
A = kostas,
B = stassa ;
A = alexandra,
B = kostas ;
A = paraskevi,
B = dora ;
A = dora,
B = stassa ;
A = stathis,
B = stassa ;
A = stefanos,
B = stassa ;
A = alexandra,
B = stassa ;
A = paraskevi,
B = stassa ;
A = stathis,
B = stassa ;
Action (h for help) ? abort
% Execution Aborted

Is there something special I need to do or is tabling just no help in this
situation? If the latter- why?

Cheers,
Stassa

Woops. Wait a bit.

I think this actually works now.

I just updated to 8.1.19. I’m declaring m/3 as tabled, dynamic and incremental, like this:

:-table (m/3) as incremental.
:-dynamic([m/3], [incremental(true)]).

And now learning works:

?- learn_dynamic(ancestor/2).
ancestor(A,B):-parent(A,B).
ancestor(A,B):-ancestor(A,C),ancestor(C,B).
true.

That above is the query that would previously “go infinite” because it eventually called m/3.

My previous version was 8.0.3 I believe. Something definitely changed in 8.1.19 because if I remove the table directive for m/3 learn_dynamic(ancestor/2) again starts going infinite.

Jan, thank you so much, this solves so many of my problems. Left-recursion has been a huge headache to me and I had to jump through crazy hoops to avoid it. I think this is really going to work now.

Cheers,
Stassa

Edit. Oh. Just one question- I don’t always know the arity of m/… predicates before hand. Is there any way to declare tabling dynamically?

I’m just spoiled, ain’t I? :slight_smile:

While I see that you have solved your problem with tabling, if you want to know more about handling infinite left recursion then might I suggest reading

“The Craft of Prolog” by Richard A. O` Keefe (WorldCat)
Section: 5.4 The problem of Transitive Closure

Thank you Eric. That’s a great suggestion.

Just for the record. If I’m correct, this was tried on 8.0.3. 8.0.3 tries to interpret this statement as an attempt to table as/2 using mode directed tabling (answer subsumption) and (in my version) complains the mode arguments are invalid. So in the end nothing is tabled and you are in normal SLD resolution land. If you remove the as dynamic, m/3 nicely terminates, also in 8.0.3. Still, if you go tabling you better use 8.1.x and follow the latest releases closely :slight_smile:

I’m not really convinced by all the as ... What are you trying to achieve, i.e…, which predicates really get clauses added or removed?

1 Like

Just for the record. If I’m correct, this was tried on 8.0.3. 8.0.3 tries to interpret this statement as an attempt to table as/2 using mode directed tabling (answer subsumption) and (in my version) complains the mode arguments are invalid.

Yes, that’s correct. With 8.0.3. I got a mode error. Sorry, I didn’t keep a copy of the error so I don’t remember it exactly. But it was something about modes.

I’m not really convinced by all the as ... What are you trying to achieve, i.e…, which predicates really get clauses added or removed?

Well, you’re right about that too. Just declaring m/3 as tabled is sufficient to allow my program to terminate:

:-table m/3.

The predicates that get clauses added and removed, are the m/n predicates (I use assert/2 to add clauses and erase/1 to remove them from the references returned by assert/2). I don’t always know what “n” is in advance. So I might have predicates m/1, m/2, m/3 … etc.

So is it possible to declare a predicate as tabled dynamically? I mean, not as a directive? Can I call table/1 from another predicate?

Still, if you go tabling you better use 8.1.x and follow the latest releases closely

Thanks. I’ll make sure I do :slight_smile:

In the 8.1.x series you can call table/1 and untable/1 at runtime to add/remove tabling at runtime. This hasn’t been tested extensively though, so be cautious (and report issues).

Thanks! I will, definitely. Well- that is, if there are any :slight_smile:

OK, here’s a tiny tiny bugsie I found you. The following is my code to table predicates at runtime:

%!	table_encapsulated(+Target) is det.
%
%	Prepare an encapsulated learning Target for tabling.
%
table_encapsulated(_F/A):-
	succ(A,A_)
	,table(user:m/A_).

PceEmacs highlights the last literal, table(user:m/A_) in angry orange, as an error. The error message is “Type error: argument must be a table_mode”.

I reckon that’s an error message that’s left-over from 8.0.3? I had thought that PceEmacs hands over source compilation and so error detection to Swi, directly?

Anyway, the same error is not reported for untable/1. For example, below, untable(user:m/A_) is not underlined etc:

%!	untable_encapsulated(+Target) is det.
%
%	Remove tabling for an encapsulated learning Target.
%
untable_encapsulated(_F/A):-
	succ(A,A_)
	,untable(user:m/A_).

Note: the “user:” module qualifier is necessary because the tabled predicates are added to, or removed from, the module user, where it’s easy to find them and easier to distinguish them from existing definitions with the same symbol and arity in other modules. Bit of a mess, that, but it seems at least to be a stable mess. Anyway, if I don’t specify a module qualifier, the right predicate is not tabled and execution goes infinite.

Not really. PceEmacs uses library(prolog_colour) which runs the code through the cross-referencer and subsequently reads the code and applies colouring rules. It has dedicated rules for many known predicates and otherwise falls back to the general rules for atoms, variables, etc. Eventually, I guess this should be integrated with type declarations.

Pushed a fix for the above, which indeed is in part a left-over of limited functionality in 8.0.3 and in part just limited support for coloring the many syntactic variations of tabling declarations.

Not really. PceEmacs uses library(prolog_colour) which runs the code through the cross-referencer and subsequently reads the code and applies colouring rules. It has dedicated rules for many known predicates and otherwise falls back to the general rules for atoms, variables, etc. Eventually, I guess this should be integrated with type declarations.

Thanks, that’s interesting. Well, it certainly fooled me - I often see type errors, like for example in write_term/2 or other built-ins that take an option list, if I start writing an option without opening-and-closing square brackets first, the term I’m writing gets highlighted and the editor displays the error “Type error: argument must be a list”. So I really thought that some part of the Prolog runtime was being called to parse your program as you’re writing it.

Pushed a fix for the above, which indeed is in part a left-over of limited functionality in 8.0.3 and in part just limited support for coloring the many syntactic variations of tabling declarations.

Thanks for the fix! I’ll get lastest build to er get it.