number.go 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. package httpexpect
  2. import (
  3. "math"
  4. )
  5. // Number provides methods to inspect attached float64 value
  6. // (Go representation of JSON number).
  7. type Number struct {
  8. chain chain
  9. value float64
  10. }
  11. // NewNumber returns a new Number given a reporter used to report
  12. // failures and value to be inspected.
  13. //
  14. // reporter should not be nil.
  15. //
  16. // Example:
  17. // number := NewNumber(t, 123.4)
  18. func NewNumber(reporter Reporter, value float64) *Number {
  19. return &Number{makeChain(reporter), value}
  20. }
  21. // Raw returns underlying value attached to Number.
  22. // This is the value originally passed to NewNumber.
  23. //
  24. // Example:
  25. // number := NewNumber(t, 123.4)
  26. // assert.Equal(t, 123.4, number.Raw())
  27. func (n *Number) Raw() float64 {
  28. return n.value
  29. }
  30. // Path is similar to Value.Path.
  31. func (n *Number) Path(path string) *Value {
  32. return getPath(&n.chain, n.value, path)
  33. }
  34. // Schema is similar to Value.Schema.
  35. func (n *Number) Schema(schema interface{}) *Number {
  36. checkSchema(&n.chain, n.value, schema)
  37. return n
  38. }
  39. // Equal succeeds if number is equal to given value.
  40. //
  41. // value should have numeric type convertible to float64. Before comparison,
  42. // it is converted to float64.
  43. //
  44. // Example:
  45. // number := NewNumber(t, 123)
  46. // number.Equal(float64(123))
  47. // number.Equal(int32(123))
  48. func (n *Number) Equal(value interface{}) *Number {
  49. v, ok := canonNumber(&n.chain, value)
  50. if !ok {
  51. return n
  52. }
  53. if !(n.value == v) {
  54. n.chain.fail("\nexpected number equal to:\n %v\n\nbut got:\n %v",
  55. v, n.value)
  56. }
  57. return n
  58. }
  59. // NotEqual succeeds if number is not equal to given value.
  60. //
  61. // value should have numeric type convertible to float64. Before comparison,
  62. // it is converted to float64.
  63. //
  64. // Example:
  65. // number := NewNumber(t, 123)
  66. // number.NotEqual(float64(321))
  67. // number.NotEqual(int32(321))
  68. func (n *Number) NotEqual(value interface{}) *Number {
  69. v, ok := canonNumber(&n.chain, value)
  70. if !ok {
  71. return n
  72. }
  73. if !(n.value != v) {
  74. n.chain.fail("\nexpected number not equal to:\n %v\n\nbut got:\n %v",
  75. v, n.value)
  76. }
  77. return n
  78. }
  79. // EqualDelta succeeds if two numerals are within delta of each other.
  80. //
  81. // Example:
  82. // number := NewNumber(t, 123.0)
  83. // number.EqualDelta(123.2, 0.3)
  84. func (n *Number) EqualDelta(value, delta float64) *Number {
  85. if math.IsNaN(n.value) || math.IsNaN(value) || math.IsNaN(delta) {
  86. n.chain.fail("\nexpected number equal to:\n %v\n\nbut got:\n %v\n\ndelta:\n %v",
  87. value, n.value, delta)
  88. return n
  89. }
  90. diff := (n.value - value)
  91. if diff < -delta || diff > delta {
  92. n.chain.fail("\nexpected number equal to:\n %v\n\nbut got:\n %v\n\ndelta:\n %v",
  93. value, n.value, delta)
  94. return n
  95. }
  96. return n
  97. }
  98. // NotEqualDelta succeeds if two numerals are not within delta of each other.
  99. //
  100. // Example:
  101. // number := NewNumber(t, 123.0)
  102. // number.NotEqualDelta(123.2, 0.1)
  103. func (n *Number) NotEqualDelta(value, delta float64) *Number {
  104. if math.IsNaN(n.value) || math.IsNaN(value) || math.IsNaN(delta) {
  105. n.chain.fail(
  106. "\nexpected number not equal to:\n %v\n\nbut got:\n %v\n\ndelta:\n %v",
  107. value, n.value, delta)
  108. return n
  109. }
  110. diff := (n.value - value)
  111. if !(diff < -delta || diff > delta) {
  112. n.chain.fail(
  113. "\nexpected number not equal to:\n %v\n\nbut got:\n %v\n\ndelta:\n %v",
  114. value, n.value, delta)
  115. return n
  116. }
  117. return n
  118. }
  119. // Gt succeeds if number is greater than given value.
  120. //
  121. // value should have numeric type convertible to float64. Before comparison,
  122. // it is converted to float64.
  123. //
  124. // Example:
  125. // number := NewNumber(t, 123)
  126. // number.Gt(float64(122))
  127. // number.Gt(int32(122))
  128. func (n *Number) Gt(value interface{}) *Number {
  129. v, ok := canonNumber(&n.chain, value)
  130. if !ok {
  131. return n
  132. }
  133. if !(n.value > v) {
  134. n.chain.fail("\nexpected number > then:\n %v\n\nbut got:\n %v",
  135. v, n.value)
  136. }
  137. return n
  138. }
  139. // Ge succeeds if number is greater than or equal to given value.
  140. //
  141. // value should have numeric type convertible to float64. Before comparison,
  142. // it is converted to float64.
  143. //
  144. // Example:
  145. // number := NewNumber(t, 123)
  146. // number.Ge(float64(122))
  147. // number.Ge(int32(122))
  148. func (n *Number) Ge(value interface{}) *Number {
  149. v, ok := canonNumber(&n.chain, value)
  150. if !ok {
  151. return n
  152. }
  153. if !(n.value >= v) {
  154. n.chain.fail("\nexpected number >= then:\n %v\n\nbut got:\n %v",
  155. v, n.value)
  156. }
  157. return n
  158. }
  159. // Lt succeeds if number is lesser than given value.
  160. //
  161. // value should have numeric type convertible to float64. Before comparison,
  162. // it is converted to float64.
  163. //
  164. // Example:
  165. // number := NewNumber(t, 123)
  166. // number.Lt(float64(124))
  167. // number.Lt(int32(124))
  168. func (n *Number) Lt(value interface{}) *Number {
  169. v, ok := canonNumber(&n.chain, value)
  170. if !ok {
  171. return n
  172. }
  173. if !(n.value < v) {
  174. n.chain.fail("\nexpected number < then:\n %v\n\nbut got:\n %v",
  175. v, n.value)
  176. }
  177. return n
  178. }
  179. // Le succeeds if number is lesser than or equal to given value.
  180. //
  181. // value should have numeric type convertible to float64. Before comparison,
  182. // it is converted to float64.
  183. //
  184. // Example:
  185. // number := NewNumber(t, 123)
  186. // number.Le(float64(124))
  187. // number.Le(int32(124))
  188. func (n *Number) Le(value interface{}) *Number {
  189. v, ok := canonNumber(&n.chain, value)
  190. if !ok {
  191. return n
  192. }
  193. if !(n.value <= v) {
  194. n.chain.fail("\nexpected number <= then:\n %v\n\nbut got:\n %v",
  195. v, n.value)
  196. }
  197. return n
  198. }
  199. // InRange succeeds if number is in given range [min; max].
  200. //
  201. // min and max should have numeric type convertible to float64. Before comparison,
  202. // they are converted to float64.
  203. //
  204. // Example:
  205. // number := NewNumber(t, 123)
  206. // number.InRange(float32(100), int32(200)) // success
  207. // number.InRange(100, 200) // success
  208. // number.InRange(123, 123) // success
  209. func (n *Number) InRange(min, max interface{}) *Number {
  210. a, ok := canonNumber(&n.chain, min)
  211. if !ok {
  212. return n
  213. }
  214. b, ok := canonNumber(&n.chain, max)
  215. if !ok {
  216. return n
  217. }
  218. if !(n.value >= a && n.value <= b) {
  219. n.chain.fail("\nexpected number in range:\n [%v; %v]\n\nbut got:\n %v",
  220. a, b, n.value)
  221. }
  222. return n
  223. }