tpl: Add math.Sqrt
authorChris Dennis <ssa@fbcs.co.uk>
Mon, 24 Feb 2020 22:45:04 +0000 (22:45 +0000)
committerGitHub <noreply@github.com>
Mon, 24 Feb 2020 22:45:04 +0000 (23:45 +0100)
Fixes #6941

docs/content/en/functions/math.md
tpl/math/init.go
tpl/math/math.go
tpl/math/math_test.go

index eb38fdd0b5406186a786856e501da0b70d929a54..29fe16985ebc57ba424e27958fdb137e79fff64e 100644 (file)
@@ -4,7 +4,7 @@ description: Hugo provides nine mathematical operators in templates.
 godocref:
 date: 2017-02-01
 publishdate: 2017-02-01
-lastmod: 2017-02-01
+lastmod: 2020-02-23
 keywords: [math, operators]
 categories: [functions]
 menu:
@@ -36,3 +36,6 @@ aliases: []
 | `math.Ceil`  | Returns the least integer value greater than or equal to the given number.  | `{{math.Ceil 2.1}}` &rarr; `3`   |
 | `math.Floor` | Returns the greatest integer value less than or equal to the given number.  | `{{math.Floor 1.9}}` &rarr; `1`  |
 | `math.Round` | Returns the nearest integer, rounding half away from zero.                  | `{{math.Round 1.5}}` &rarr; `2`  |
+| `math.Log`   | Returns the natural logarithm of the given number.                          | `{{math.Log 42}}` &rarr; `3.737` |
+| `math.Sqrt`  | Returns the square root of the given number.                                | `{{math.Sqrt 81}}` &rarr; `9`    |
+
index bbffb23aafd4fcf1fe5d8cff4d0a8c040ff429e8..45240c5e1aca4dfae3bceffb22969d4f350257b5 100644 (file)
@@ -64,6 +64,13 @@ func init() {
                        },
                )
 
+               ns.AddMethodMapping(ctx.Sqrt,
+                       nil,
+                       [][2]string{
+                               {"{{math.Sqrt 81}}", "9"},
+                       },
+               )
+
                ns.AddMethodMapping(ctx.Mod,
                        []string{"mod"},
                        [][2]string{
index 08be42b47d8d544519ede8a56e1eaf00d854dade..950d95905dd3095bc13d68835bc223f95ac273d3 100644 (file)
@@ -72,6 +72,18 @@ func (ns *Namespace) Log(a interface{}) (float64, error) {
        return math.Log(af), nil
 }
 
+// Sqrt returns the square root of a number.
+// NOTE: will return for NaN for negative values of a
+func (ns *Namespace) Sqrt(a interface{}) (float64, error) {
+       af, err := cast.ToFloat64E(a)
+
+       if err != nil {
+               return 0, errors.New("Sqrt operator can't be used with non integer or float value")
+       }
+
+       return math.Sqrt(af), nil
+}
+
 // Mod returns a % b.
 func (ns *Namespace) Mod(a, b interface{}) (int64, error) {
        ai, erra := cast.ToInt64E(a)
index 0beec8204e7bb64db095ba04db3b3ba46bdc7dc6..70f6749baff865cbbef74db656020bdf672c2568 100644 (file)
@@ -153,6 +153,51 @@ func TestLog(t *testing.T) {
                c.Assert(err, qt.IsNil)
                c.Assert(result, qt.Equals, test.expect)
        }
+
+       // Separate test for Log(-1) -- returns NaN
+       result, err := ns.Log(-1)
+       c.Assert(err, qt.IsNil)
+       c.Assert(result, qt.Satisfies, math.IsNaN)
+}
+
+func TestSqrt(t *testing.T) {
+       t.Parallel()
+       c := qt.New(t)
+
+       ns := New()
+
+       for _, test := range []struct {
+               a      interface{}
+               expect interface{}
+       }{
+               {81, float64(9)},
+               {0.25, float64(0.5)},
+               {0, float64(0)},
+               {"abc", false},
+       } {
+
+               result, err := ns.Sqrt(test.a)
+
+               if b, ok := test.expect.(bool); ok && !b {
+                       c.Assert(err, qt.Not(qt.IsNil))
+                       continue
+               }
+
+               // we compare only 4 digits behind point if its a real float
+               // otherwise we usually get different float values on the last positions
+               if result != math.Inf(-1) {
+                       result = float64(int(result*10000)) / 10000
+               }
+
+               c.Assert(err, qt.IsNil)
+               c.Assert(result, qt.Equals, test.expect)
+       }
+
+       // Separate test for Sqrt(-1) -- returns NaN
+       result, err := ns.Sqrt(-1)
+       c.Assert(err, qt.IsNil)
+       c.Assert(result, qt.Satisfies, math.IsNaN)
+
 }
 
 func TestMod(t *testing.T) {