From 019bd5576be87c9f06b6a928ede1a5e78677f7b3 Mon Sep 17 00:00:00 2001 From: "David E. Wheeler" Date: Mon, 4 Jun 2018 13:47:03 -0400 Subject: [PATCH] tpl/strings: strings.RuneCount --- .../content/en/functions/strings.RuneCount.md | 28 +++++++++++++++++++ docs/data/docs.json | 20 ++++++++++++- tpl/strings/init.go | 5 ++++ tpl/strings/strings.go | 9 ++++++ tpl/strings/strings_test.go | 27 ++++++++++++++++++ 5 files changed, 88 insertions(+), 1 deletion(-) create mode 100644 docs/content/en/functions/strings.RuneCount.md diff --git a/docs/content/en/functions/strings.RuneCount.md b/docs/content/en/functions/strings.RuneCount.md new file mode 100644 index 00000000..63012ab3 --- /dev/null +++ b/docs/content/en/functions/strings.RuneCount.md @@ -0,0 +1,28 @@ +--- +title: strings.RuneCount +description: Determines the number of runes in a string. +godocref: +date: 2018-06-01 +publishdate: 2018-06-01 +lastmod: 2018-06-01 +categories: [functions] +menu: + docs: + parent: "functions" +keywords: [counting, character count, length, rune length, rune count] +signature: ["strings.RuneCount INPUT"] +workson: [] +hugoversion: +relatedfuncs: ["len", "countrunes"] +deprecated: false +aliases: [] +--- + +In contrast with `strings.CountRunes` function, which strips HTML and whitespace before counting runes, `strings.RuneCount` simply counts all the runes in a string. It relies on the Go [`utf8.RuneCountInString`] function. + +``` +{{ "Hello, 世界" | strings.RuneCount }} + +``` + +[`utf8.RuneCount`]: https://golang.org/pkg/unicode/utf8/#RuneCount \ No newline at end of file diff --git a/docs/data/docs.json b/docs/data/docs.json index 4eddf8ea..3020a21f 100644 --- a/docs/data/docs.json +++ b/docs/data/docs.json @@ -3146,7 +3146,25 @@ "Aliases": [ "countrunes" ], - "Examples": [] + "Examples": [ + [ + "{{ \"Hello, 世界\" | countrunes }}", + "8" + ] + ] + }, + "RuneCount": { + "Description": "RuneCount returns the number of runes in s", + "Args": [ + "s" + ], + "Aliases": [], + "Examples": [ + [ + "{{ \"Hello, 世界\" | strings.RuneCount }}", + "9" + ] + ] }, "CountWords": { "Description": "CountWords returns the approximate word count in s.", diff --git a/tpl/strings/init.go b/tpl/strings/init.go index 883ce76d..8ff4cf98 100644 --- a/tpl/strings/init.go +++ b/tpl/strings/init.go @@ -41,6 +41,11 @@ func init() { [][2]string{}, ) + ns.AddMethodMapping(ctx.RuneCount, + nil, + [][2]string{}, + ) + ns.AddMethodMapping(ctx.CountWords, []string{"countwords"}, [][2]string{}, diff --git a/tpl/strings/strings.go b/tpl/strings/strings.go index d7d8f2d8..1864bb9e 100644 --- a/tpl/strings/strings.go +++ b/tpl/strings/strings.go @@ -57,6 +57,15 @@ func (ns *Namespace) CountRunes(s interface{}) (int, error) { return counter, nil } +// RuneCount returns the number of runes in s. +func (ns *Namespace) RuneCount(s interface{}) (int, error) { + ss, err := cast.ToStringE(s) + if err != nil { + return 0, fmt.Errorf("Failed to convert content to string: %s", err) + } + return utf8.RuneCountInString(ss), nil +} + // CountWords returns the approximate word count in s. func (ns *Namespace) CountWords(s interface{}) (int, error) { ss, err := cast.ToStringE(s) diff --git a/tpl/strings/strings_test.go b/tpl/strings/strings_test.go index 69863c30..22695ba0 100644 --- a/tpl/strings/strings_test.go +++ b/tpl/strings/strings_test.go @@ -173,6 +173,33 @@ func TestCountRunes(t *testing.T) { } } +func TestRuneCount(t *testing.T) { + t.Parallel() + + for i, test := range []struct { + s interface{} + expect interface{} + }{ + {"foo bar", 7}, + {"旁边", 2}, + {`
旁边
`, 26}, + // errors + {tstNoStringer{}, false}, + } { + errMsg := fmt.Sprintf("[%d] %v", i, test.s) + + result, err := ns.RuneCount(test.s) + + if b, ok := test.expect.(bool); ok && !b { + require.Error(t, err, errMsg) + continue + } + + require.NoError(t, err, errMsg) + assert.Equal(t, test.expect, result, errMsg) + } +} + func TestCountWords(t *testing.T) { t.Parallel() -- 2.30.2