gitea源码

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. // Copyright 2023 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package eval
  4. import (
  5. "math"
  6. "strings"
  7. "testing"
  8. "github.com/stretchr/testify/assert"
  9. )
  10. func tokens(s string) (a []any) {
  11. for v := range strings.FieldsSeq(s) {
  12. a = append(a, v)
  13. }
  14. return a
  15. }
  16. func TestEval(t *testing.T) {
  17. n, err := Expr(0, "/", 0.0)
  18. assert.NoError(t, err)
  19. assert.True(t, math.IsNaN(n.Value.(float64)))
  20. _, err = Expr(nil)
  21. assert.ErrorContains(t, err, "unsupported token type")
  22. _, err = Expr([]string{})
  23. assert.ErrorContains(t, err, "unsupported token type")
  24. _, err = Expr(struct{}{})
  25. assert.ErrorContains(t, err, "unsupported token type")
  26. cases := []struct {
  27. expr string
  28. want any
  29. }{
  30. {"-1", int64(-1)},
  31. {"1 + 2", int64(3)},
  32. {"3 - 2 + 4", int64(5)},
  33. {"1 + 2 * 3", int64(7)},
  34. {"1 + ( 2 * 3 )", int64(7)},
  35. {"( 1 + 2 ) * 3", int64(9)},
  36. {"( 1 + 2.0 ) / 3", float64(1)},
  37. {"sum( 1 , 2 , 3 , 4 )", int64(10)},
  38. {"100 + sum( 1 , 2 + 3 , 0.0 ) / 2", float64(103)},
  39. {"100 * 5 / ( 5 + 15 )", int64(25)},
  40. {"9 == 5", int64(0)},
  41. {"5 == 5", int64(1)},
  42. {"9 != 5", int64(1)},
  43. {"5 != 5", int64(0)},
  44. {"9 > 5", int64(1)},
  45. {"5 > 9", int64(0)},
  46. {"5 >= 9", int64(0)},
  47. {"9 >= 9", int64(1)},
  48. {"9 < 5", int64(0)},
  49. {"5 < 9", int64(1)},
  50. {"9 <= 5", int64(0)},
  51. {"5 <= 5", int64(1)},
  52. {"1 and 2", int64(1)}, // Golang template definition: non-zero values are all truth
  53. {"1 and 0", int64(0)},
  54. {"0 and 0", int64(0)},
  55. {"1 or 2", int64(1)},
  56. {"1 or 0", int64(1)},
  57. {"0 or 1", int64(1)},
  58. {"0 or 0", int64(0)},
  59. {"not 2 == 1", int64(1)},
  60. {"not not ( 9 < 5 )", int64(0)},
  61. }
  62. for _, c := range cases {
  63. n, err := Expr(tokens(c.expr)...)
  64. if assert.NoError(t, err, "expr: %s", c.expr) {
  65. assert.Equal(t, c.want, n.Value)
  66. }
  67. }
  68. bads := []struct {
  69. expr string
  70. errMsg string
  71. }{
  72. {"0 / 0", "integer divide by zero"},
  73. {"1 +", "num stack is empty"},
  74. {"+ 1", "num stack is empty"},
  75. {"( 1", "incomplete sub-expression"},
  76. {"1 )", "op stack is empty"}, // can not find the corresponding open bracket after the stack becomes empty
  77. {"1 , 2", "expect 1 value as final result"},
  78. {"( 1 , 2 )", "too many values in one bracket"},
  79. {"1 a 2", "unknown operator"},
  80. }
  81. for _, c := range bads {
  82. _, err = Expr(tokens(c.expr)...)
  83. assert.ErrorContains(t, err, c.errMsg, "expr: %s", c.expr)
  84. }
  85. }