aboutsummaryrefslogtreecommitdiffhomepage
path: root/hugolib/doctree/support.go
diff options
context:
space:
mode:
Diffstat (limited to 'hugolib/doctree/support.go')
-rw-r--r--hugolib/doctree/support.go251
1 files changed, 251 insertions, 0 deletions
diff --git a/hugolib/doctree/support.go b/hugolib/doctree/support.go
new file mode 100644
index 000000000..8083df127
--- /dev/null
+++ b/hugolib/doctree/support.go
@@ -0,0 +1,251 @@
+// 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
+
+import (
+ "fmt"
+ "strings"
+ "sync"
+
+ radix "github.com/armon/go-radix"
+)
+
+var _ MutableTrees = MutableTrees{}
+
+const (
+ LockTypeNone LockType = iota
+ LockTypeRead
+ LockTypeWrite
+)
+
+// AddEventListener adds an event listener to the tree.
+// Note that the handler func may not add listeners.
+func (ctx *WalkContext[T]) AddEventListener(event, path string, handler func(*Event[T])) {
+ if ctx.eventHandlers == nil {
+ ctx.eventHandlers = make(eventHandlers[T])
+ }
+ if ctx.eventHandlers[event] == nil {
+ ctx.eventHandlers[event] = make([]func(*Event[T]), 0)
+ }
+
+ // We want to match all above the path, so we need to exclude any similar named siblings.
+ if !strings.HasSuffix(path, "/") {
+ path += "/"
+ }
+
+ ctx.eventHandlers[event] = append(
+ ctx.eventHandlers[event], func(e *Event[T]) {
+ // Propagate events up the tree only.
+ if strings.HasPrefix(e.Path, path) {
+ handler(e)
+ }
+ },
+ )
+}
+
+// AddPostHook adds a post hook to the tree.
+// This will be run after the tree has been walked.
+func (ctx *WalkContext[T]) AddPostHook(handler func() error) {
+ ctx.HooksPost = append(ctx.HooksPost, handler)
+}
+
+func (ctx *WalkContext[T]) Data() *SimpleTree[any] {
+ ctx.dataInit.Do(func() {
+ ctx.data = &SimpleTree[any]{
+ tree: radix.New(),
+ }
+ })
+ return ctx.data
+}
+
+// SendEvent sends an event up the tree.
+func (ctx *WalkContext[T]) SendEvent(event *Event[T]) {
+ ctx.events = append(ctx.events, event)
+}
+
+// StopPropagation stops the propagation of the event.
+func (e *Event[T]) StopPropagation() {
+ e.stopPropagation = true
+}
+
+// ValidateKey returns an error if the key is not valid.
+func ValidateKey(key string) error {
+ if key == "" {
+ // Root node.
+ return nil
+ }
+
+ if len(key) < 2 {
+ return fmt.Errorf("too short key: %q", key)
+ }
+
+ if key[0] != '/' {
+ return fmt.Errorf("key must start with '/': %q", key)
+ }
+
+ if key[len(key)-1] == '/' {
+ return fmt.Errorf("key must not end with '/': %q", key)
+ }
+
+ return nil
+}
+
+// Event is used to communicate events in the tree.
+type Event[T any] struct {
+ Name string
+ Path string
+ Source T
+ stopPropagation bool
+}
+
+type LockType int
+
+// MutableTree is a tree that can be modified.
+type MutableTree interface {
+ Delete(key string)
+ DeleteAll(key string)
+ DeletePrefix(prefix string) int
+ DeletePrefixAll(prefix string) int
+ Lock(writable bool) (commit func())
+ CanLock() bool // Used for troubleshooting only.
+}
+
+// WalkableTree is a tree that can be walked.
+type WalkableTree[T any] interface {
+ WalkPrefixRaw(prefix string, walker func(key string, value T) bool)
+}
+
+var _ WalkableTree[any] = (*WalkableTrees[any])(nil)
+
+type WalkableTrees[T any] []WalkableTree[T]
+
+func (t WalkableTrees[T]) WalkPrefixRaw(prefix string, walker func(key string, value T) bool) {
+ for _, tree := range t {
+ tree.WalkPrefixRaw(prefix, walker)
+ }
+}
+
+var _ MutableTree = MutableTrees(nil)
+
+type MutableTrees []MutableTree
+
+func (t MutableTrees) Delete(key string) {
+ for _, tree := range t {
+ tree.Delete(key)
+ }
+}
+
+func (t MutableTrees) DeleteAll(key string) {
+ for _, tree := range t {
+ tree.DeleteAll(key)
+ }
+}
+
+func (t MutableTrees) DeletePrefix(prefix string) int {
+ var count int
+ for _, tree := range t {
+ count += tree.DeletePrefix(prefix)
+ }
+ return count
+}
+
+func (t MutableTrees) DeletePrefixAll(prefix string) int {
+ var count int
+ for _, tree := range t {
+ count += tree.DeletePrefixAll(prefix)
+ }
+ return count
+}
+
+func (t MutableTrees) Lock(writable bool) (commit func()) {
+ commits := make([]func(), len(t))
+ for i, tree := range t {
+ commits[i] = tree.Lock(writable)
+ }
+ return func() {
+ for _, commit := range commits {
+ commit()
+ }
+ }
+}
+
+func (t MutableTrees) CanLock() bool {
+ for _, tree := range t {
+ if !tree.CanLock() {
+ return false
+ }
+ }
+ return true
+}
+
+// WalkContext is passed to the Walk callback.
+type WalkContext[T any] struct {
+ data *SimpleTree[any]
+ dataInit sync.Once
+ eventHandlers eventHandlers[T]
+ events []*Event[T]
+
+ HooksPost []func() error
+}
+
+type eventHandlers[T any] map[string][]func(*Event[T])
+
+func cleanKey(key string) string {
+ if key == "/" {
+ // The path to the home page is logically "/",
+ // but for technical reasons, it's stored as "".
+ // This allows us to treat the home page as a section,
+ // and a prefix search for "/" will return the home page's descendants.
+ return ""
+ }
+ return key
+}
+
+func (ctx *WalkContext[T]) HandleEvents() error {
+ for len(ctx.events) > 0 {
+ event := ctx.events[0]
+ ctx.events = ctx.events[1:]
+
+ // Loop the event handlers in reverse order so
+ // that events created by the handlers themselves will
+ // be picked up further up the tree.
+ for i := len(ctx.eventHandlers[event.Name]) - 1; i >= 0; i-- {
+ ctx.eventHandlers[event.Name][i](event)
+ if event.stopPropagation {
+ break
+ }
+ }
+ }
+ return nil
+}
+
+func (ctx *WalkContext[T]) HandleEventsAndHooks() error {
+ if err := ctx.HandleEvents(); err != nil {
+ return err
+ }
+
+ for _, hook := range ctx.HooksPost {
+ if err := hook(); err != nil {
+ return err
+ }
+ }
+ return nil
+}
+
+func mustValidateKey(key string) string {
+ if err := ValidateKey(key); err != nil {
+ panic(err)
+ }
+ return key
+}