Skip to content

Commit

Permalink
Add rule indexing to compiler
Browse files Browse the repository at this point in the history
These changes add basic indexing support to the compiler. Rules can be
indexed based on their contents. In this case, the first indexing
strategy we support is based on expressions of the form: eq(term,
base_doc_ref) where the term doesn't require evaluation (except vars).
In these cases, we construct a trie that allows us to efficiently search
for rules that must be evaluated (rules that do not have to be evaluated
are excluded by the search.)

As part of these changes the virtual doc benchmark has been updated to
include rules that are hit and missed by the rule indexer.
  • Loading branch information
tsandall committed May 10, 2017
1 parent b23cb4e commit 0267568
Show file tree
Hide file tree
Showing 7 changed files with 1,051 additions and 50 deletions.
37 changes: 37 additions & 0 deletions ast/compile.go
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@ type Compiler struct {

generatedVars map[*Module]VarSet
moduleLoader ModuleLoader
ruleIndices *util.HashMap
stages []func()
}

Expand Down Expand Up @@ -161,6 +162,12 @@ func NewCompiler() *Compiler {
Modules: map[string]*Module{},
TypeEnv: NewTypeEnv(),
generatedVars: map[*Module]VarSet{},
ruleIndices: util.NewHashMap(func(a, b util.T) bool {
r1, r2 := a.(Ref), b.(Ref)
return r1.Equal(r2)
}, func(x util.T) int {
return x.(Ref).Hash()
}),
}

c.ModuleTree = NewModuleTree(nil)
Expand All @@ -178,6 +185,7 @@ func NewCompiler() *Compiler {
c.checkSafetyRuleBodies,
c.checkRecursion,
c.checkTypes,
c.buildRuleIndices,
}

return c
Expand Down Expand Up @@ -339,6 +347,18 @@ func (c *Compiler) GetRules(ref Ref) (rules []*Rule) {
return rules
}

// RuleIndex returns a RuleIndex built for the rule set referred to by path.
// The path must refer to the rule set exactly, i.e., given a rule set at path
// data.a.b.c.p, refs data.a.b.c.p.x and data.a.b.c would not return a
// RuleIndex built for the rule.
func (c *Compiler) RuleIndex(path Ref) RuleIndex {
r, ok := c.ruleIndices.Get(path)
if !ok {
return nil
}
return r.(RuleIndex)
}

// ModuleLoader defines the interface that callers can implement to enable lazy
// loading of modules during compilation.
type ModuleLoader func(resolved map[string]*Module) (parsed map[string]*Module, err error)
Expand All @@ -356,6 +376,23 @@ func (c *Compiler) WithModuleLoader(f ModuleLoader) *Compiler {
return c
}

// buildRuleIndices constructs indices for rules.
func (c *Compiler) buildRuleIndices() {

c.RuleTree.DepthFirst(func(node *RuleTreeNode) bool {
if len(node.Rules) > 1 {
index := newBaseDocEqIndex(func(ref Ref) bool {
return len(c.GetRules(ref.GroundPrefix())) > 0
})
if index.Build(node.Rules) {
c.ruleIndices.Put(node.Rules[0].Path(), index)
}
}
return false
})

}

// checkRecursion ensures that there are no recursive rule definitions, i.e., there are
// no cycles in the RuleGraph.
func (c *Compiler) checkRecursion() {
Expand Down
Loading

0 comments on commit 0267568

Please sign in to comment.