mirror of
https://github.com/gohugoio/hugo.git
synced 2025-04-27 14:10:31 +03:00
Merge c91de885b5
into a88b488181
This commit is contained in:
commit
5ddfbd0112
2 changed files with 107 additions and 1 deletions
|
@ -15,8 +15,10 @@
|
||||||
package debug
|
package debug
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"reflect"
|
||||||
"sort"
|
"sort"
|
||||||
|
|
||||||
|
"encoding/json"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -183,3 +185,47 @@ func (ns *Namespace) TestDeprecationErr(item, alternative string) string {
|
||||||
hugo.Deprecate(item, alternative, v.String())
|
hugo.Deprecate(item, alternative, v.String())
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// List returns a slice of field names and method names of the struct/pointer or keys of the map
|
||||||
|
// This method scans the provided value shallow, non-recursively.
|
||||||
|
func (ns *Namespace) List(val any) []string {
|
||||||
|
|
||||||
|
fields := make([]string, 0)
|
||||||
|
value := reflect.ValueOf(val)
|
||||||
|
|
||||||
|
if value.Kind() == reflect.Map {
|
||||||
|
for _, key := range value.MapKeys() {
|
||||||
|
fields = append(fields, key.String())
|
||||||
|
sort.Strings(fields)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dereference the pointer if needed
|
||||||
|
if value.Kind() == reflect.Pointer {
|
||||||
|
value = value.Elem()
|
||||||
|
}
|
||||||
|
|
||||||
|
if value.Kind() == reflect.Struct {
|
||||||
|
// Iterate over the fields
|
||||||
|
for i := 0; i < value.NumField(); i++ {
|
||||||
|
field := value.Type().Field(i)
|
||||||
|
|
||||||
|
// Only add exported fields
|
||||||
|
if field.PkgPath == "" {
|
||||||
|
fields = append(fields, field.Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calling NumMethod() on the pointer type returns the number of methods
|
||||||
|
// defined for the pointer type as well as the non pointer type.
|
||||||
|
// Calling NumMethod() on the non pointer type returns on the other hand only the number of non pointer methods.
|
||||||
|
pointerType := reflect.PointerTo(value.Type())
|
||||||
|
|
||||||
|
for i := 0; i < pointerType.NumMethod(); i++ {
|
||||||
|
method := pointerType.Method(i)
|
||||||
|
fields = append(fields, method.Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return fields
|
||||||
|
}
|
||||||
|
|
60
tpl/debug/debug_test.go
Normal file
60
tpl/debug/debug_test.go
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
// Copyright 2023 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 debug
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Foo struct {
|
||||||
|
Bar string
|
||||||
|
foo any
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *Foo) M1() string { return "" }
|
||||||
|
func (f *Foo) M2(v string) string { return "" }
|
||||||
|
func (f *Foo) m3(v string) string { return "" }
|
||||||
|
|
||||||
|
// Non Pointer type methods
|
||||||
|
func (f Foo) M4(v string) string { return "" }
|
||||||
|
func (f Foo) m5(v string) string { return "" }
|
||||||
|
|
||||||
|
func TestList(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
namespace := new(Namespace)
|
||||||
|
|
||||||
|
for i, test := range []struct {
|
||||||
|
val any
|
||||||
|
expect []string
|
||||||
|
}{
|
||||||
|
// Map
|
||||||
|
{map[string]any{"key1": 1, "key2": 2, "key3": 3}, []string{"key1", "key2", "key3"}},
|
||||||
|
// Map non string keys
|
||||||
|
{map[int]any{1: 1, 2: 2, 3: 3}, []string{"<int Value>", "<int Value>", "<int Value>"}},
|
||||||
|
// Struct
|
||||||
|
{Foo{}, []string{"Bar", "M1", "M2", "M4"}},
|
||||||
|
// Pointer
|
||||||
|
{&Foo{}, []string{"Bar", "M1", "M2", "M4"}},
|
||||||
|
} {
|
||||||
|
t.Run(fmt.Sprintf("test%d", i), func(t *testing.T) {
|
||||||
|
result := namespace.List(test.val)
|
||||||
|
|
||||||
|
if !reflect.DeepEqual(result, test.expect) {
|
||||||
|
t.Fatalf("List called with value: %#v got\n%#v but expected\n%#v", test.val, result, test.expect)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue