mirror of
https://github.com/gohugoio/hugo.git
synced 2024-11-07 20:30:36 -05:00
Merge pull request #6149 from bep/sort-caseinsensitive
Implement lexicographically string sorting
This commit is contained in:
parent
a4f96a9d8c
commit
53077b0da5
14 changed files with 246 additions and 34 deletions
|
@ -1,4 +1,4 @@
|
||||||
// Copyright 2017-present The Hugo Authors. All rights reserved.
|
// Copyright 2019 The Hugo Authors. All rights reserved.
|
||||||
//
|
//
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
// you may not use this file except in compliance with the License.
|
// you may not use this file except in compliance with the License.
|
||||||
|
@ -20,7 +20,7 @@ type Eqer interface {
|
||||||
Eq(other interface{}) bool
|
Eq(other interface{}) bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// ProbablyEq is an equal check that may return false positives, but never
|
// ProbablyEqer is an equal check that may return false positives, but never
|
||||||
// a false negative.
|
// a false negative.
|
||||||
type ProbablyEqer interface {
|
type ProbablyEqer interface {
|
||||||
ProbablyEq(other interface{}) bool
|
ProbablyEq(other interface{}) bool
|
||||||
|
|
113
compare/compare_strings.go
Normal file
113
compare/compare_strings.go
Normal file
|
@ -0,0 +1,113 @@
|
||||||
|
// Copyright 2019 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 compare
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
"unicode"
|
||||||
|
"unicode/utf8"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Strings returns an integer comparing two strings lexicographically.
|
||||||
|
func Strings(s, t string) int {
|
||||||
|
c := compareFold(s, t)
|
||||||
|
|
||||||
|
if c == 0 {
|
||||||
|
// "B" and "b" would be the same so we need a tiebreaker.
|
||||||
|
return strings.Compare(s, t)
|
||||||
|
}
|
||||||
|
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// This function is derived from strings.EqualFold in Go's stdlib.
|
||||||
|
// https://github.com/golang/go/blob/ad4a58e31501bce5de2aad90a620eaecdc1eecb8/src/strings/strings.go#L893
|
||||||
|
func compareFold(s, t string) int {
|
||||||
|
for s != "" && t != "" {
|
||||||
|
var sr, tr rune
|
||||||
|
if s[0] < utf8.RuneSelf {
|
||||||
|
sr, s = rune(s[0]), s[1:]
|
||||||
|
} else {
|
||||||
|
r, size := utf8.DecodeRuneInString(s)
|
||||||
|
sr, s = r, s[size:]
|
||||||
|
}
|
||||||
|
if t[0] < utf8.RuneSelf {
|
||||||
|
tr, t = rune(t[0]), t[1:]
|
||||||
|
} else {
|
||||||
|
r, size := utf8.DecodeRuneInString(t)
|
||||||
|
tr, t = r, t[size:]
|
||||||
|
}
|
||||||
|
|
||||||
|
if tr == sr {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
c := 1
|
||||||
|
if tr < sr {
|
||||||
|
tr, sr = sr, tr
|
||||||
|
c = -c
|
||||||
|
}
|
||||||
|
|
||||||
|
// ASCII only.
|
||||||
|
if tr < utf8.RuneSelf {
|
||||||
|
if sr >= 'A' && sr <= 'Z' {
|
||||||
|
if tr <= 'Z' {
|
||||||
|
// Same case.
|
||||||
|
return -c
|
||||||
|
}
|
||||||
|
|
||||||
|
diff := tr - (sr + 'a' - 'A')
|
||||||
|
|
||||||
|
if diff == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if diff < 0 {
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
if diff > 0 {
|
||||||
|
return -c
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unicode.
|
||||||
|
r := unicode.SimpleFold(sr)
|
||||||
|
for r != sr && r < tr {
|
||||||
|
r = unicode.SimpleFold(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
if r == tr {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
return -c
|
||||||
|
}
|
||||||
|
|
||||||
|
if s == "" && t == "" {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
if s == "" {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
|
// LessStrings returns whether s is less than t lexicographically.
|
||||||
|
func LessStrings(s, t string) bool {
|
||||||
|
return Strings(s, t) < 0
|
||||||
|
}
|
66
compare/compare_strings_test.go
Normal file
66
compare/compare_strings_test.go
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
// Copyright 2019 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 compare
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestCompare(t *testing.T) {
|
||||||
|
assert := require.New(t)
|
||||||
|
for i, test := range []struct {
|
||||||
|
a string
|
||||||
|
b string
|
||||||
|
}{
|
||||||
|
{"a", "a"},
|
||||||
|
{"A", "a"},
|
||||||
|
{"Ab", "Ac"},
|
||||||
|
{"az", "Za"},
|
||||||
|
{"C", "D"},
|
||||||
|
{"B", "a"},
|
||||||
|
{"C", ""},
|
||||||
|
{"", ""},
|
||||||
|
{"αβδC", "ΑΒΔD"},
|
||||||
|
{"αβδC", "ΑΒΔ"},
|
||||||
|
{"αβδ", "ΑΒΔD"},
|
||||||
|
{"αβδ", "ΑΒΔ"},
|
||||||
|
{"β", "δ"},
|
||||||
|
{"好", strings.ToLower("好")},
|
||||||
|
} {
|
||||||
|
|
||||||
|
expect := strings.Compare(strings.ToLower(test.a), strings.ToLower(test.b))
|
||||||
|
got := compareFold(test.a, test.b)
|
||||||
|
|
||||||
|
assert.Equal(expect, got, fmt.Sprintf("test %d: %d", i, expect))
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLexicographicSort(t *testing.T) {
|
||||||
|
assert := require.New(t)
|
||||||
|
|
||||||
|
s := []string{"b", "Bz", "ba", "A", "Ba", "ba"}
|
||||||
|
|
||||||
|
sort.Slice(s, func(i, j int) bool {
|
||||||
|
return LessStrings(s[i], s[j])
|
||||||
|
})
|
||||||
|
|
||||||
|
assert.Equal([]string{"A", "b", "Ba", "ba", "ba", "Bz"}, s)
|
||||||
|
|
||||||
|
}
|
|
@ -213,8 +213,8 @@ menu: "main"
|
||||||
|
|
||||||
b.Build(BuildCfg{})
|
b.Build(BuildCfg{})
|
||||||
|
|
||||||
b.AssertFileContent("public/index.html", "AMP and HTML|/blog/html-amp/|AMP only|/amp/blog/amp/|HTML only|/blog/html/|Home Sweet Home|/|")
|
b.AssertFileContent("public/index.html", "AMP and HTML|/blog/html-amp/|AMP only|/amp/blog/amp/|Home Sweet Home|/|HTML only|/blog/html/|")
|
||||||
b.AssertFileContent("public/amp/index.html", "AMP and HTML|/amp/blog/html-amp/|AMP only|/amp/blog/amp/|HTML only|/blog/html/|Home Sweet Home|/amp/|")
|
b.AssertFileContent("public/amp/index.html", "AMP and HTML|/amp/blog/html-amp/|AMP only|/amp/blog/amp/|Home Sweet Home|/amp/|HTML only|/blog/html/|")
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://github.com/gohugoio/hugo/issues/5989
|
// https://github.com/gohugoio/hugo/issues/5989
|
||||||
|
|
|
@ -18,6 +18,8 @@ import (
|
||||||
"path"
|
"path"
|
||||||
"sort"
|
"sort"
|
||||||
|
|
||||||
|
"github.com/gohugoio/hugo/compare"
|
||||||
|
|
||||||
"github.com/gohugoio/hugo/resources/page"
|
"github.com/gohugoio/hugo/resources/page"
|
||||||
"github.com/gohugoio/hugo/resources/resource"
|
"github.com/gohugoio/hugo/resources/resource"
|
||||||
)
|
)
|
||||||
|
@ -73,7 +75,7 @@ func (i Taxonomy) TaxonomyArray() OrderedTaxonomy {
|
||||||
// Alphabetical returns an ordered taxonomy sorted by key name.
|
// Alphabetical returns an ordered taxonomy sorted by key name.
|
||||||
func (i Taxonomy) Alphabetical() OrderedTaxonomy {
|
func (i Taxonomy) Alphabetical() OrderedTaxonomy {
|
||||||
name := func(i1, i2 *OrderedTaxonomyEntry) bool {
|
name := func(i1, i2 *OrderedTaxonomyEntry) bool {
|
||||||
return i1.Name < i2.Name
|
return compare.LessStrings(i1.Name, i2.Name)
|
||||||
}
|
}
|
||||||
|
|
||||||
ia := i.TaxonomyArray()
|
ia := i.TaxonomyArray()
|
||||||
|
@ -89,7 +91,7 @@ func (i Taxonomy) ByCount() OrderedTaxonomy {
|
||||||
li2 := len(i2.WeightedPages)
|
li2 := len(i2.WeightedPages)
|
||||||
|
|
||||||
if li1 == li2 {
|
if li1 == li2 {
|
||||||
return i1.Name < i2.Name
|
return compare.LessStrings(i1.Name, i2.Name)
|
||||||
}
|
}
|
||||||
return li1 > li2
|
return li1 > li2
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,6 +15,7 @@ package navigation
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/gohugoio/hugo/common/types"
|
"github.com/gohugoio/hugo/common/types"
|
||||||
|
"github.com/gohugoio/hugo/compare"
|
||||||
|
|
||||||
"html/template"
|
"html/template"
|
||||||
"sort"
|
"sort"
|
||||||
|
@ -159,10 +160,11 @@ func (by menuEntryBy) Sort(menu Menu) {
|
||||||
|
|
||||||
var defaultMenuEntrySort = func(m1, m2 *MenuEntry) bool {
|
var defaultMenuEntrySort = func(m1, m2 *MenuEntry) bool {
|
||||||
if m1.Weight == m2.Weight {
|
if m1.Weight == m2.Weight {
|
||||||
if m1.Name == m2.Name {
|
c := compare.Strings(m1.Name, m2.Name)
|
||||||
|
if c == 0 {
|
||||||
return m1.Identifier < m2.Identifier
|
return m1.Identifier < m2.Identifier
|
||||||
}
|
}
|
||||||
return m1.Name < m2.Name
|
return c < 0
|
||||||
}
|
}
|
||||||
|
|
||||||
if m2.Weight == 0 {
|
if m2.Weight == 0 {
|
||||||
|
@ -205,7 +207,7 @@ func (m Menu) ByWeight() Menu {
|
||||||
// ByName sorts the menu by the name defined in the menu configuration.
|
// ByName sorts the menu by the name defined in the menu configuration.
|
||||||
func (m Menu) ByName() Menu {
|
func (m Menu) ByName() Menu {
|
||||||
title := func(m1, m2 *MenuEntry) bool {
|
title := func(m1, m2 *MenuEntry) bool {
|
||||||
return m1.Name < m2.Name
|
return compare.LessStrings(m1.Name, m2.Name)
|
||||||
}
|
}
|
||||||
|
|
||||||
menuEntryBy(title).Sort(m)
|
menuEntryBy(title).Sort(m)
|
||||||
|
|
|
@ -52,7 +52,7 @@ func (s mapKeyByInt) Less(i, j int) bool { return s.mapKeyValues[i].Int() < s.ma
|
||||||
type mapKeyByStr struct{ mapKeyValues }
|
type mapKeyByStr struct{ mapKeyValues }
|
||||||
|
|
||||||
func (s mapKeyByStr) Less(i, j int) bool {
|
func (s mapKeyByStr) Less(i, j int) bool {
|
||||||
return s.mapKeyValues[i].String() < s.mapKeyValues[j].String()
|
return compare.LessStrings(s.mapKeyValues[i].String(), s.mapKeyValues[j].String())
|
||||||
}
|
}
|
||||||
|
|
||||||
func sortKeys(v []reflect.Value, order string) []reflect.Value {
|
func sortKeys(v []reflect.Value, order string) []reflect.Value {
|
||||||
|
|
|
@ -18,6 +18,7 @@ import (
|
||||||
|
|
||||||
"github.com/gohugoio/hugo/resources/resource"
|
"github.com/gohugoio/hugo/resources/resource"
|
||||||
|
|
||||||
|
"github.com/gohugoio/hugo/compare"
|
||||||
"github.com/spf13/cast"
|
"github.com/spf13/cast"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -50,13 +51,14 @@ func (by pageBy) Sort(pages Pages) {
|
||||||
var DefaultPageSort = func(p1, p2 Page) bool {
|
var DefaultPageSort = func(p1, p2 Page) bool {
|
||||||
if p1.Weight() == p2.Weight() {
|
if p1.Weight() == p2.Weight() {
|
||||||
if p1.Date().Unix() == p2.Date().Unix() {
|
if p1.Date().Unix() == p2.Date().Unix() {
|
||||||
if p1.LinkTitle() == p2.LinkTitle() {
|
c := compare.Strings(p1.LinkTitle(), p2.LinkTitle())
|
||||||
|
if c == 0 {
|
||||||
if p1.File().IsZero() || p2.File().IsZero() {
|
if p1.File().IsZero() || p2.File().IsZero() {
|
||||||
return p1.File().IsZero()
|
return p1.File().IsZero()
|
||||||
}
|
}
|
||||||
return p1.File().Filename() < p2.File().Filename()
|
return compare.LessStrings(p1.File().Filename(), p2.File().Filename())
|
||||||
}
|
}
|
||||||
return (p1.LinkTitle() < p2.LinkTitle())
|
return c < 0
|
||||||
}
|
}
|
||||||
return p1.Date().Unix() > p2.Date().Unix()
|
return p1.Date().Unix() > p2.Date().Unix()
|
||||||
}
|
}
|
||||||
|
@ -76,12 +78,13 @@ var languagePageSort = func(p1, p2 Page) bool {
|
||||||
|
|
||||||
if p1.Language().Weight == p2.Language().Weight {
|
if p1.Language().Weight == p2.Language().Weight {
|
||||||
if p1.Date().Unix() == p2.Date().Unix() {
|
if p1.Date().Unix() == p2.Date().Unix() {
|
||||||
if p1.LinkTitle() == p2.LinkTitle() {
|
c := compare.Strings(p1.LinkTitle(), p2.LinkTitle())
|
||||||
|
if c == 0 {
|
||||||
if !p1.File().IsZero() && !p2.File().IsZero() {
|
if !p1.File().IsZero() && !p2.File().IsZero() {
|
||||||
return p1.File().Filename() < p2.File().Filename()
|
return compare.LessStrings(p1.File().Filename(), p2.File().Filename())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return (p1.LinkTitle() < p2.LinkTitle())
|
return c < 0
|
||||||
}
|
}
|
||||||
return p1.Date().Unix() > p2.Date().Unix()
|
return p1.Date().Unix() > p2.Date().Unix()
|
||||||
}
|
}
|
||||||
|
@ -137,7 +140,7 @@ func (p Pages) ByTitle() Pages {
|
||||||
const key = "pageSort.ByTitle"
|
const key = "pageSort.ByTitle"
|
||||||
|
|
||||||
title := func(p1, p2 Page) bool {
|
title := func(p1, p2 Page) bool {
|
||||||
return p1.Title() < p2.Title()
|
return compare.LessStrings(p1.Title(), p2.Title())
|
||||||
}
|
}
|
||||||
|
|
||||||
pages, _ := spc.get(key, pageBy(title).Sort, p)
|
pages, _ := spc.get(key, pageBy(title).Sort, p)
|
||||||
|
@ -154,7 +157,7 @@ func (p Pages) ByLinkTitle() Pages {
|
||||||
const key = "pageSort.ByLinkTitle"
|
const key = "pageSort.ByLinkTitle"
|
||||||
|
|
||||||
linkTitle := func(p1, p2 Page) bool {
|
linkTitle := func(p1, p2 Page) bool {
|
||||||
return p1.LinkTitle() < p2.LinkTitle()
|
return compare.LessStrings(p1.LinkTitle(), p2.LinkTitle())
|
||||||
}
|
}
|
||||||
|
|
||||||
pages, _ := spc.get(key, pageBy(linkTitle).Sort, p)
|
pages, _ := spc.get(key, pageBy(linkTitle).Sort, p)
|
||||||
|
@ -339,7 +342,8 @@ func (p Pages) ByParam(paramsKey interface{}) Pages {
|
||||||
s1 := cast.ToString(v1)
|
s1 := cast.ToString(v1)
|
||||||
s2 := cast.ToString(v2)
|
s2 := cast.ToString(v2)
|
||||||
|
|
||||||
return s1 < s2
|
return compare.LessStrings(s1, s2)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pages, _ := spc.get(key, pageBy(paramsKeyComparator).Sort, p)
|
pages, _ := spc.get(key, pageBy(paramsKeyComparator).Sort, p)
|
||||||
|
|
|
@ -23,7 +23,7 @@ import (
|
||||||
"github.com/spf13/cast"
|
"github.com/spf13/cast"
|
||||||
)
|
)
|
||||||
|
|
||||||
var comp = compare.New()
|
var sortComp = compare.New(true)
|
||||||
|
|
||||||
// Sort returns a sorted sequence.
|
// Sort returns a sorted sequence.
|
||||||
func (ns *Namespace) Sort(seq interface{}, args ...interface{}) (interface{}, error) {
|
func (ns *Namespace) Sort(seq interface{}, args ...interface{}) (interface{}, error) {
|
||||||
|
@ -133,15 +133,15 @@ func (p pairList) Less(i, j int) bool {
|
||||||
if iv.IsValid() {
|
if iv.IsValid() {
|
||||||
if jv.IsValid() {
|
if jv.IsValid() {
|
||||||
// can only call Interface() on valid reflect Values
|
// can only call Interface() on valid reflect Values
|
||||||
return comp.Lt(iv.Interface(), jv.Interface())
|
return sortComp.Lt(iv.Interface(), jv.Interface())
|
||||||
}
|
}
|
||||||
// if j is invalid, test i against i's zero value
|
// if j is invalid, test i against i's zero value
|
||||||
return comp.Lt(iv.Interface(), reflect.Zero(iv.Type()))
|
return sortComp.Lt(iv.Interface(), reflect.Zero(iv.Type()))
|
||||||
}
|
}
|
||||||
|
|
||||||
if jv.IsValid() {
|
if jv.IsValid() {
|
||||||
// if i is invalid, test j against j's zero value
|
// if i is invalid, test j against j's zero value
|
||||||
return comp.Lt(reflect.Zero(jv.Type()), jv.Interface())
|
return sortComp.Lt(reflect.Zero(jv.Type()), jv.Interface())
|
||||||
}
|
}
|
||||||
|
|
||||||
return false
|
return false
|
||||||
|
|
|
@ -45,6 +45,7 @@ func TestSort(t *testing.T) {
|
||||||
}{
|
}{
|
||||||
{[]string{"class1", "class2", "class3"}, nil, "asc", []string{"class1", "class2", "class3"}},
|
{[]string{"class1", "class2", "class3"}, nil, "asc", []string{"class1", "class2", "class3"}},
|
||||||
{[]string{"class3", "class1", "class2"}, nil, "asc", []string{"class1", "class2", "class3"}},
|
{[]string{"class3", "class1", "class2"}, nil, "asc", []string{"class1", "class2", "class3"}},
|
||||||
|
{[]string{"CLASS3", "class1", "class2"}, nil, "asc", []string{"class1", "class2", "CLASS3"}},
|
||||||
// Issue 6023
|
// Issue 6023
|
||||||
{stringsSlice{"class3", "class1", "class2"}, nil, "asc", stringsSlice{"class1", "class2", "class3"}},
|
{stringsSlice{"class3", "class1", "class2"}, nil, "asc", stringsSlice{"class1", "class2", "class3"}},
|
||||||
|
|
||||||
|
|
|
@ -20,18 +20,20 @@ import (
|
||||||
"strconv"
|
"strconv"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/gohugoio/hugo/common/types"
|
|
||||||
|
|
||||||
"github.com/gohugoio/hugo/compare"
|
"github.com/gohugoio/hugo/compare"
|
||||||
|
|
||||||
|
"github.com/gohugoio/hugo/common/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
// New returns a new instance of the compare-namespaced template functions.
|
// New returns a new instance of the compare-namespaced template functions.
|
||||||
func New() *Namespace {
|
func New(caseInsensitive bool) *Namespace {
|
||||||
return &Namespace{}
|
return &Namespace{caseInsensitive: caseInsensitive}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Namespace provides template functions for the "compare" namespace.
|
// Namespace provides template functions for the "compare" namespace.
|
||||||
type Namespace struct {
|
type Namespace struct {
|
||||||
|
// Enable to do case insensitive string compares.
|
||||||
|
caseInsensitive bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Default checks whether a given value is set and returns a default value if it
|
// Default checks whether a given value is set and returns a default value if it
|
||||||
|
@ -89,7 +91,10 @@ func (*Namespace) Default(dflt interface{}, given ...interface{}) (interface{},
|
||||||
}
|
}
|
||||||
|
|
||||||
// Eq returns the boolean truth of arg1 == arg2.
|
// Eq returns the boolean truth of arg1 == arg2.
|
||||||
func (*Namespace) Eq(x, y interface{}) bool {
|
func (ns *Namespace) Eq(x, y interface{}) bool {
|
||||||
|
if ns.caseInsensitive {
|
||||||
|
panic("caseInsensitive not implemented for Eq")
|
||||||
|
}
|
||||||
if e, ok := x.(compare.Eqer); ok {
|
if e, ok := x.(compare.Eqer); ok {
|
||||||
return e.Eq(y)
|
return e.Eq(y)
|
||||||
}
|
}
|
||||||
|
@ -157,7 +162,7 @@ func (n *Namespace) Conditional(condition bool, a, b interface{}) interface{} {
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (*Namespace) compareGet(a interface{}, b interface{}) (float64, float64) {
|
func (ns *Namespace) compareGet(a interface{}, b interface{}) (float64, float64) {
|
||||||
if ac, ok := a.(compare.Comparer); ok {
|
if ac, ok := a.(compare.Comparer); ok {
|
||||||
c := ac.Compare(b)
|
c := ac.Compare(b)
|
||||||
if c < 0 {
|
if c < 0 {
|
||||||
|
@ -228,6 +233,17 @@ func (*Namespace) compareGet(a interface{}, b interface{}) (float64, float64) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ns.caseInsensitive && leftStr != nil && rightStr != nil {
|
||||||
|
c := compare.Strings(*leftStr, *rightStr)
|
||||||
|
if c < 0 {
|
||||||
|
return 0, 1
|
||||||
|
} else if c > 0 {
|
||||||
|
return 1, 0
|
||||||
|
} else {
|
||||||
|
return 0, 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
switch {
|
switch {
|
||||||
case leftStr == nil || rightStr == nil:
|
case leftStr == nil || rightStr == nil:
|
||||||
case *leftStr < *rightStr:
|
case *leftStr < *rightStr:
|
||||||
|
|
|
@ -83,7 +83,7 @@ func TestDefaultFunc(t *testing.T) {
|
||||||
|
|
||||||
then := time.Now()
|
then := time.Now()
|
||||||
now := time.Now()
|
now := time.Now()
|
||||||
ns := New()
|
ns := New(false)
|
||||||
|
|
||||||
for i, test := range []struct {
|
for i, test := range []struct {
|
||||||
dflt interface{}
|
dflt interface{}
|
||||||
|
@ -139,7 +139,7 @@ func TestDefaultFunc(t *testing.T) {
|
||||||
func TestCompare(t *testing.T) {
|
func TestCompare(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
n := New()
|
n := New(false)
|
||||||
|
|
||||||
for _, test := range []struct {
|
for _, test := range []struct {
|
||||||
tstCompareType
|
tstCompareType
|
||||||
|
@ -233,6 +233,14 @@ func doTestCompare(t *testing.T, tp tstCompareType, funcUnderTest func(a, b inte
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestCase(t *testing.T) {
|
||||||
|
assert := require.New(t)
|
||||||
|
n := New(true)
|
||||||
|
|
||||||
|
assert.True(n.Lt("az", "Za"))
|
||||||
|
assert.True(n.Gt("ab", "Ab"))
|
||||||
|
}
|
||||||
|
|
||||||
func TestTimeUnix(t *testing.T) {
|
func TestTimeUnix(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
var sec int64 = 1234567890
|
var sec int64 = 1234567890
|
||||||
|
@ -258,7 +266,7 @@ func TestTimeUnix(t *testing.T) {
|
||||||
|
|
||||||
func TestConditional(t *testing.T) {
|
func TestConditional(t *testing.T) {
|
||||||
assert := require.New(t)
|
assert := require.New(t)
|
||||||
n := New()
|
n := New(false)
|
||||||
a, b := "a", "b"
|
a, b := "a", "b"
|
||||||
|
|
||||||
assert.Equal(a, n.Conditional(true, a, b))
|
assert.Equal(a, n.Conditional(true, a, b))
|
||||||
|
|
|
@ -22,7 +22,7 @@ const name = "compare"
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
f := func(d *deps.Deps) *internal.TemplateFuncsNamespace {
|
f := func(d *deps.Deps) *internal.TemplateFuncsNamespace {
|
||||||
ctx := New()
|
ctx := New(false)
|
||||||
|
|
||||||
ns := &internal.TemplateFuncsNamespace{
|
ns := &internal.TemplateFuncsNamespace{
|
||||||
Name: name,
|
Name: name,
|
||||||
|
|
|
@ -23,7 +23,7 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestTruth(t *testing.T) {
|
func TestTruth(t *testing.T) {
|
||||||
n := New()
|
n := New(false)
|
||||||
|
|
||||||
truthv, falsev := reflect.ValueOf(time.Now()), reflect.ValueOf(false)
|
truthv, falsev := reflect.ValueOf(time.Now()), reflect.ValueOf(false)
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue