dec_test.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411
  1. package inf
  2. import (
  3. "bytes"
  4. "encoding/gob"
  5. "fmt"
  6. "math/big"
  7. "strings"
  8. "testing"
  9. )
  10. type decFunZZ func(z, x, y *Dec) *Dec
  11. type decArgZZ struct {
  12. z, x, y *Dec
  13. }
  14. var decSumZZ = []decArgZZ{
  15. {NewDec(0, 0), NewDec(0, 0), NewDec(0, 0)},
  16. {NewDec(1, 0), NewDec(1, 0), NewDec(0, 0)},
  17. {NewDec(1111111110, 0), NewDec(123456789, 0), NewDec(987654321, 0)},
  18. {NewDec(-1, 0), NewDec(-1, 0), NewDec(0, 0)},
  19. {NewDec(864197532, 0), NewDec(-123456789, 0), NewDec(987654321, 0)},
  20. {NewDec(-1111111110, 0), NewDec(-123456789, 0), NewDec(-987654321, 0)},
  21. {NewDec(12, 2), NewDec(1, 1), NewDec(2, 2)},
  22. }
  23. var decProdZZ = []decArgZZ{
  24. {NewDec(0, 0), NewDec(0, 0), NewDec(0, 0)},
  25. {NewDec(0, 0), NewDec(1, 0), NewDec(0, 0)},
  26. {NewDec(1, 0), NewDec(1, 0), NewDec(1, 0)},
  27. {NewDec(-991*991, 0), NewDec(991, 0), NewDec(-991, 0)},
  28. {NewDec(2, 3), NewDec(1, 1), NewDec(2, 2)},
  29. {NewDec(2, -3), NewDec(1, -1), NewDec(2, -2)},
  30. {NewDec(2, 3), NewDec(1, 1), NewDec(2, 2)},
  31. }
  32. func TestDecSignZ(t *testing.T) {
  33. var zero Dec
  34. for _, a := range decSumZZ {
  35. s := a.z.Sign()
  36. e := a.z.Cmp(&zero)
  37. if s != e {
  38. t.Errorf("got %d; want %d for z = %v", s, e, a.z)
  39. }
  40. }
  41. }
  42. func TestDecAbsZ(t *testing.T) {
  43. var zero Dec
  44. for _, a := range decSumZZ {
  45. var z Dec
  46. z.Abs(a.z)
  47. var e Dec
  48. e.Set(a.z)
  49. if e.Cmp(&zero) < 0 {
  50. e.Sub(&zero, &e)
  51. }
  52. if z.Cmp(&e) != 0 {
  53. t.Errorf("got z = %v; want %v", z, e)
  54. }
  55. }
  56. }
  57. func testDecFunZZ(t *testing.T, msg string, f decFunZZ, a decArgZZ) {
  58. var z Dec
  59. f(&z, a.x, a.y)
  60. if (&z).Cmp(a.z) != 0 {
  61. t.Errorf("%s%+v\n\tgot z = %v; want %v", msg, a, &z, a.z)
  62. }
  63. }
  64. func TestDecSumZZ(t *testing.T) {
  65. AddZZ := func(z, x, y *Dec) *Dec { return z.Add(x, y) }
  66. SubZZ := func(z, x, y *Dec) *Dec { return z.Sub(x, y) }
  67. for _, a := range decSumZZ {
  68. arg := a
  69. testDecFunZZ(t, "AddZZ", AddZZ, arg)
  70. arg = decArgZZ{a.z, a.y, a.x}
  71. testDecFunZZ(t, "AddZZ symmetric", AddZZ, arg)
  72. arg = decArgZZ{a.x, a.z, a.y}
  73. testDecFunZZ(t, "SubZZ", SubZZ, arg)
  74. arg = decArgZZ{a.y, a.z, a.x}
  75. testDecFunZZ(t, "SubZZ symmetric", SubZZ, arg)
  76. }
  77. }
  78. func TestDecProdZZ(t *testing.T) {
  79. MulZZ := func(z, x, y *Dec) *Dec { return z.Mul(x, y) }
  80. for _, a := range decProdZZ {
  81. arg := a
  82. testDecFunZZ(t, "MulZZ", MulZZ, arg)
  83. arg = decArgZZ{a.z, a.y, a.x}
  84. testDecFunZZ(t, "MulZZ symmetric", MulZZ, arg)
  85. }
  86. }
  87. var decQuoRemZZZ = []struct {
  88. z, x, y *Dec
  89. r *big.Rat
  90. srA, srB int
  91. }{
  92. // basic examples
  93. {NewDec(1, 0), NewDec(2, 0), NewDec(2, 0), big.NewRat(0, 1), 0, 1},
  94. {NewDec(15, 1), NewDec(3, 0), NewDec(2, 0), big.NewRat(0, 1), 0, 1},
  95. {NewDec(1, 1), NewDec(1, 0), NewDec(10, 0), big.NewRat(0, 1), 0, 1},
  96. {NewDec(0, 0), NewDec(2, 0), NewDec(3, 0), big.NewRat(2, 3), 1, 1},
  97. {NewDec(0, 0), NewDec(2, 0), NewDec(6, 0), big.NewRat(1, 3), 1, 1},
  98. {NewDec(1, 1), NewDec(2, 0), NewDec(12, 0), big.NewRat(2, 3), 1, 1},
  99. // examples from the Go Language Specification
  100. {NewDec(1, 0), NewDec(5, 0), NewDec(3, 0), big.NewRat(2, 3), 1, 1},
  101. {NewDec(-1, 0), NewDec(-5, 0), NewDec(3, 0), big.NewRat(-2, 3), -1, 1},
  102. {NewDec(-1, 0), NewDec(5, 0), NewDec(-3, 0), big.NewRat(-2, 3), 1, -1},
  103. {NewDec(1, 0), NewDec(-5, 0), NewDec(-3, 0), big.NewRat(2, 3), -1, -1},
  104. }
  105. func TestDecQuoRem(t *testing.T) {
  106. for i, a := range decQuoRemZZZ {
  107. z, rA, rB := new(Dec), new(big.Int), new(big.Int)
  108. s := scaleQuoExact{}.Scale(a.x, a.y)
  109. z.quoRem(a.x, a.y, s, true, rA, rB)
  110. if a.z.Cmp(z) != 0 || a.r.Cmp(new(big.Rat).SetFrac(rA, rB)) != 0 {
  111. t.Errorf("#%d QuoRemZZZ got %v, %v, %v; expected %v, %v", i, z, rA, rB, a.z, a.r)
  112. }
  113. if a.srA != rA.Sign() || a.srB != rB.Sign() {
  114. t.Errorf("#%d QuoRemZZZ wrong signs, got %v, %v; expected %v, %v", i, rA.Sign(), rB.Sign(), a.srA, a.srB)
  115. }
  116. }
  117. }
  118. var decUnscaledTests = []struct {
  119. d *Dec
  120. u int64 // ignored when ok == false
  121. ok bool
  122. }{
  123. {new(Dec), 0, true},
  124. {NewDec(-1<<63, 0), -1 << 63, true},
  125. {NewDec(-(-1<<63 + 1), 0), -(-1<<63 + 1), true},
  126. {new(Dec).Neg(NewDec(-1<<63, 0)), 0, false},
  127. {new(Dec).Sub(NewDec(-1<<63, 0), NewDec(1, 0)), 0, false},
  128. {NewDecBig(new(big.Int).Lsh(big.NewInt(1), 64), 0), 0, false},
  129. }
  130. func TestDecUnscaled(t *testing.T) {
  131. for i, tt := range decUnscaledTests {
  132. u, ok := tt.d.Unscaled()
  133. if ok != tt.ok {
  134. t.Errorf("#%d Unscaled: got %v, expected %v", i, ok, tt.ok)
  135. } else if ok && u != tt.u {
  136. t.Errorf("#%d Unscaled: got %v, expected %v", i, u, tt.u)
  137. }
  138. }
  139. }
  140. var decRoundTests = [...]struct {
  141. in *Dec
  142. s Scale
  143. r Rounder
  144. exp *Dec
  145. }{
  146. {NewDec(123424999999999993, 15), 2, RoundHalfUp, NewDec(12342, 2)},
  147. {NewDec(123425000000000001, 15), 2, RoundHalfUp, NewDec(12343, 2)},
  148. {NewDec(123424999999999993, 15), 15, RoundHalfUp, NewDec(123424999999999993, 15)},
  149. {NewDec(123424999999999993, 15), 16, RoundHalfUp, NewDec(1234249999999999930, 16)},
  150. {NewDecBig(new(big.Int).Lsh(big.NewInt(1), 64), 0), -1, RoundHalfUp, NewDec(1844674407370955162, -1)},
  151. {NewDecBig(new(big.Int).Lsh(big.NewInt(1), 64), 0), -2, RoundHalfUp, NewDec(184467440737095516, -2)},
  152. {NewDecBig(new(big.Int).Lsh(big.NewInt(1), 64), 0), -3, RoundHalfUp, NewDec(18446744073709552, -3)},
  153. {NewDecBig(new(big.Int).Lsh(big.NewInt(1), 64), 0), -4, RoundHalfUp, NewDec(1844674407370955, -4)},
  154. {NewDecBig(new(big.Int).Lsh(big.NewInt(1), 64), 0), -5, RoundHalfUp, NewDec(184467440737096, -5)},
  155. {NewDecBig(new(big.Int).Lsh(big.NewInt(1), 64), 0), -6, RoundHalfUp, NewDec(18446744073710, -6)},
  156. }
  157. func TestDecRound(t *testing.T) {
  158. for i, tt := range decRoundTests {
  159. z := new(Dec).Round(tt.in, tt.s, tt.r)
  160. if tt.exp.Cmp(z) != 0 {
  161. t.Errorf("#%d Round got %v; expected %v", i, z, tt.exp)
  162. }
  163. }
  164. }
  165. var decStringTests = []struct {
  166. in string
  167. out string
  168. val int64
  169. scale Scale // skip SetString if negative
  170. ok bool
  171. scanOk bool
  172. }{
  173. {in: "", ok: false, scanOk: false},
  174. {in: "a", ok: false, scanOk: false},
  175. {in: "z", ok: false, scanOk: false},
  176. {in: "+", ok: false, scanOk: false},
  177. {in: "-", ok: false, scanOk: false},
  178. {in: "g", ok: false, scanOk: false},
  179. {in: ".", ok: false, scanOk: false},
  180. {in: ".-0", ok: false, scanOk: false},
  181. {in: ".+0", ok: false, scanOk: false},
  182. // Scannable but not SetStringable
  183. {"0b", "ignored", 0, 0, false, true},
  184. {"0x", "ignored", 0, 0, false, true},
  185. {"0xg", "ignored", 0, 0, false, true},
  186. {"0.0g", "ignored", 0, 1, false, true},
  187. // examples from godoc for Dec
  188. {"0", "0", 0, 0, true, true},
  189. {"0.00", "0.00", 0, 2, true, true},
  190. {"ignored", "0", 0, -2, true, false},
  191. {"1", "1", 1, 0, true, true},
  192. {"1.00", "1.00", 100, 2, true, true},
  193. {"10", "10", 10, 0, true, true},
  194. {"ignored", "10", 1, -1, true, false},
  195. // other tests
  196. {"+0", "0", 0, 0, true, true},
  197. {"-0", "0", 0, 0, true, true},
  198. {"0.0", "0.0", 0, 1, true, true},
  199. {"0.1", "0.1", 1, 1, true, true},
  200. {"0.", "0", 0, 0, true, true},
  201. {"-10", "-10", -1, -1, true, true},
  202. {"-1", "-1", -1, 0, true, true},
  203. {"-0.1", "-0.1", -1, 1, true, true},
  204. {"-0.01", "-0.01", -1, 2, true, true},
  205. {"+0.", "0", 0, 0, true, true},
  206. {"-0.", "0", 0, 0, true, true},
  207. {".0", "0.0", 0, 1, true, true},
  208. {"+.0", "0.0", 0, 1, true, true},
  209. {"-.0", "0.0", 0, 1, true, true},
  210. {"0.0000000000", "0.0000000000", 0, 10, true, true},
  211. {"0.0000000001", "0.0000000001", 1, 10, true, true},
  212. {"-0.0000000000", "0.0000000000", 0, 10, true, true},
  213. {"-0.0000000001", "-0.0000000001", -1, 10, true, true},
  214. {"-10", "-10", -10, 0, true, true},
  215. {"+10", "10", 10, 0, true, true},
  216. {"00", "0", 0, 0, true, true},
  217. {"023", "23", 23, 0, true, true}, // decimal, not octal
  218. {"-02.3", "-2.3", -23, 1, true, true}, // decimal, not octal
  219. }
  220. func TestDecGetString(t *testing.T) {
  221. z := new(Dec)
  222. for i, test := range decStringTests {
  223. if !test.ok {
  224. continue
  225. }
  226. z.SetUnscaled(test.val)
  227. z.SetScale(test.scale)
  228. s := z.String()
  229. if s != test.out {
  230. t.Errorf("#%da got %s; want %s", i, s, test.out)
  231. }
  232. s = fmt.Sprintf("%d", z)
  233. if s != test.out {
  234. t.Errorf("#%db got %s; want %s", i, s, test.out)
  235. }
  236. }
  237. }
  238. func TestDecSetString(t *testing.T) {
  239. tmp := new(Dec)
  240. for i, test := range decStringTests {
  241. if test.scale < 0 {
  242. // SetString only supports scale >= 0
  243. continue
  244. }
  245. // initialize to a non-zero value so that issues with parsing
  246. // 0 are detected
  247. tmp.Set(NewDec(1234567890, 123))
  248. n1, ok1 := new(Dec).SetString(test.in)
  249. n2, ok2 := tmp.SetString(test.in)
  250. expected := NewDec(test.val, test.scale)
  251. if ok1 != test.ok || ok2 != test.ok {
  252. t.Errorf("#%d (input '%s') ok incorrect (should be %t)", i, test.in, test.ok)
  253. continue
  254. }
  255. if !ok1 {
  256. if n1 != nil {
  257. t.Errorf("#%d (input '%s') n1 != nil", i, test.in)
  258. }
  259. continue
  260. }
  261. if !ok2 {
  262. if n2 != nil {
  263. t.Errorf("#%d (input '%s') n2 != nil", i, test.in)
  264. }
  265. continue
  266. }
  267. if n1.Cmp(expected) != 0 {
  268. t.Errorf("#%d (input '%s') got: %s want: %d", i, test.in, n1, test.val)
  269. }
  270. if n2.Cmp(expected) != 0 {
  271. t.Errorf("#%d (input '%s') got: %s want: %d", i, test.in, n2, test.val)
  272. }
  273. }
  274. }
  275. func TestDecScan(t *testing.T) {
  276. tmp := new(Dec)
  277. for i, test := range decStringTests {
  278. if test.scale < 0 {
  279. // SetString only supports scale >= 0
  280. continue
  281. }
  282. // initialize to a non-zero value so that issues with parsing
  283. // 0 are detected
  284. tmp.Set(NewDec(1234567890, 123))
  285. n1, n2 := new(Dec), tmp
  286. nn1, err1 := fmt.Sscan(test.in, n1)
  287. nn2, err2 := fmt.Sscan(test.in, n2)
  288. if !test.scanOk {
  289. if err1 == nil || err2 == nil {
  290. t.Errorf("#%d (input '%s') ok incorrect, should be %t", i, test.in, test.scanOk)
  291. }
  292. continue
  293. }
  294. expected := NewDec(test.val, test.scale)
  295. if nn1 != 1 || err1 != nil || nn2 != 1 || err2 != nil {
  296. t.Errorf("#%d (input '%s') error %d %v, %d %v", i, test.in, nn1, err1, nn2, err2)
  297. continue
  298. }
  299. if n1.Cmp(expected) != 0 {
  300. t.Errorf("#%d (input '%s') got: %s want: %d", i, test.in, n1, test.val)
  301. }
  302. if n2.Cmp(expected) != 0 {
  303. t.Errorf("#%d (input '%s') got: %s want: %d", i, test.in, n2, test.val)
  304. }
  305. }
  306. }
  307. var decScanNextTests = []struct {
  308. in string
  309. ok bool
  310. next rune
  311. }{
  312. {"", false, 0},
  313. {"a", false, 'a'},
  314. {"z", false, 'z'},
  315. {"+", false, 0},
  316. {"-", false, 0},
  317. {"g", false, 'g'},
  318. {".", false, 0},
  319. {".-0", false, '-'},
  320. {".+0", false, '+'},
  321. {"0b", true, 'b'},
  322. {"0x", true, 'x'},
  323. {"0xg", true, 'x'},
  324. {"0.0g", true, 'g'},
  325. }
  326. func TestDecScanNext(t *testing.T) {
  327. for i, test := range decScanNextTests {
  328. rdr := strings.NewReader(test.in)
  329. n1 := new(Dec)
  330. nn1, _ := fmt.Fscan(rdr, n1)
  331. if (test.ok && nn1 == 0) || (!test.ok && nn1 > 0) {
  332. t.Errorf("#%d (input '%s') ok incorrect should be %t", i, test.in, test.ok)
  333. continue
  334. }
  335. r := rune(0)
  336. nn2, err := fmt.Fscanf(rdr, "%c", &r)
  337. if test.next != r {
  338. t.Errorf("#%d (input '%s') next incorrect, got %c should be %c, %d, %v", i, test.in, r, test.next, nn2, err)
  339. }
  340. }
  341. }
  342. var decGobEncodingTests = []string{
  343. "0",
  344. "1",
  345. "2",
  346. "10",
  347. "42",
  348. "1234567890",
  349. "298472983472983471903246121093472394872319615612417471234712061",
  350. }
  351. func TestDecGobEncoding(t *testing.T) {
  352. var medium bytes.Buffer
  353. enc := gob.NewEncoder(&medium)
  354. dec := gob.NewDecoder(&medium)
  355. for i, test := range decGobEncodingTests {
  356. for j := 0; j < 2; j++ {
  357. for k := Scale(-5); k <= 5; k++ {
  358. medium.Reset() // empty buffer for each test case (in case of failures)
  359. stest := test
  360. if j != 0 {
  361. // negative numbers
  362. stest = "-" + test
  363. }
  364. var tx Dec
  365. tx.SetString(stest)
  366. tx.SetScale(k) // test with positive, negative, and zero scale
  367. if err := enc.Encode(&tx); err != nil {
  368. t.Errorf("#%d%c: encoding failed: %s", i, 'a'+j, err)
  369. }
  370. var rx Dec
  371. if err := dec.Decode(&rx); err != nil {
  372. t.Errorf("#%d%c: decoding failed: %s", i, 'a'+j, err)
  373. }
  374. if rx.Cmp(&tx) != 0 {
  375. t.Errorf("#%d%c: transmission failed: got %s want %s", i, 'a'+j, &rx, &tx)
  376. }
  377. }
  378. }
  379. }
  380. }