Commit e4b29aa3 authored by Sander Mathijs van Veen's avatar Sander Mathijs van Veen Committed by Taddeus Kroes

Pick the best possibiltity using the shortest path with breadth first traversal.

parent 1e478565
...@@ -22,7 +22,7 @@ from node import ExpressionNode as Node, \ ...@@ -22,7 +22,7 @@ from node import ExpressionNode as Node, \
from rules.utils import find_variable from rules.utils import find_variable
from rules.precedences import IMPLICIT_RULES from rules.precedences import IMPLICIT_RULES
from strategy import find_possibilities from strategy import find_possibilities
from possibilities import apply_suggestion from possibilities import apply_suggestion, pick_best_possibility
import Queue import Queue
import re import re
...@@ -258,7 +258,7 @@ class Parser(BisonParser): ...@@ -258,7 +258,7 @@ class Parser(BisonParser):
if self.interactive: if self.interactive:
if self.possibilities: if self.possibilities:
print self.possibilities[0] print pick_best_possibility(self, self.possibilities)
else: else:
print 'No further reduction is possible.' print 'No further reduction is possible.'
...@@ -268,12 +268,15 @@ class Parser(BisonParser): ...@@ -268,12 +268,15 @@ class Parser(BisonParser):
for i, p in enumerate(self.possibilities): for i, p in enumerate(self.possibilities):
print '%d %s' % (i, p) print '%d %s' % (i, p)
def rewrite(self, index=0, verbose=False, check_implicit=True): def rewrite(self, index=None, verbose=False, check_implicit=True):
self.find_possibilities() self.find_possibilities()
if not self.possibilities: if not self.possibilities:
return return
if index is None:
suggestion = pick_best_possibility(self, self.possibilities)
else:
suggestion = self.possibilities[index] suggestion = self.possibilities[index]
if self.verbose: if self.verbose:
...@@ -327,10 +330,10 @@ class Parser(BisonParser): ...@@ -327,10 +330,10 @@ class Parser(BisonParser):
return self.root_node return self.root_node
def rewrite_all(self, verbose=False): def rewrite_all(self, index=None, verbose=False):
i = 0 i = 0
while self.rewrite(verbose=verbose): while self.rewrite(index=index, verbose=verbose):
i += 1 i += 1
if i > 100: if i > 100:
......
from node import TYPE_OPERATOR from node import TYPE_OPERATOR, OP_NEG
# Each rule will append its hint message to the following dictionary. The # Each rule will append its hint message to the following dictionary. The
...@@ -7,6 +7,12 @@ from node import TYPE_OPERATOR ...@@ -7,6 +7,12 @@ from node import TYPE_OPERATOR
# message. The string will be processed using string.format(). # message. The string will be processed using string.format().
MESSAGES = {} MESSAGES = {}
# Dictionary used to map a stringified expression tree to the index of the
# best / shortest possibility.
BEST_POSSIBILITY_CACHE = {}
MAX_TREE_DEPTH = 20
class Possibility(object): class Possibility(object):
def __init__(self, root, handler, args=()): def __init__(self, root, handler, args=()):
...@@ -39,6 +45,36 @@ class Possibility(object): ...@@ -39,6 +45,36 @@ class Possibility(object):
and self.args == other.args and self.args == other.args
def find_possibilities(node, parent_op=None):
possibilities = []
handlers = []
from rules import RULES
if not node.is_leaf:
# Traverse through child nodes first using postorder traversal
for child in node:
possibilities += find_possibilities(child, node.op)
# Add operator-specific handlers. Prevent duplicate possibilities in
# n-ary nodes by only executing the handlers on the outermost node of
# related nodes with the same operator
#if node.op in RULES and (node.op != parent_op \
# or node.op not in NARY_OPERATORS):
# handlers += RULES[node.op]
if node.op in RULES:
handlers += RULES[node.op]
if node.negated:
handlers += RULES[OP_NEG]
# Run handlers
for handler in handlers:
possibilities += handler(node)
return possibilities
def find_parent_node(root, child): def find_parent_node(root, child):
nodes = [root] nodes = [root]
...@@ -59,25 +95,97 @@ def find_parent_node(root, child): ...@@ -59,25 +95,97 @@ def find_parent_node(root, child):
def apply_suggestion(root, suggestion): def apply_suggestion(root, suggestion):
# TODO: clone the root node before modifying. After deep copying the root # Apply the suggestion on the sub tree root with the given arguments.
# node, the subtree_map cannot be used since the hash() of each node in the
# deep copied root node has changed.
#root = root.clone()
subtree = suggestion.handler(suggestion.root, suggestion.args) subtree = suggestion.handler(suggestion.root, suggestion.args)
parent_node = find_parent_node(root, suggestion.root)
# There is either a parent node or the subtree is the root node. # Find the parent node of the sub tree and, if it has a parent node,
# FIXME: FAIL: test_diagnostic_test_application in tests/test_b1_ch08.py # substitute the new sub tree into the parent's list of child nodes.
#try: parent_node = find_parent_node(root, suggestion.root)
# assert bool(parent_node) != (subtree == root)
#except:
# print 'parent_node: %s' % (str(parent_node))
# print 'subtree: %s == %s' % (str(subtree), str(root))
# raise
if parent_node: if parent_node:
parent_node.substitute(suggestion.root, subtree) parent_node.substitute(suggestion.root, subtree)
return root return root
# Apparently the whole tree is updated. Return the new tree instead of the
# old root node
return subtree return subtree
def pick_best_possibility(parser, possibilities):
root = parser.root_node
#print 'pick_best_possibility for:', root
# Get the final expression
parser.set_root_node(root.clone())
result = parser.rewrite_all(index=0)
def traverse_breadth_first(node, result, depth=0):
if depth > MAX_TREE_DEPTH:
return
node_expr = str(node)
if node_expr in BEST_POSSIBILITY_CACHE:
return BEST_POSSIBILITY_CACHE[node_expr]
# If the nodes match, there is no rewrite step needed. If the nodes to
# not converge at all, the step_index will be None as well (since there
# still is no rewrite step to do).
step_index = None
#print ' ' * depth + 'node:', node, 'result:', result, 'equal:', \
# node.equals(result)
if not node.equals(result):
children = []
possibilities = find_possibilities(node)
#print ' ' * depth + (' ' * depth + '\n').join(map(str,
# enumerate(possibilities)))
for p, possibility in enumerate(possibilities):
#print ' ' * depth + 'possibility:', possibility, 'on:', node
# Clone the root node because it will be used in multiple
# substitutions
child = apply_suggestion(node.clone(), possibility)
#print ' ' * depth + 'child:', child
if child.equals(result):
step_index = p
break
children.append(child)
# If the final expression is not found in the direct children,
# start searching in the children of the children.
if step_index is None:
for c, child in enumerate(children):
child_step = traverse_breadth_first(child, result, depth + 1)
#print ' ' * depth + 'child_step:', child_step
if child_step is not None:
step_index = c
break
BEST_POSSIBILITY_CACHE[node_expr] = step_index
return step_index
#print '--- start traversal ---'
step_index = traverse_breadth_first(root, result)
#print 'step_index:', step_index
#print '--- cache: ---'
#print BEST_POSSIBILITY_CACHE
# Reset the parser's original state
parser.set_root_node(root)
parser.find_possibilities()
return parser.possibilities[step_index]
...@@ -87,16 +87,10 @@ def depth_possibilities(node, depth=0, parent_op=None): ...@@ -87,16 +87,10 @@ def depth_possibilities(node, depth=0, parent_op=None):
def find_possibilities(node): def find_possibilities(node):
""" """
Find all possibilities inside a node and return them in a list. Find all possibilities inside a node and return them as a list.
""" """
possibilities = depth_possibilities(node) possibilities = depth_possibilities(node)
#import copy
#old_possibilities = copy.deepcopy(possibilities)
possibilities.sort(compare_possibilities) possibilities.sort(compare_possibilities)
#get_handler = lambda (p, d): str(p.handler)
#if old_possibilities != possibilities:
# print 'before:', '\n '.join(map(get_handler, old_possibilities))
# print 'after:', '\n '.join(map(get_handler, possibilities))
return [p for p, depth in possibilities] return [p for p, depth in possibilities]
......
from parser import Parser from parser import Parser
from possibilities import apply_suggestion from possibilities import apply_suggestion, BEST_POSSIBILITY_CACHE
from strategy import find_possibilities from strategy import find_possibilities
from tests.parser import ParserWrapper from tests.parser import ParserWrapper
...@@ -16,7 +16,20 @@ def validate(exp, result): ...@@ -16,7 +16,20 @@ def validate(exp, result):
if node.equals(result): if node.equals(result):
return True return True
for p in find_possibilities(node): node_expr = str(node)
if node_expr in BEST_POSSIBILITY_CACHE:
possibility_index = BEST_POSSIBILITY_CACHE[node_expr]
# If there is no possible rewrite step, bail out
if possibility_index is None:
return False
possibilities = [possibility_index]
else:
possibilities = find_possibilities(node)
for p in possibilities:
# Clone the root node because it will be used in multiple # Clone the root node because it will be used in multiple
# substitutions # substitutions
child = apply_suggestion(node.clone(), p) child = apply_suggestion(node.clone(), p)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment