mirror of
https://github.com/gohugoio/hugo.git
synced 2024-11-14 20:37:55 -05:00
375 lines
9.6 KiB
Go
375 lines
9.6 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 {
|
||
|
return new
|
||
|
}
|
||
|
|
||
|
func (s *testShifter) InsertInto(old, new *testValue, dimension doctree.Dimension) *testValue {
|
||
|
return new
|
||
|
}
|
||
|
|
||
|
func (s *testShifter) Delete(n *testValue, dimension doctree.Dimension) (bool, bool) {
|
||
|
return 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)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
|
||
|
}
|
||
|
}
|