2018-03-17 07:07:48 +08:00
|
|
|
package parse
|
|
|
|
|
|
|
|
import "fmt"
|
|
|
|
|
|
|
|
// checkParseTree checks whether the parse tree part of a Node is well-formed.
|
|
|
|
func checkParseTree(n Node) error {
|
2020-04-26 01:55:44 +08:00
|
|
|
children := Children(n)
|
2018-03-17 07:07:48 +08:00
|
|
|
if len(children) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parent pointers of all children should point to me.
|
|
|
|
for i, ch := range children {
|
2020-04-26 01:55:44 +08:00
|
|
|
if Parent(ch) != n {
|
2018-03-17 07:07:48 +08:00
|
|
|
return fmt.Errorf("parent of child %d (%s) is wrong: %s", i, summary(ch), summary(n))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// The Begin of the first child should be equal to mine.
|
2018-10-13 21:07:54 +08:00
|
|
|
if children[0].Range().From != n.Range().From {
|
2018-03-17 07:07:48 +08:00
|
|
|
return fmt.Errorf("gap between node and first child: %s", summary(n))
|
|
|
|
}
|
|
|
|
// The End of the last child should be equal to mine.
|
|
|
|
nch := len(children)
|
2018-10-13 21:07:54 +08:00
|
|
|
if children[nch-1].Range().To != n.Range().To {
|
2018-03-17 07:07:48 +08:00
|
|
|
return fmt.Errorf("gap between node and last child: %s", summary(n))
|
|
|
|
}
|
|
|
|
// Consecutive children have consecutive position ranges.
|
|
|
|
for i := 0; i < nch-1; i++ {
|
2018-10-13 21:07:54 +08:00
|
|
|
if children[i].Range().To != children[i+1].Range().From {
|
2018-03-17 07:07:48 +08:00
|
|
|
return fmt.Errorf("gap between child %d and %d of: %s", i, i+1, summary(n))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check children recursively.
|
2020-04-26 01:55:44 +08:00
|
|
|
for _, ch := range Children(n) {
|
2018-03-17 07:07:48 +08:00
|
|
|
err := checkParseTree(ch)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|