mirror of
https://github.com/zrepl/zrepl.git
synced 2025-01-06 22:39:12 +01:00
126 lines
2.5 KiB
Go
126 lines
2.5 KiB
Go
package zfs
|
|
|
|
type DatasetPathForest struct {
|
|
roots []*datasetPathTree
|
|
}
|
|
|
|
func NewDatasetPathForest() *DatasetPathForest {
|
|
return &DatasetPathForest{
|
|
make([]*datasetPathTree, 0),
|
|
}
|
|
}
|
|
|
|
func (f *DatasetPathForest) Add(p *DatasetPath) {
|
|
if len(p.comps) <= 0 {
|
|
panic("dataset path too short. must have length > 0")
|
|
}
|
|
|
|
// Find its root
|
|
var root *datasetPathTree
|
|
for _, r := range f.roots {
|
|
if r.Add(p.comps) {
|
|
root = r
|
|
break
|
|
}
|
|
}
|
|
if root == nil {
|
|
root = newDatasetPathTree(p.comps)
|
|
f.roots = append(f.roots, root)
|
|
}
|
|
}
|
|
|
|
type DatasetPathVisit struct {
|
|
Path *DatasetPath
|
|
// If true, the dataset referenced by Path was not in the list of datasets to traverse
|
|
FilledIn bool
|
|
}
|
|
|
|
type DatasetPathsVisitor func(v DatasetPathVisit) (visitChildTree bool)
|
|
|
|
// Traverse a list of DatasetPaths top down, i.e. given a set of datasets with same
|
|
// path prefix, those with shorter prefix are traversed first.
|
|
// If there are gaps, i.e. the intermediary component a/b bewtween a and a/b/c,
|
|
// those gaps are still visited but the FilledIn property of the visit is set to true.
|
|
func (f *DatasetPathForest) WalkTopDown(visitor DatasetPathsVisitor) {
|
|
|
|
for _, r := range f.roots {
|
|
r.WalkTopDown([]string{}, visitor)
|
|
}
|
|
|
|
}
|
|
|
|
/* PRIVATE IMPLEMENTATION */
|
|
|
|
type datasetPathTree struct {
|
|
Component string
|
|
FilledIn bool
|
|
Children []*datasetPathTree
|
|
}
|
|
|
|
func (t *datasetPathTree) Add(p []string) bool {
|
|
|
|
if len(p) == 0 {
|
|
return true
|
|
}
|
|
|
|
if p[0] == t.Component {
|
|
|
|
remainder := p[1:]
|
|
if len(remainder) == 0 {
|
|
t.FilledIn = false
|
|
return true
|
|
}
|
|
|
|
for _, c := range t.Children {
|
|
if c.Add(remainder) {
|
|
return true
|
|
}
|
|
}
|
|
|
|
t.Children = append(t.Children, newDatasetPathTree(remainder))
|
|
return true
|
|
|
|
} else {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
func (t *datasetPathTree) WalkTopDown(parent []string, visitor DatasetPathsVisitor) {
|
|
|
|
this := append(parent, t.Component)
|
|
|
|
thisVisit := DatasetPathVisit{
|
|
&DatasetPath{this},
|
|
t.FilledIn,
|
|
}
|
|
visitChildTree := visitor(thisVisit)
|
|
|
|
if visitChildTree {
|
|
for _, c := range t.Children {
|
|
c.WalkTopDown(this, visitor)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
func newDatasetPathTree(initialComps []string) (t *datasetPathTree) {
|
|
t = &datasetPathTree{}
|
|
cur := t
|
|
for i, comp := range initialComps {
|
|
cur.Component = comp
|
|
cur.FilledIn = true
|
|
cur.Children = make([]*datasetPathTree, 0, 1)
|
|
if i == len(initialComps)-1 {
|
|
cur.FilledIn = false // last component is not filled in
|
|
break
|
|
}
|
|
child := &datasetPathTree{}
|
|
cur.Children = append(cur.Children, child)
|
|
cur = child
|
|
}
|
|
return t
|
|
}
|