1
0
Fork 0
hugo/hugolib/doctree/nodeshiftree_test.go

375 lines
9.7 KiB
Go

// Copyright 2024 The Hugo Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package doctree_test
import (
"context"
"fmt"
"math/rand"
"path"
"strings"
"testing"
qt "github.com/frankban/quicktest"
"github.com/gohugoio/hugo/common/para"
"github.com/gohugoio/hugo/hugolib/doctree"
"github.com/google/go-cmp/cmp"
)
var eq = qt.CmpEquals(
cmp.Comparer(func(n1, n2 *testValue) bool {
if n1 == n2 {
return true
}
return n1.ID == n2.ID && n1.Lang == n2.Lang
}),
)
func TestTree(t *testing.T) {
c := qt.New(t)
zeroZero := doctree.New(
doctree.Config[*testValue]{
Shifter: &testShifter{},
},
)
a := &testValue{ID: "/a"}
zeroZero.InsertIntoValuesDimension("/a", a)
ab := &testValue{ID: "/a/b"}
zeroZero.InsertIntoValuesDimension("/a/b", ab)
c.Assert(zeroZero.Get("/a"), eq, &testValue{ID: "/a", Lang: 0})
s, v := zeroZero.LongestPrefix("/a/b/c", true, nil)
c.Assert(v, eq, ab)
c.Assert(s, eq, "/a/b")
// Change language.
oneZero := zeroZero.Increment(0)
c.Assert(zeroZero.Get("/a"), eq, &testValue{ID: "/a", Lang: 0})
c.Assert(oneZero.Get("/a"), eq, &testValue{ID: "/a", Lang: 1})
}
func TestTreeData(t *testing.T) {
c := qt.New(t)
tree := doctree.New(
doctree.Config[*testValue]{
Shifter: &testShifter{},
},
)
tree.InsertIntoValuesDimension("", &testValue{ID: "HOME"})
tree.InsertIntoValuesDimension("/a", &testValue{ID: "/a"})
tree.InsertIntoValuesDimension("/a/b", &testValue{ID: "/a/b"})
tree.InsertIntoValuesDimension("/b", &testValue{ID: "/b"})
tree.InsertIntoValuesDimension("/b/c", &testValue{ID: "/b/c"})
tree.InsertIntoValuesDimension("/b/c/d", &testValue{ID: "/b/c/d"})
var values []string
ctx := &doctree.WalkContext[*testValue]{}
w := &doctree.NodeShiftTreeWalker[*testValue]{
Tree: tree,
WalkContext: ctx,
Handle: func(s string, t *testValue, match doctree.DimensionFlag) (bool, error) {
ctx.Data().Insert(s, map[string]any{
"id": t.ID,
})
if s != "" {
p, v := ctx.Data().LongestPrefix(path.Dir(s))
values = append(values, fmt.Sprintf("%s:%s:%v", s, p, v))
}
return false, nil
},
}
w.Walk(context.Background())
c.Assert(strings.Join(values, "|"), qt.Equals, "/a::map[id:HOME]|/a/b:/a:map[id:/a]|/b::map[id:HOME]|/b/c:/b:map[id:/b]|/b/c/d:/b/c:map[id:/b/c]")
}
func TestTreeEvents(t *testing.T) {
c := qt.New(t)
tree := doctree.New(
doctree.Config[*testValue]{
Shifter: &testShifter{echo: true},
},
)
tree.InsertIntoValuesDimension("/a", &testValue{ID: "/a", Weight: 2, IsBranch: true})
tree.InsertIntoValuesDimension("/a/p1", &testValue{ID: "/a/p1", Weight: 5})
tree.InsertIntoValuesDimension("/a/p", &testValue{ID: "/a/p2", Weight: 6})
tree.InsertIntoValuesDimension("/a/s1", &testValue{ID: "/a/s1", Weight: 5, IsBranch: true})
tree.InsertIntoValuesDimension("/a/s1/p1", &testValue{ID: "/a/s1/p1", Weight: 8})
tree.InsertIntoValuesDimension("/a/s1/p1", &testValue{ID: "/a/s1/p2", Weight: 9})
tree.InsertIntoValuesDimension("/a/s1/s2", &testValue{ID: "/a/s1/s2", Weight: 6, IsBranch: true})
tree.InsertIntoValuesDimension("/a/s1/s2/p1", &testValue{ID: "/a/s1/s2/p1", Weight: 8})
tree.InsertIntoValuesDimension("/a/s1/s2/p2", &testValue{ID: "/a/s1/s2/p2", Weight: 7})
w := &doctree.NodeShiftTreeWalker[*testValue]{
Tree: tree,
WalkContext: &doctree.WalkContext[*testValue]{},
}
w.Handle = func(s string, t *testValue, match doctree.DimensionFlag) (bool, error) {
if t.IsBranch {
w.WalkContext.AddEventListener("weight", s, func(e *doctree.Event[*testValue]) {
if e.Source.Weight > t.Weight {
t.Weight = e.Source.Weight
w.WalkContext.SendEvent(&doctree.Event[*testValue]{Source: t, Path: s, Name: "weight"})
}
// Reduces the amount of events bubbling up the tree. If the weight for this branch has
// increased, that will be announced in its own event.
e.StopPropagation()
})
} else {
w.WalkContext.SendEvent(&doctree.Event[*testValue]{Source: t, Path: s, Name: "weight"})
}
return false, nil
}
c.Assert(w.Walk(context.Background()), qt.IsNil)
c.Assert(w.WalkContext.HandleEventsAndHooks(), qt.IsNil)
c.Assert(tree.Get("/a").Weight, eq, 9)
c.Assert(tree.Get("/a/s1").Weight, eq, 9)
c.Assert(tree.Get("/a/p").Weight, eq, 6)
c.Assert(tree.Get("/a/s1/s2").Weight, eq, 8)
c.Assert(tree.Get("/a/s1/s2/p2").Weight, eq, 7)
}
func TestTreeInsert(t *testing.T) {
c := qt.New(t)
tree := doctree.New(
doctree.Config[*testValue]{
Shifter: &testShifter{},
},
)
a := &testValue{ID: "/a"}
tree.InsertIntoValuesDimension("/a", a)
ab := &testValue{ID: "/a/b"}
tree.InsertIntoValuesDimension("/a/b", ab)
c.Assert(tree.Get("/a"), eq, &testValue{ID: "/a", Lang: 0})
c.Assert(tree.Get("/notfound"), qt.IsNil)
ab2 := &testValue{ID: "/a/b", Lang: 0}
v, _, ok := tree.InsertIntoValuesDimension("/a/b", ab2)
c.Assert(ok, qt.IsTrue)
c.Assert(v, qt.DeepEquals, ab2)
tree1 := tree.Increment(0)
c.Assert(tree1.Get("/a/b"), qt.DeepEquals, &testValue{ID: "/a/b", Lang: 1})
}
func TestTreePara(t *testing.T) {
c := qt.New(t)
p := para.New(4)
r, _ := p.Start(context.Background())
tree := doctree.New(
doctree.Config[*testValue]{
Shifter: &testShifter{},
},
)
for i := 0; i < 8; i++ {
i := i
r.Run(func() error {
a := &testValue{ID: "/a"}
lock := tree.Lock(true)
defer lock()
tree.InsertIntoValuesDimension("/a", a)
ab := &testValue{ID: "/a/b"}
tree.InsertIntoValuesDimension("/a/b", ab)
key := fmt.Sprintf("/a/b/c/%d", i)
val := &testValue{ID: key}
tree.InsertIntoValuesDimension(key, val)
c.Assert(tree.Get(key), eq, val)
// s, _ := tree.LongestPrefix(key, nil)
// c.Assert(s, eq, "/a/b")
return nil
})
}
c.Assert(r.Wait(), qt.IsNil)
}
func TestValidateKey(t *testing.T) {
c := qt.New(t)
c.Assert(doctree.ValidateKey(""), qt.IsNil)
c.Assert(doctree.ValidateKey("/a/b/c"), qt.IsNil)
c.Assert(doctree.ValidateKey("/"), qt.IsNotNil)
c.Assert(doctree.ValidateKey("a"), qt.IsNotNil)
c.Assert(doctree.ValidateKey("abc"), qt.IsNotNil)
c.Assert(doctree.ValidateKey("/abc/"), qt.IsNotNil)
}
type testShifter struct {
echo bool
}
func (s *testShifter) ForEeachInDimension(n *testValue, d int, f func(n *testValue) bool) {
if d != doctree.DimensionLanguage.Index() {
panic("not implemented")
}
f(n)
}
func (s *testShifter) Insert(old, new *testValue) (*testValue, *testValue, bool) {
return new, old, true
}
func (s *testShifter) InsertInto(old, new *testValue, dimension doctree.Dimension) (*testValue, *testValue, bool) {
return new, old, true
}
func (s *testShifter) Delete(n *testValue, dimension doctree.Dimension) (*testValue, bool, bool) {
return nil, true, true
}
func (s *testShifter) Shift(n *testValue, dimension doctree.Dimension, exact bool) (*testValue, bool, doctree.DimensionFlag) {
if s.echo {
return n, true, doctree.DimensionLanguage
}
if n.NoCopy {
if n.Lang == dimension[0] {
return n, true, doctree.DimensionLanguage
}
return nil, false, doctree.DimensionLanguage
}
c := *n
c.Lang = dimension[0]
return &c, true, doctree.DimensionLanguage
}
func (s *testShifter) All(n *testValue) []*testValue {
return []*testValue{n}
}
type testValue struct {
ID string
Lang int
Weight int
IsBranch bool
NoCopy bool
}
func BenchmarkTreeInsert(b *testing.B) {
runBench := func(b *testing.B, numElements int) {
for i := 0; i < b.N; i++ {
tree := doctree.New(
doctree.Config[*testValue]{
Shifter: &testShifter{},
},
)
for i := 0; i < numElements; i++ {
lang := rand.Intn(2)
tree.InsertIntoValuesDimension(fmt.Sprintf("/%d", i), &testValue{ID: fmt.Sprintf("/%d", i), Lang: lang, Weight: i, NoCopy: true})
}
}
}
b.Run("1000", func(b *testing.B) {
runBench(b, 1000)
})
b.Run("10000", func(b *testing.B) {
runBench(b, 10000)
})
b.Run("100000", func(b *testing.B) {
runBench(b, 100000)
})
b.Run("300000", func(b *testing.B) {
runBench(b, 300000)
})
}
func BenchmarkWalk(b *testing.B) {
const numElements = 1000
createTree := func() *doctree.NodeShiftTree[*testValue] {
tree := doctree.New(
doctree.Config[*testValue]{
Shifter: &testShifter{},
},
)
for i := 0; i < numElements; i++ {
lang := rand.Intn(2)
tree.InsertIntoValuesDimension(fmt.Sprintf("/%d", i), &testValue{ID: fmt.Sprintf("/%d", i), Lang: lang, Weight: i, NoCopy: true})
}
return tree
}
handle := func(s string, t *testValue, match doctree.DimensionFlag) (bool, error) {
return false, nil
}
for _, numElements := range []int{1000, 10000, 100000} {
b.Run(fmt.Sprintf("Walk one dimension %d", numElements), func(b *testing.B) {
tree := createTree()
b.ResetTimer()
for i := 0; i < b.N; i++ {
w := &doctree.NodeShiftTreeWalker[*testValue]{
Tree: tree,
Handle: handle,
}
if err := w.Walk(context.Background()); err != nil {
b.Fatal(err)
}
}
})
b.Run(fmt.Sprintf("Walk all dimensions %d", numElements), func(b *testing.B) {
base := createTree()
b.ResetTimer()
for i := 0; i < b.N; i++ {
for d1 := 0; d1 < 1; d1++ {
for d2 := 0; d2 < 2; d2++ {
tree := base.Shape(d1, d2)
w := &doctree.NodeShiftTreeWalker[*testValue]{
Tree: tree,
Handle: handle,
}
if err := w.Walk(context.Background()); err != nil {
b.Fatal(err)
}
}
}
}
})
}
}