trackler 2.2.1.45 → 2.2.1.46

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (508) hide show
  1. checksums.yaml +4 -4
  2. data/lib/trackler/version.rb +1 -1
  3. data/problem-specifications/exercises/complex-numbers/description.md +27 -0
  4. data/problem-specifications/exercises/crypto-square/canonical-data.json +39 -43
  5. data/problem-specifications/exercises/crypto-square/description.md +6 -4
  6. data/problem-specifications/exercises/hamming/canonical-data.json +5 -5
  7. data/problem-specifications/exercises/kindergarten-garden/description.md +10 -10
  8. data/problem-specifications/exercises/leap/canonical-data.json +2 -2
  9. data/problem-specifications/exercises/markdown/canonical-data.json +2 -2
  10. data/problem-specifications/exercises/nucleotide-count/canonical-data.json +12 -1
  11. data/problem-specifications/exercises/pangram/canonical-data.json +7 -1
  12. data/problem-specifications/exercises/queen-attack/canonical-data.json +83 -26
  13. data/problem-specifications/exercises/space-age/description.md +1 -2
  14. data/problem-specifications/exercises/zipper/canonical-data.json +671 -0
  15. data/tracks/bash/config.json +13 -1
  16. data/tracks/bash/exercises/atbash-cipher/README.md +45 -0
  17. data/tracks/bash/exercises/atbash-cipher/atbash_cipher_tests.sh +73 -0
  18. data/tracks/bash/exercises/atbash-cipher/example.sh +14 -0
  19. data/tracks/bash/exercises/gigasecond/.meta/hints.md +33 -0
  20. data/tracks/bash/exercises/two-fer/README.md +0 -22
  21. data/tracks/c/exercises/allergies/src/allergies.h +9 -9
  22. data/tracks/c/exercises/allergies/src/example.c +2 -2
  23. data/tracks/c/exercises/allergies/src/example.h +9 -9
  24. data/tracks/c/exercises/allergies/test/test_allergies.c +34 -34
  25. data/tracks/c/exercises/clock/test/test_clock.c +0 -52
  26. data/tracks/c/exercises/meetup/test/test_meetup.c +604 -114
  27. data/tracks/c/exercises/nucleotide-count/src/example.c +10 -10
  28. data/tracks/c/exercises/palindrome-products/src/example.c +8 -8
  29. data/tracks/c/exercises/palindrome-products/src/example.h +5 -5
  30. data/tracks/c/exercises/palindrome-products/test/test_palindrome_products.c +8 -6
  31. data/tracks/c/exercises/robot-simulator/src/example.c +19 -19
  32. data/tracks/c/exercises/robot-simulator/src/robot_simulator.h +24 -24
  33. data/tracks/c/exercises/robot-simulator/test/test_robot_simulator.c +35 -34
  34. data/tracks/c/exercises/series/src/example.c +10 -10
  35. data/tracks/c/exercises/series/src/series.h +5 -5
  36. data/tracks/c/exercises/series/test/test_series.c +77 -30
  37. data/tracks/c/exercises/sieve/src/example.c +17 -17
  38. data/tracks/c/exercises/sieve/src/sieve.h +2 -2
  39. data/tracks/c/exercises/sieve/test/test_sieve.c +6 -6
  40. data/tracks/c/exercises/space-age/src/example.c +1 -1
  41. data/tracks/c/exercises/space-age/src/example.h +3 -3
  42. data/tracks/c/exercises/space-age/test/test_space_age.c +2 -0
  43. data/tracks/c/exercises/sum-of-multiples/src/example.c +3 -1
  44. data/tracks/c/exercises/sum-of-multiples/test/test_sum_of_multiples.c +17 -0
  45. data/tracks/c/exercises/triangle/src/example.c +4 -4
  46. data/tracks/c/exercises/triangle/src/example.h +4 -4
  47. data/tracks/c/exercises/triangle/src/triangle.h +1 -1
  48. data/tracks/c/exercises/triangle/test/test_triangle.c +18 -16
  49. data/tracks/c/exercises/word-count/src/example.c +24 -24
  50. data/tracks/c/exercises/word-count/src/word_count.h +3 -3
  51. data/tracks/c/exercises/word-count/test/test_word_count.c +139 -139
  52. data/tracks/cfml/exercises/acronym/README.md +31 -0
  53. data/tracks/cfml/exercises/atbash-cipher/README.md +50 -0
  54. data/tracks/cfml/exercises/bob/README.md +34 -0
  55. data/tracks/cfml/exercises/diamond/README.md +75 -0
  56. data/tracks/cfml/exercises/difference-of-squares/README.md +35 -0
  57. data/tracks/cfml/exercises/flatten-array/README.md +34 -0
  58. data/tracks/cfml/exercises/gigasecond/README.md +27 -0
  59. data/tracks/cfml/exercises/grains/README.md +50 -0
  60. data/tracks/cfml/exercises/hamming/README.md +58 -0
  61. data/tracks/cfml/exercises/hello-world/README.md +37 -0
  62. data/tracks/cfml/exercises/isogram/README.md +35 -0
  63. data/tracks/cfml/exercises/largest-series-product/README.md +36 -0
  64. data/tracks/cfml/exercises/leap/README.md +16 -0
  65. data/tracks/cfml/exercises/luhn/README.md +87 -0
  66. data/tracks/cfml/exercises/markdown/README.md +34 -0
  67. data/tracks/cfml/exercises/nth-prime/README.md +31 -0
  68. data/tracks/cfml/exercises/pangram/README.md +31 -0
  69. data/tracks/cfml/exercises/pig-latin/README.md +40 -0
  70. data/tracks/cfml/exercises/raindrops/README.md +40 -0
  71. data/tracks/cfml/exercises/rna-transcription/README.md +41 -0
  72. data/tracks/cfml/exercises/saddle-points/README.md +49 -0
  73. data/tracks/cfml/exercises/scrabble-score/README.md +60 -0
  74. data/tracks/cfml/exercises/secret-handshake/README.md +51 -0
  75. data/tracks/cfml/exercises/space-age/README.md +41 -0
  76. data/tracks/cfml/exercises/sum-of-multiples/README.md +34 -0
  77. data/tracks/cfml/exercises/triangle/README.md +42 -0
  78. data/tracks/cfml/exercises/word-count/README.md +35 -0
  79. data/tracks/clojure/config.json +10 -0
  80. data/tracks/clojure/exercises/spiral-matrix/README.md +32 -0
  81. data/tracks/clojure/exercises/spiral-matrix/project.clj +4 -0
  82. data/tracks/clojure/exercises/spiral-matrix/src/example.clj +11 -0
  83. data/tracks/clojure/exercises/spiral-matrix/src/spiral_matrix.clj +3 -0
  84. data/tracks/clojure/exercises/spiral-matrix/test/spiral_matrix_test.clj +58 -0
  85. data/tracks/common-lisp/config.json +20 -0
  86. data/tracks/common-lisp/exercises/hello-world/README.md +68 -0
  87. data/tracks/common-lisp/exercises/hello-world/example.lisp +9 -0
  88. data/tracks/common-lisp/exercises/hello-world/hello-world-test.lisp +21 -0
  89. data/tracks/common-lisp/exercises/hello-world/hello-world.lisp +6 -0
  90. data/tracks/common-lisp/exercises/perfect-numbers/README.md +72 -0
  91. data/tracks/common-lisp/exercises/perfect-numbers/example.lisp +17 -0
  92. data/tracks/common-lisp/exercises/perfect-numbers/perfect-numbers-test.lisp +56 -0
  93. data/tracks/common-lisp/exercises/perfect-numbers/perfect-numbers.lisp +7 -0
  94. data/tracks/cpp/.travis.yml +34 -14
  95. data/tracks/cpp/docs/INSTALLATION.md +17 -28
  96. data/tracks/cpp/docs/TESTS.md +13 -21
  97. data/tracks/crystal/config.json +10 -0
  98. data/tracks/crystal/exercises/triangle/spec/triangle_spec.cr +60 -0
  99. data/tracks/crystal/exercises/triangle/src/example.cr +31 -0
  100. data/tracks/crystal/exercises/triangle/src/triangle.cr +1 -0
  101. data/tracks/crystal/generator/spec/remote_data_file_spec.cr +0 -1
  102. data/tracks/crystal/generator/src/generators/triangle.cr +38 -0
  103. data/tracks/crystal/generator/src/remote_data_file.cr +1 -1
  104. data/tracks/csharp/exercises/difference-of-squares/DifferenceOfSquares.cs +4 -4
  105. data/tracks/csharp/exercises/difference-of-squares/DifferenceOfSquaresTest.cs +22 -27
  106. data/tracks/csharp/exercises/difference-of-squares/Example.cs +5 -5
  107. data/tracks/csharp/exercises/error-handling/ErrorHandlingTest.cs +4 -4
  108. data/tracks/csharp/exercises/flatten-array/Example.cs +1 -1
  109. data/tracks/csharp/exercises/flatten-array/FlattenArray.cs +1 -1
  110. data/tracks/csharp/exercises/flatten-array/FlattenArrayTest.cs +106 -72
  111. data/tracks/csharp/exercises/largest-series-product/LargestSeriesProductTest.cs +24 -98
  112. data/tracks/csharp/exercises/minesweeper/Example.cs +3 -3
  113. data/tracks/csharp/exercises/minesweeper/Minesweeper.cs +1 -1
  114. data/tracks/csharp/exercises/minesweeper/MinesweeperTest.cs +155 -57
  115. data/tracks/csharp/generators/Exercises/DifferenceOfSquares.cs +25 -0
  116. data/tracks/csharp/generators/Exercises/FlattenArray.cs +36 -0
  117. data/tracks/csharp/generators/Exercises/LargestSeriesProduct.cs +18 -0
  118. data/tracks/csharp/generators/Exercises/Minesweeper.cs +32 -0
  119. data/tracks/d/docs/EXERCISE_README_INSERT.md +1 -1
  120. data/tracks/d/exercises/bob/README.md +1 -1
  121. data/tracks/d/exercises/circular-buffer/README.md +1 -1
  122. data/tracks/d/exercises/crypto-square/README.md +1 -1
  123. data/tracks/d/exercises/difference-of-squares/README.md +1 -1
  124. data/tracks/d/exercises/etl/README.md +1 -1
  125. data/tracks/d/exercises/gigasecond/README.md +1 -1
  126. data/tracks/d/exercises/hamming/README.md +1 -1
  127. data/tracks/d/exercises/hello-world/README.md +1 -1
  128. data/tracks/d/exercises/leap/README.md +1 -1
  129. data/tracks/d/exercises/nucleotide-count/README.md +1 -1
  130. data/tracks/d/exercises/pangram/README.md +1 -1
  131. data/tracks/d/exercises/raindrops/README.md +1 -1
  132. data/tracks/d/exercises/react/README.md +1 -1
  133. data/tracks/d/exercises/rna-transcription/README.md +1 -1
  134. data/tracks/d/exercises/robot-name/README.md +1 -1
  135. data/tracks/d/exercises/roman-numerals/README.md +1 -1
  136. data/tracks/d/exercises/series/README.md +1 -1
  137. data/tracks/d/exercises/triangle/README.md +1 -1
  138. data/tracks/dart/CONTRIBUTING.md +61 -0
  139. data/tracks/dart/README.md +10 -47
  140. data/tracks/dart/config.json +13 -1
  141. data/tracks/dart/exercises/anagram/README.md +10 -0
  142. data/tracks/dart/exercises/bob/README.md +11 -0
  143. data/tracks/dart/exercises/difference-of-squares/README.md +11 -0
  144. data/tracks/dart/exercises/gigasecond/README.md +11 -0
  145. data/tracks/dart/exercises/gigasecond/test/gigasecond_test.dart +56 -33
  146. data/tracks/dart/exercises/hamming/README.md +11 -0
  147. data/tracks/dart/exercises/hamming/lib/example.dart +2 -2
  148. data/tracks/dart/exercises/hamming/test/hamming_test.dart +41 -24
  149. data/tracks/dart/exercises/hello-world/README.md +11 -0
  150. data/tracks/dart/exercises/leap/README.md +12 -1
  151. data/tracks/dart/exercises/phone-number/README.md +56 -0
  152. data/tracks/dart/exercises/phone-number/lib/example.dart +45 -0
  153. data/tracks/dart/exercises/phone-number/lib/phone_number.dart +3 -0
  154. data/tracks/dart/exercises/phone-number/pubspec.lock +293 -0
  155. data/tracks/dart/exercises/phone-number/pubspec.yaml +3 -0
  156. data/tracks/dart/exercises/phone-number/test/phone_number_test.dart +66 -0
  157. data/tracks/dart/exercises/raindrops/README.md +35 -0
  158. data/tracks/dart/exercises/rna-transcription/README.md +11 -0
  159. data/tracks/dart/exercises/word-count/README.md +14 -3
  160. data/tracks/delphi/docs/RESOURCES.md +10 -11
  161. data/tracks/elixir/exercises/hello-world/hello_world.exs +1 -1
  162. data/tracks/elixir/exercises/secret-handshake/README.md +3 -5
  163. data/tracks/elm/exercises/isogram/Isogram.example.elm +4 -4
  164. data/tracks/erlang/config.json +10 -0
  165. data/tracks/erlang/exercises/sieve/README.md +88 -0
  166. data/tracks/erlang/exercises/sieve/include/exercism.hrl +11 -0
  167. data/tracks/erlang/exercises/sieve/rebar.config +30 -0
  168. data/tracks/erlang/exercises/sieve/src/example.erl +16 -0
  169. data/tracks/erlang/exercises/sieve/src/sieve.app.src +9 -0
  170. data/tracks/erlang/exercises/sieve/src/sieve.erl +8 -0
  171. data/tracks/erlang/exercises/sieve/test/sieve_tests.erl +38 -0
  172. data/tracks/fsharp/.gitignore +2 -1
  173. data/tracks/fsharp/exercises/crypto-square/CryptoSquareTest.fs +10 -6
  174. data/tracks/fsharp/exercises/leap/LeapTest.fs +2 -2
  175. data/tracks/fsharp/exercises/pangram/PangramTest.fs +5 -1
  176. data/tracks/fsharp/exercises/queen-attack/QueenAttackTest.fs +7 -7
  177. data/tracks/fsharp/exercises/run-length-encoding/Example.fs +2 -2
  178. data/tracks/fsharp/exercises/run-length-encoding/RunLengthEncodingTest.fs +42 -30
  179. data/tracks/fsharp/generators/Generators.fs +21 -8
  180. data/tracks/gnu-apl/test.apl +4 -4
  181. data/tracks/go/README.md +1 -2
  182. data/tracks/go/config/maintainers.json +5 -6
  183. data/tracks/groovy/config.json +23 -0
  184. data/tracks/groovy/exercises/binary-search/BinarySearch.groovy +14 -0
  185. data/tracks/groovy/exercises/binary-search/BinarySearchSpec.groovy +56 -0
  186. data/tracks/groovy/exercises/binary-search/Example.groovy +43 -0
  187. data/tracks/groovy/exercises/binary-search/README.md +59 -0
  188. data/tracks/groovy/exercises/triangle/Example.groovy +23 -0
  189. data/tracks/groovy/exercises/triangle/README.md +42 -0
  190. data/tracks/groovy/exercises/triangle/Triangle.groovy +15 -0
  191. data/tracks/groovy/exercises/triangle/TriangleSpec.groovy +66 -0
  192. data/tracks/haskell/config.json +9 -0
  193. data/tracks/haskell/exercises/all-your-base/package.yaml +1 -1
  194. data/tracks/haskell/exercises/all-your-base/test/Tests.hs +18 -18
  195. data/tracks/haskell/exercises/bowling/package.yaml +1 -1
  196. data/tracks/haskell/exercises/bowling/test/Tests.hs +9 -9
  197. data/tracks/haskell/exercises/change/package.yaml +1 -1
  198. data/tracks/haskell/exercises/change/test/Tests.hs +5 -0
  199. data/tracks/haskell/exercises/complex-numbers/README.md +92 -0
  200. data/tracks/haskell/exercises/complex-numbers/examples/success-standard/package.yaml +16 -0
  201. data/tracks/haskell/exercises/complex-numbers/examples/success-standard/src/ComplexNumbers.hs +50 -0
  202. data/tracks/haskell/exercises/complex-numbers/package.yaml +20 -0
  203. data/tracks/haskell/exercises/complex-numbers/src/ComplexNumbers.hs +45 -0
  204. data/tracks/haskell/exercises/complex-numbers/stack.yaml +1 -0
  205. data/tracks/haskell/exercises/complex-numbers/test/Tests.hs +194 -0
  206. data/tracks/haskell/exercises/crypto-square/package.yaml +1 -1
  207. data/tracks/haskell/exercises/crypto-square/test/Tests.hs +20 -12
  208. data/tracks/haskell/exercises/dominoes/package.yaml +1 -1
  209. data/tracks/haskell/exercises/forth/package.yaml +1 -1
  210. data/tracks/haskell/exercises/forth/test/Tests.hs +4 -3
  211. data/tracks/haskell/exercises/hamming/package.yaml +1 -1
  212. data/tracks/haskell/exercises/hamming/test/Tests.hs +6 -6
  213. data/tracks/haskell/exercises/isogram/package.yaml +1 -1
  214. data/tracks/haskell/exercises/isogram/test/Tests.hs +2 -2
  215. data/tracks/haskell/exercises/leap/package.yaml +1 -1
  216. data/tracks/haskell/exercises/leap/test/Tests.hs +1 -1
  217. data/tracks/haskell/exercises/pascals-triangle/package.yaml +1 -1
  218. data/tracks/haskell/exercises/pig-latin/examples/success-standard/src/PigLatin.hs +2 -0
  219. data/tracks/haskell/exercises/pig-latin/package.yaml +1 -1
  220. data/tracks/haskell/exercises/pig-latin/test/Tests.hs +8 -0
  221. data/tracks/haskell/exercises/queen-attack/package.yaml +1 -1
  222. data/tracks/haskell/exercises/rna-transcription/package.yaml +1 -1
  223. data/tracks/haskell/exercises/rna-transcription/test/Tests.hs +8 -8
  224. data/tracks/haskell/exercises/robot-simulator/package.yaml +1 -1
  225. data/tracks/haskell/exercises/secret-handshake/package.yaml +1 -1
  226. data/tracks/haskell/exercises/secret-handshake/test/Tests.hs +0 -3
  227. data/tracks/haskell/exercises/sum-of-multiples/package.yaml +1 -1
  228. data/tracks/haskell/exercises/sum-of-multiples/test/Tests.hs +4 -0
  229. data/tracks/haskell/exercises/zipper/package.yaml +1 -1
  230. data/tracks/haskell/exercises/zipper/test/Tests.hs +23 -0
  231. data/tracks/java/config.json +85 -13
  232. data/tracks/java/config/maintainers.json +1 -1
  233. data/tracks/java/exercises/accumulate/README.md +0 -3
  234. data/tracks/java/exercises/acronym/.meta/.version +1 -0
  235. data/tracks/java/exercises/acronym/README.md +0 -1
  236. data/tracks/java/exercises/acronym/src/test/java/AcronymTest.java +0 -3
  237. data/tracks/java/exercises/all-your-base/.meta/.version +1 -0
  238. data/tracks/java/exercises/all-your-base/README.md +1 -1
  239. data/tracks/java/exercises/all-your-base/src/test/java/BaseConverterTest.java +0 -3
  240. data/tracks/java/exercises/allergies/README.md +0 -1
  241. data/tracks/java/exercises/atbash-cipher/README.md +2 -1
  242. data/tracks/java/exercises/beer-song/README.md +1 -1
  243. data/tracks/java/exercises/binary/README.md +2 -0
  244. data/tracks/java/exercises/book-store/README.md +5 -5
  245. data/tracks/java/exercises/bowling/README.md +20 -6
  246. data/tracks/java/exercises/change/.meta/.version +1 -0
  247. data/tracks/java/exercises/change/README.md +1 -1
  248. data/tracks/java/exercises/change/src/test/java/ChangeCalculatorTest.java +0 -3
  249. data/tracks/java/exercises/circular-buffer/README.md +13 -6
  250. data/tracks/java/exercises/clock/.meta/.version +1 -0
  251. data/tracks/java/exercises/clock/src/test/java/ClockAddTest.java +0 -3
  252. data/tracks/java/exercises/clock/src/test/java/ClockCreationTest.java +0 -3
  253. data/tracks/java/exercises/clock/src/test/java/ClockEqualTest.java +0 -3
  254. data/tracks/java/exercises/collatz-conjecture/.meta/.version +1 -0
  255. data/tracks/java/exercises/collatz-conjecture/README.md +1 -1
  256. data/tracks/java/exercises/collatz-conjecture/src/test/java/CollatzCalculatorTest.java +0 -3
  257. data/tracks/java/exercises/complex-numbers/README.md +27 -0
  258. data/tracks/java/exercises/crypto-square/README.md +10 -8
  259. data/tracks/java/exercises/diamond/.meta/.version +1 -0
  260. data/tracks/java/exercises/diamond/README.md +6 -6
  261. data/tracks/java/exercises/diamond/src/test/java/DiamondPrinterTest.java +0 -3
  262. data/tracks/java/exercises/difference-of-squares/.meta/.version +1 -0
  263. data/tracks/java/exercises/difference-of-squares/src/test/java/DifferenceOfSquaresCalculatorTest.java +0 -3
  264. data/tracks/java/exercises/etl/README.md +2 -0
  265. data/tracks/java/exercises/flatten-array/.meta/.version +1 -0
  266. data/tracks/java/exercises/flatten-array/README.md +1 -2
  267. data/tracks/java/exercises/flatten-array/src/test/java/FlattenerTest.java +0 -3
  268. data/tracks/java/exercises/food-chain/README.md +1 -1
  269. data/tracks/java/exercises/forth/.meta/.version +1 -0
  270. data/tracks/java/exercises/forth/src/test/java/ForthEvaluatorTest.java +0 -3
  271. data/tracks/java/exercises/grade-school/README.md +0 -1
  272. data/tracks/java/exercises/hamming/.meta/.version +1 -0
  273. data/tracks/java/exercises/hamming/src/test/java/HammingTest.java +0 -3
  274. data/tracks/java/exercises/house/README.md +1 -2
  275. data/tracks/java/exercises/isogram/README.md +2 -1
  276. data/tracks/java/exercises/kindergarten-garden/README.md +13 -13
  277. data/tracks/java/exercises/linked-list/README.md +10 -10
  278. data/tracks/java/exercises/list-ops/.meta/.version +1 -0
  279. data/tracks/java/exercises/list-ops/README.md +4 -0
  280. data/tracks/java/exercises/list-ops/src/test/java/ListOpsTest.java +0 -3
  281. data/tracks/java/exercises/luhn/README.md +7 -7
  282. data/tracks/java/exercises/matrix/README.md +6 -4
  283. data/tracks/java/exercises/meetup/README.md +1 -2
  284. data/tracks/java/exercises/minesweeper/.meta/.version +1 -0
  285. data/tracks/java/exercises/minesweeper/src/test/java/MinesweeperBoardTest.java +0 -3
  286. data/tracks/java/exercises/nucleotide-count/README.md +8 -22
  287. data/tracks/java/exercises/ocr-numbers/README.md +5 -5
  288. data/tracks/java/exercises/octal/README.md +6 -2
  289. data/tracks/java/exercises/palindrome-products/README.md +15 -16
  290. data/tracks/java/exercises/pangram/.meta/.version +1 -0
  291. data/tracks/java/exercises/pangram/README.md +1 -1
  292. data/tracks/java/exercises/pangram/src/test/java/PangramCheckerTest.java +0 -3
  293. data/tracks/java/exercises/pascals-triangle/.meta/.version +1 -0
  294. data/tracks/java/exercises/pascals-triangle/README.md +1 -1
  295. data/tracks/java/exercises/pascals-triangle/src/test/java/PascalsTriangleGeneratorTest.java +0 -3
  296. data/tracks/java/exercises/perfect-numbers/.meta/.version +1 -0
  297. data/tracks/java/exercises/perfect-numbers/README.md +2 -2
  298. data/tracks/java/exercises/perfect-numbers/src/test/java/NaturalNumberTest.java +0 -3
  299. data/tracks/java/exercises/phone-number/README.md +3 -2
  300. data/tracks/java/exercises/pythagorean-triplet/README.md +3 -3
  301. data/tracks/java/exercises/queen-attack/README.md +1 -1
  302. data/tracks/java/exercises/rectangles/README.md +7 -7
  303. data/tracks/java/exercises/rna-transcription/.meta/.version +1 -0
  304. data/tracks/java/exercises/rna-transcription/src/test/java/RnaTranscriptionTest.java +0 -3
  305. data/tracks/java/exercises/roman-numerals/README.md +1 -1
  306. data/tracks/java/exercises/rotational-cipher/README.md +2 -1
  307. data/tracks/java/exercises/run-length-encoding/README.md +4 -4
  308. data/tracks/java/exercises/saddle-points/.meta/.version +1 -0
  309. data/tracks/java/exercises/saddle-points/README.md +1 -1
  310. data/tracks/java/exercises/saddle-points/src/test/java/MatrixTest.java +0 -3
  311. data/tracks/java/exercises/scrabble-score/.meta/.version +1 -0
  312. data/tracks/java/exercises/scrabble-score/README.md +3 -1
  313. data/tracks/java/exercises/scrabble-score/src/test/java/ScrabbleScoreTest.java +0 -3
  314. data/tracks/java/exercises/secret-handshake/README.md +1 -1
  315. data/tracks/java/exercises/simple-cipher/README.md +1 -1
  316. data/tracks/java/exercises/simple-linked-list/src/test/java/SimpleLinkedListTest.java +25 -25
  317. data/tracks/java/exercises/space-age/README.md +1 -1
  318. data/tracks/java/exercises/spiral-matrix/.meta/.version +1 -0
  319. data/tracks/java/exercises/spiral-matrix/README.md +2 -2
  320. data/tracks/java/exercises/spiral-matrix/src/test/java/SpiralMatrixBuilderTest.java +0 -3
  321. data/tracks/java/exercises/sum-of-multiples/README.md +0 -3
  322. data/tracks/java/exercises/tournament/README.md +6 -5
  323. data/tracks/java/exercises/transpose/README.md +7 -7
  324. data/tracks/java/exercises/triangle/README.md +9 -6
  325. data/tracks/java/exercises/trinary/README.md +1 -1
  326. data/tracks/java/exercises/twelve-days/README.md +1 -1
  327. data/tracks/java/exercises/two-fer/README.md +1 -2
  328. data/tracks/java/exercises/word-count/README.md +1 -2
  329. data/tracks/java/exercises/word-search/.meta/.version +1 -0
  330. data/tracks/java/exercises/word-search/README.md +1 -1
  331. data/tracks/java/exercises/word-search/src/test/java/WordSearcherTest.java +0 -3
  332. data/tracks/java/exercises/wordy/README.md +0 -5
  333. data/tracks/javascript/config.json +13 -0
  334. data/tracks/javascript/docs/TESTS.md +1 -1
  335. data/tracks/javascript/exercises/diffie-hellman/example.js +1 -1
  336. data/tracks/javascript/exercises/hello-world/example.js +2 -3
  337. data/tracks/javascript/exercises/meetup/example.js +24 -27
  338. data/tracks/javascript/exercises/meetup/meetup.spec.js +0 -6
  339. data/tracks/javascript/exercises/nucleotide-count/nucleotide-count.spec.js +1 -1
  340. data/tracks/javascript/exercises/protein-translation/README.md +73 -0
  341. data/tracks/javascript/exercises/protein-translation/example.js +68 -0
  342. data/tracks/javascript/exercises/protein-translation/protein-translation.spec.js +63 -0
  343. data/tracks/julia/README.md +1 -1
  344. data/tracks/kotlin/config.json +11 -0
  345. data/tracks/kotlin/exercises/acronym/README.md +0 -1
  346. data/tracks/kotlin/exercises/all-your-base/README.md +1 -1
  347. data/tracks/kotlin/exercises/allergies/README.md +0 -1
  348. data/tracks/kotlin/exercises/atbash-cipher/README.md +2 -1
  349. data/tracks/kotlin/exercises/beer-song/README.md +1 -1
  350. data/tracks/kotlin/exercises/binary/README.md +2 -0
  351. data/tracks/kotlin/exercises/change/README.md +1 -1
  352. data/tracks/kotlin/exercises/collatz-conjecture/README.md +1 -1
  353. data/tracks/kotlin/exercises/complex-numbers/README.md +27 -0
  354. data/tracks/kotlin/exercises/diamond/README.md +6 -6
  355. data/tracks/kotlin/exercises/etl/README.md +2 -0
  356. data/tracks/kotlin/exercises/flatten-array/README.md +1 -2
  357. data/tracks/kotlin/exercises/grade-school/README.md +0 -1
  358. data/tracks/kotlin/exercises/grains/README.md +0 -1
  359. data/tracks/kotlin/exercises/hello-world/.meta/hints.md +4 -0
  360. data/tracks/kotlin/exercises/hello-world/.meta/src/reference/kotlin/HelloWorld.kt +2 -2
  361. data/tracks/kotlin/exercises/hello-world/README.md +6 -0
  362. data/tracks/kotlin/exercises/hello-world/TUTORIAL.md +32 -434
  363. data/tracks/kotlin/exercises/hello-world/src/main/kotlin/HelloWorld.kt +1 -1
  364. data/tracks/kotlin/exercises/hello-world/src/test/kotlin/HelloWorldTest.kt +2 -21
  365. data/tracks/kotlin/exercises/isogram/README.md +2 -1
  366. data/tracks/kotlin/exercises/leap/README.md +1 -1
  367. data/tracks/kotlin/exercises/linked-list/README.md +10 -10
  368. data/tracks/kotlin/exercises/luhn/README.md +7 -7
  369. data/tracks/kotlin/exercises/meetup/README.md +1 -2
  370. data/tracks/kotlin/exercises/nucleotide-count/README.md +8 -22
  371. data/tracks/kotlin/exercises/pangram/README.md +1 -1
  372. data/tracks/kotlin/exercises/pascals-triangle/README.md +1 -1
  373. data/tracks/kotlin/exercises/perfect-numbers/README.md +2 -2
  374. data/tracks/kotlin/exercises/phone-number/README.md +3 -2
  375. data/tracks/kotlin/exercises/roman-numerals/README.md +1 -1
  376. data/tracks/kotlin/exercises/rotational-cipher/README.md +2 -1
  377. data/tracks/kotlin/exercises/saddle-points/README.md +1 -1
  378. data/tracks/kotlin/exercises/scrabble-score/README.md +3 -1
  379. data/tracks/kotlin/exercises/secret-handshake/README.md +1 -1
  380. data/tracks/kotlin/exercises/settings.gradle +1 -0
  381. data/tracks/kotlin/exercises/space-age/README.md +1 -2
  382. data/tracks/kotlin/exercises/spiral-matrix/README.md +2 -2
  383. data/tracks/kotlin/exercises/sum-of-multiples/README.md +0 -3
  384. data/tracks/kotlin/exercises/triangle/README.md +9 -6
  385. data/tracks/kotlin/exercises/two-fer/.meta/src/reference/kotlin/Twofer.kt +3 -0
  386. data/tracks/kotlin/exercises/two-fer/README.md +50 -0
  387. data/tracks/kotlin/exercises/two-fer/build.gradle +28 -0
  388. data/tracks/kotlin/exercises/two-fer/src/main/kotlin/.keep +0 -0
  389. data/tracks/kotlin/exercises/two-fer/src/test/kotlin/TwoferTest.kt +31 -0
  390. data/tracks/kotlin/exercises/word-count/README.md +1 -2
  391. data/tracks/ocaml/exercises/pangram/test.ml +3 -1
  392. data/tracks/perl5/.travis.yml +3 -1
  393. data/tracks/php/Makefile +2 -2
  394. data/tracks/php/README.md +4 -4
  395. data/tracks/php/config.json +15 -0
  396. data/tracks/php/exercises/transpose/example.php +30 -0
  397. data/tracks/php/exercises/transpose/transpose_test.php +116 -0
  398. data/tracks/php/{phpcs-xphp.xml → phpcs-php.xml} +0 -0
  399. data/tracks/python/.travis.yml +1 -1
  400. data/tracks/python/config.json +199 -39
  401. data/tracks/python/exercises/acronym/acronym_test.py +1 -1
  402. data/tracks/python/exercises/allergies/allergies_test.py +1 -1
  403. data/tracks/python/exercises/alphametics/alphametics_test.py +1 -1
  404. data/tracks/python/exercises/anagram/anagram_test.py +1 -1
  405. data/tracks/python/exercises/atbash-cipher/atbash_cipher_test.py +1 -1
  406. data/tracks/python/exercises/bob/bob.py +1 -1
  407. data/tracks/python/exercises/bob/bob_test.py +1 -1
  408. data/tracks/python/exercises/book-store/book_store_test.py +1 -1
  409. data/tracks/python/exercises/bracket-push/bracket_push.py +1 -1
  410. data/tracks/python/exercises/bracket-push/bracket_push_test.py +1 -1
  411. data/tracks/python/exercises/collatz-conjecture/collatz_conjecture_test.py +1 -1
  412. data/tracks/python/exercises/diamond/diamond_test.py +1 -1
  413. data/tracks/python/exercises/difference-of-squares/difference_of_squares_test.py +1 -1
  414. data/tracks/python/exercises/diffie-hellman/README.md +58 -0
  415. data/tracks/python/exercises/diffie-hellman/diffie_hellman.py +10 -0
  416. data/tracks/python/exercises/diffie-hellman/diffie_hellman_test.py +87 -0
  417. data/tracks/python/exercises/diffie-hellman/example.py +13 -0
  418. data/tracks/python/exercises/etl/etl_test.py +1 -1
  419. data/tracks/python/exercises/food-chain/README.md +81 -0
  420. data/tracks/python/exercises/food-chain/example.py +45 -0
  421. data/tracks/python/exercises/food-chain/food_chain.py +2 -0
  422. data/tracks/python/exercises/food-chain/food_chain_test.py +109 -0
  423. data/tracks/python/exercises/forth/README.md +43 -0
  424. data/tracks/python/exercises/forth/example.py +58 -0
  425. data/tracks/python/exercises/forth/forth.py +2 -0
  426. data/tracks/python/exercises/forth/forth_test.py +254 -0
  427. data/tracks/python/exercises/gigasecond/gigasecond_test.py +1 -1
  428. data/tracks/python/exercises/grep/grep_test.py +1 -1
  429. data/tracks/python/exercises/hello-world/hello_world_test.py +1 -1
  430. data/tracks/python/exercises/isogram/isogram_test.py +10 -10
  431. data/tracks/python/exercises/largest-series-product/largest_series_product_test.py +1 -1
  432. data/tracks/python/exercises/leap/leap_test.py +5 -5
  433. data/tracks/python/exercises/luhn/luhn_test.py +16 -16
  434. data/tracks/python/exercises/markdown/README.md +30 -0
  435. data/tracks/python/exercises/markdown/example.py +80 -0
  436. data/tracks/python/exercises/markdown/markdown.py +71 -0
  437. data/tracks/python/exercises/markdown/markdown_test.py +51 -0
  438. data/tracks/python/exercises/meetup/meetup_test.py +1 -1
  439. data/tracks/python/exercises/nth-prime/nth_prime_test.py +1 -1
  440. data/tracks/python/exercises/pascals-triangle/pascals_triangle_test.py +2 -2
  441. data/tracks/python/exercises/perfect-numbers/perfect_numbers_test.py +11 -11
  442. data/tracks/python/exercises/phone-number/phone_number_test.py +1 -1
  443. data/tracks/python/exercises/prime-factors/prime_factors_test.py +1 -1
  444. data/tracks/python/exercises/pythagorean-triplet/pythagorean_triplet_test.py +3 -3
  445. data/tracks/python/exercises/queen-attack/queen_attack_test.py +8 -8
  446. data/tracks/python/exercises/rail-fence-cipher/rail_fence_cipher_test.py +1 -1
  447. data/tracks/python/exercises/rectangles/rectangles_test.py +1 -1
  448. data/tracks/python/exercises/robot-simulator/robot_simulator.py +9 -1
  449. data/tracks/python/exercises/roman-numerals/roman_numerals_test.py +1 -1
  450. data/tracks/python/exercises/run-length-encoding/run_length_encoding_test.py +1 -1
  451. data/tracks/python/exercises/say/say_test.py +1 -1
  452. data/tracks/python/exercises/scrabble-score/scrabble_score_test.py +1 -1
  453. data/tracks/python/exercises/series/series.py +1 -1
  454. data/tracks/python/exercises/sieve/sieve_test.py +1 -1
  455. data/tracks/python/exercises/sublist/sublist_test.py +1 -1
  456. data/tracks/python/exercises/tournament/tournament_test.py +1 -1
  457. data/tracks/python/exercises/transpose/transpose_test.py +1 -1
  458. data/tracks/python/exercises/two-bucket/example.py +49 -54
  459. data/tracks/python/exercises/two-bucket/two_bucket_test.py +1 -1
  460. data/tracks/python/exercises/variable-length-quantity/variable_length_quantity_test.py +1 -1
  461. data/tracks/rust/config.json +24 -0
  462. data/tracks/rust/exercises/collatz-conjecture/Cargo.lock +4 -0
  463. data/tracks/rust/exercises/collatz-conjecture/Cargo.toml +3 -0
  464. data/tracks/rust/exercises/collatz-conjecture/README.md +65 -0
  465. data/tracks/rust/exercises/collatz-conjecture/example.rs +19 -0
  466. data/tracks/rust/exercises/collatz-conjecture/src/lib.rs +4 -0
  467. data/tracks/rust/exercises/collatz-conjecture/tests/collatz-conjecture.rs +32 -0
  468. data/tracks/rust/exercises/decimal/.gitignore +7 -0
  469. data/tracks/rust/exercises/decimal/.meta/description.md +19 -0
  470. data/tracks/rust/exercises/decimal/.meta/metadata.yml +3 -0
  471. data/tracks/rust/exercises/decimal/Cargo-example.toml +9 -0
  472. data/tracks/rust/exercises/decimal/Cargo.toml +6 -0
  473. data/tracks/rust/exercises/decimal/README.md +59 -0
  474. data/tracks/rust/exercises/decimal/example.rs +187 -0
  475. data/tracks/rust/exercises/decimal/src/lib.rs +10 -0
  476. data/tracks/rust/exercises/decimal/tests/decimal.rs +332 -0
  477. data/tracks/sml/exercises/accumulate/README.md +9 -1
  478. data/tracks/swift/exercises/luhn/Sources/LuhnExample.swift +17 -26
  479. data/tracks/swift/exercises/luhn/Tests/LinuxMain.swift +1 -1
  480. data/tracks/swift/exercises/luhn/Tests/LuhnTests/LuhnTests.swift +59 -35
  481. data/tracks/typescript/config.json +34 -0
  482. data/tracks/typescript/exercises/binary-search/README.md +67 -0
  483. data/tracks/typescript/exercises/binary-search/binary-search.example.ts +42 -0
  484. data/tracks/typescript/exercises/binary-search/binary-search.test.ts +27 -0
  485. data/tracks/typescript/exercises/binary-search/binary-search.ts +3 -0
  486. data/tracks/typescript/exercises/binary-search/package.json +36 -0
  487. data/tracks/typescript/exercises/binary-search/tsconfig.json +22 -0
  488. data/tracks/typescript/exercises/binary-search/tslint.json +127 -0
  489. data/tracks/typescript/exercises/binary-search/yarn.lock +2305 -0
  490. data/tracks/typescript/exercises/pascals-triangle/README.md +47 -0
  491. data/tracks/typescript/exercises/pascals-triangle/package.json +36 -0
  492. data/tracks/typescript/exercises/pascals-triangle/pascals-triangle.example.ts +32 -0
  493. data/tracks/typescript/exercises/pascals-triangle/pascals-triangle.test.ts +28 -0
  494. data/tracks/typescript/exercises/pascals-triangle/pascals-triangle.ts +2 -0
  495. data/tracks/typescript/exercises/pascals-triangle/tsconfig.json +22 -0
  496. data/tracks/typescript/exercises/pascals-triangle/tslint.json +127 -0
  497. data/tracks/typescript/exercises/pascals-triangle/yarn.lock +2305 -0
  498. data/tracks/typescript/exercises/rotational-cipher/README.md +63 -0
  499. data/tracks/typescript/exercises/rotational-cipher/package.json +36 -0
  500. data/tracks/typescript/exercises/rotational-cipher/rotational-cipher.example.ts +19 -0
  501. data/tracks/typescript/exercises/rotational-cipher/rotational-cipher.test.ts +56 -0
  502. data/tracks/typescript/exercises/rotational-cipher/rotational-cipher.ts +1 -0
  503. data/tracks/typescript/exercises/rotational-cipher/tsconfig.json +22 -0
  504. data/tracks/typescript/exercises/rotational-cipher/tslint.json +127 -0
  505. data/tracks/typescript/exercises/rotational-cipher/yarn.lock +2305 -0
  506. metadata +171 -5
  507. data/tracks/javascript/exercises/twelve-days/package.json +0 -69
  508. data/tracks/kotlin/exercises/hello-world/GETTING_STARTED.md +0 -50
@@ -0,0 +1,3 @@
1
+ ---
2
+ blurb: "Implement a Decimal type"
3
+ source: "Peter Goodspeed-Niklaus"
@@ -0,0 +1,9 @@
1
+ [package]
2
+ name = "decimal"
3
+ version = "0.1.0"
4
+ authors = ["Peter Goodspeed-Niklaus <peter.r.goodspeedniklaus@gmail.com>"]
5
+
6
+ [dependencies]
7
+ try_opt = "0.1.1"
8
+ num-bigint = "0.1.40"
9
+ num-traits = "0.1.40"
@@ -0,0 +1,6 @@
1
+ [package]
2
+ name = "decimal"
3
+ version = "0.1.0"
4
+ authors = ["Peter Goodspeed-Niklaus <peter.r.goodspeedniklaus@gmail.com>"]
5
+
6
+ [dependencies]
@@ -0,0 +1,59 @@
1
+ # Decimal
2
+
3
+ Implement an arbitrary-precision `Decimal` class.
4
+
5
+ Floating point numbers are the most common representation of non-integer real numbers in computing, and they're a common standard defined by [IEEE 754](https://en.wikipedia.org/wiki/IEEE_754). They're very flexible and versatile, but they do have some limitations. Famously, in floating point arithmetic, [`0.1 + 0.2 != 0.3`](http://0.30000000000000004.com/).
6
+
7
+ The solution to this issue is to find another, lossless way to model arbitrary-precision non-integer reals. This may be less efficient in terms of memory or processing speed than floating point numbers; the goal is to provide exact results.
8
+
9
+ Despite `Decimal` being a custom type, we should still be able to treat them as numbers: the `==`, `<`, `>`, `+`, `-`, and `*` operators should all work as expected on Decimals. For expediency, you are not required to implement division, as arbitrary-precision division can very quickly get out of hand. (How do you represent arbitrary-precision `1/3`?)
10
+
11
+ In Rust, the way to get these operations on custom types is to implement the relevant traits for your custom object. In particular, you'll need to implement at least `PartialEq`, `PartialOrd`, `Add`, `Sub`, and `Mul`. Strictly speaking, given that the decimal numbers form a total ordering, you should also implement `Eq` and `Ord`, though those traits are not checked for by these tests.
12
+
13
+ # Note
14
+
15
+ It would be very easy to implement this exercise by using the [bigdecimal](https://crates.io/crates/bigdecimal) crate. Don't do that; implement this yourself.
16
+
17
+ # Hints
18
+
19
+ - Instead of implementing arbitrary-precision arithmetic from scratch, consider building your type on top of the [num_bigint](https://crates.io/crates/num-bigint) crate.
20
+ - You might be able to [derive](https://doc.rust-lang.org/book/first-edition/traits.html#deriving) some of the required traits.
21
+ - `Decimal` is assumed to be a signed type. You do not have to create a separate unsigned type, though you may do so as an implementation detail if you so choose.
22
+
23
+ ## Rust Installation
24
+
25
+ Refer to the [exercism help page][help-page] for Rust installation and learning
26
+ resources.
27
+
28
+ ## Writing the Code
29
+
30
+ Execute the tests with:
31
+
32
+ ```bash
33
+ $ cargo test
34
+ ```
35
+
36
+ All but the first test have been ignored. After you get the first test to
37
+ pass, remove the ignore flag (`#[ignore]`) from the next test and get the tests
38
+ to pass again. The test file is located in the `tests` directory. You can
39
+ also remove the ignore flag from all the tests to get them to run all at once
40
+ if you wish.
41
+
42
+ Make sure to read the [Crates and Modules](https://doc.rust-lang.org/stable/book/crates-and-modules.html) chapter if you
43
+ haven't already, it will help you with organizing your files.
44
+
45
+ ## Feedback, Issues, Pull Requests
46
+
47
+ The [exercism/rust](https://github.com/exercism/rust) repository on GitHub is the home for all of the Rust exercises. If you have feedback about an exercise, or want to help implement new exercises, head over there and create an issue. Members of the [rust track team](https://github.com/orgs/exercism/teams/rust) are happy to help!
48
+
49
+ If you want to know more about Exercism, take a look at the [contribution guide](https://github.com/exercism/docs/blob/master/contributing-to-language-tracks/README.md).
50
+
51
+ [help-page]: http://exercism.io/languages/rust
52
+ [crates-and-modules]: http://doc.rust-lang.org/stable/book/crates-and-modules.html
53
+
54
+ ## Source
55
+
56
+ Peter Goodspeed-Niklaus
57
+
58
+ ## Submitting Incomplete Solutions
59
+ It's possible to submit an incomplete solution so you can see how others have completed the exercise.
@@ -0,0 +1,187 @@
1
+ use std::cmp::Ordering;
2
+ use std::fmt;
3
+ use std::ops::{Add, Sub, Mul};
4
+
5
+ #[macro_use]
6
+ extern crate try_opt;
7
+
8
+ extern crate num_bigint;
9
+ use num_bigint::BigInt;
10
+ extern crate num_traits;
11
+ use num_traits::pow;
12
+
13
+ /// Type implementing arbitrary-precision decimal arithmetic
14
+ #[derive(Debug, Eq, Clone)]
15
+ pub struct Decimal {
16
+ digits: BigInt,
17
+ decimal_index: usize,
18
+ }
19
+
20
+ impl Decimal {
21
+ fn new(digits: BigInt, decimal_index: usize) -> Decimal {
22
+ let mut value = Decimal {
23
+ digits: digits,
24
+ decimal_index: decimal_index,
25
+ };
26
+ value.reduce();
27
+ value
28
+ }
29
+
30
+ pub fn try_from(mut input: &str) -> Option<Decimal> {
31
+ // clear extraneous whitespace
32
+ input = input.trim();
33
+
34
+ // don't bother to trim extraneous zeroes
35
+ // leave it to users to manage their own memory
36
+
37
+ // now build a representation of the number to parse
38
+ let mut digits = String::with_capacity(input.len());
39
+ let mut decimal_index = None;
40
+ for ch in input.chars() {
41
+ match ch {
42
+ '0'...'9' | '-' | '+' => {
43
+ digits.push(ch);
44
+ if let Some(idx) = decimal_index.as_mut() {
45
+ *idx += 1;
46
+ }
47
+ }
48
+ '.' => {
49
+ if decimal_index.is_some() {
50
+ return None;
51
+ }
52
+ decimal_index = Some(0)
53
+ }
54
+ _ => return None,
55
+ }
56
+ }
57
+ Some(Decimal::new(
58
+ try_opt!(digits.parse::<BigInt>().ok()),
59
+ match decimal_index {
60
+ Some(idx) => idx,
61
+ None => 0,
62
+ },
63
+ ))
64
+ }
65
+
66
+ /// Add precision to the less-precise value until precisions match
67
+ ///
68
+ /// Precision, in this case, is defined as the decimal index.
69
+ fn equalize_precision(mut one: &mut Decimal, mut two: &mut Decimal) {
70
+ fn expand(lower_precision: &mut Decimal, higher_precision: &Decimal) {
71
+ let precision_difference =
72
+ (higher_precision.decimal_index - lower_precision.decimal_index) as usize;
73
+
74
+ lower_precision.digits = &lower_precision.digits *
75
+ pow(BigInt::from(10_usize), precision_difference);
76
+ lower_precision.decimal_index += precision_difference;
77
+ }
78
+ if one.decimal_index < two.decimal_index {
79
+ expand(&mut one, &two)
80
+ } else if one.decimal_index > two.decimal_index {
81
+ expand(&mut two, &one)
82
+ }
83
+ assert_eq!(one.decimal_index, two.decimal_index);
84
+ }
85
+
86
+ /// Eliminate extraneous trailing zeroes
87
+ ///
88
+ /// This reduces the decimal index, so that the raw values are easier to parse
89
+ fn reduce(&mut self) {
90
+ let extra_zeroes = self.digits
91
+ .to_string() // produce a decimal representation
92
+ .chars()
93
+ .rev() // trailing values
94
+ .take(self.decimal_index) // not counting past the decimal point
95
+ .take_while(|&c| c == '0') // counting only `0` digits
96
+ .count();
97
+ self.digits = &self.digits / pow(BigInt::from(10_usize), extra_zeroes);
98
+ self.decimal_index -= extra_zeroes;
99
+ }
100
+ }
101
+
102
+ macro_rules! auto_impl_decimal_ops {
103
+ ($trait:ident, $func_name:ident, $digits_operation:expr, $index_operation:expr) => {
104
+ impl $trait for Decimal {
105
+ type Output = Self;
106
+ fn $func_name(mut self, mut rhs: Self) -> Self {
107
+ Decimal::equalize_precision(&mut self, &mut rhs);
108
+ Decimal::new(
109
+ $digits_operation(self.digits, rhs.digits),
110
+ $index_operation(self.decimal_index, rhs.decimal_index),
111
+ )
112
+ }
113
+ }
114
+ }
115
+ }
116
+
117
+ auto_impl_decimal_ops!(Add, add, |s, o| s + o, |s, _| s);
118
+ auto_impl_decimal_ops!(Sub, sub, |s, o| s - o, |s, _| s);
119
+ auto_impl_decimal_ops!(Mul, mul, |s, o| s * o, |s, o| s + o);
120
+
121
+ macro_rules! auto_impl_decimal_cow {
122
+ ($trait:ident, $func_name:ident, $digits_operation:expr, $return_type:ty) => {
123
+ impl $trait for Decimal {
124
+ fn $func_name(&self, other: &Self) -> $return_type {
125
+ if self.decimal_index == other.decimal_index {
126
+ $digits_operation(&self.digits, &other.digits)
127
+ } else {
128
+ // if we're here, the decimal indexes are unmatched.
129
+ // We have to compare equal terms.
130
+ // clone both sides so we can modify either of them as necessary.
131
+ // not efficient, but whatever.
132
+ let mut one = self.clone();
133
+ let mut two = other.clone();
134
+ Decimal::equalize_precision(&mut one, &mut two);
135
+ one.$func_name(&two)
136
+ }
137
+ }
138
+ }
139
+ }
140
+ }
141
+
142
+ auto_impl_decimal_cow!(PartialEq, eq, |a, b| a == b, bool);
143
+ auto_impl_decimal_cow!(Ord, cmp, |a: &BigInt, b: &BigInt| a.cmp(b), Ordering);
144
+
145
+ impl PartialOrd for Decimal {
146
+ fn partial_cmp(&self, other: &Decimal) -> Option<Ordering> {
147
+ Some(self.cmp(other))
148
+ }
149
+ }
150
+
151
+ impl fmt::Display for Decimal {
152
+ fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
153
+ // get a representation of the pure digits,
154
+ // left-padded with zeroes
155
+ let digits = format!("{:0>width$}", self.digits, width = self.decimal_index);
156
+ if self.decimal_index == digits.len() {
157
+ write!(f, "0.{}", digits)
158
+ } else if self.decimal_index == 0 {
159
+ write!(f, "{}", digits)
160
+ } else {
161
+ let (before_index, after_index) = digits.split_at(digits.len() - self.decimal_index);
162
+ write!(f, "{}.{}", before_index, after_index)
163
+ }
164
+ }
165
+ }
166
+
167
+ #[cfg(test)]
168
+ mod tests {
169
+ use super::*;
170
+
171
+ #[test]
172
+ fn test_display_temp() {
173
+ for test_str in vec!["0", "1", "20", "0.3", "0.04", "50.05", "66.0006", "0.007"] {
174
+ println!(
175
+ "Decimal representation of \"{}\": {}",
176
+ test_str,
177
+ Decimal::try_from(test_str).expect("This should always become a decimal")
178
+ );
179
+ assert_eq!(
180
+ test_str,
181
+ Decimal::try_from(test_str)
182
+ .expect("This should always become a decimal")
183
+ .to_string()
184
+ )
185
+ }
186
+ }
187
+ }
@@ -0,0 +1,10 @@
1
+ /// Type implementing arbitrary-precision decimal arithmetic
2
+ pub struct Decimal {
3
+ // implement your type here
4
+ }
5
+
6
+ impl Decimal {
7
+ pub fn try_from(_: &str) -> Option<Decimal> {
8
+ unimplemented!()
9
+ }
10
+ }
@@ -0,0 +1,332 @@
1
+ extern crate decimal;
2
+ use decimal::Decimal;
3
+
4
+ /// Create a Decimal from a string literal
5
+ ///
6
+ /// Use only when you _know_ that your value is valid.
7
+ fn decimal(input: &str) -> Decimal {
8
+ Decimal::try_from(input).expect("That was supposed to be a valid value")
9
+ }
10
+
11
+ /// Some big and precise values we can use for testing. [0] + [1] == [2]
12
+ const BIGS: [&'static str; 3] = [
13
+ "100000000000000000000000000000000000000000000.00000000000000000000000000000000000000001",
14
+ "100000000000000000000000000000000000000000000.00000000000000000000000000000000000000002",
15
+ "200000000000000000000000000000000000000000000.00000000000000000000000000000000000000003",
16
+ ];
17
+
18
+ // test simple properties of required operations
19
+ #[test]
20
+ fn test_eq() {
21
+ assert!(decimal("0.0") == decimal("0.0"));
22
+ assert!(decimal("1.0") == decimal("1.0"));
23
+ for big in BIGS.iter() {
24
+ assert!(decimal(big) == decimal(big));
25
+ }
26
+ }
27
+
28
+ #[test]
29
+ #[ignore]
30
+ fn test_ne() {
31
+ assert!(decimal("0.0") != decimal("1.0"));
32
+ assert!(decimal(BIGS[0]) != decimal(BIGS[1]));
33
+ }
34
+
35
+ #[test]
36
+ #[ignore]
37
+ fn test_gt() {
38
+ for slice_2 in BIGS.windows(2) {
39
+ assert!(decimal(slice_2[1]) > decimal(slice_2[0]));
40
+ assert!(!(decimal(slice_2[0]) > decimal(slice_2[1])));
41
+ }
42
+ }
43
+
44
+ #[test]
45
+ #[ignore]
46
+ fn test_lt() {
47
+ for slice_2 in BIGS.windows(2) {
48
+ assert!(decimal(slice_2[0]) < decimal(slice_2[1]));
49
+ assert!(!(decimal(slice_2[1]) < decimal(slice_2[0])));
50
+ }
51
+ }
52
+
53
+ #[test]
54
+ #[ignore]
55
+ fn test_add() {
56
+ assert_eq!(decimal("0.1") + decimal("0.2"), decimal("0.3"));
57
+ assert_eq!(decimal(BIGS[0]) + decimal(BIGS[1]), decimal(BIGS[2]));
58
+ assert_eq!(decimal(BIGS[1]) + decimal(BIGS[0]), decimal(BIGS[2]));
59
+ }
60
+
61
+ #[test]
62
+ #[ignore]
63
+ fn test_sub() {
64
+ assert_eq!(decimal(BIGS[2]) - decimal(BIGS[1]), decimal(BIGS[0]));
65
+ assert_eq!(decimal(BIGS[2]) - decimal(BIGS[0]), decimal(BIGS[1]));
66
+ }
67
+
68
+ #[test]
69
+ #[ignore]
70
+ fn test_mul() {
71
+ for big in BIGS.iter() {
72
+ assert_eq!(decimal(big) * decimal("2"), decimal(big) + decimal(big));
73
+ }
74
+ }
75
+
76
+ // test identities
77
+ #[test]
78
+ #[ignore]
79
+ fn test_add_id() {
80
+ assert_eq!(decimal("1.0") + decimal("0.0"), decimal("1.0"));
81
+ assert_eq!(decimal("0.1") + decimal("0.0"), decimal("0.1"));
82
+ assert_eq!(decimal("0.0") + decimal("1.0"), decimal("1.0"));
83
+ assert_eq!(decimal("0.0") + decimal("0.1"), decimal("0.1"));
84
+ }
85
+
86
+ #[test]
87
+ #[ignore]
88
+ fn test_sub_id() {
89
+ assert_eq!(decimal("1.0") - decimal("0.0"), decimal("1.0"));
90
+ assert_eq!(decimal("0.1") - decimal("0.0"), decimal("0.1"));
91
+ }
92
+
93
+ #[test]
94
+ #[ignore]
95
+ fn test_mul_id() {
96
+ assert_eq!(decimal("2.1") * decimal("1.0"), decimal("2.1"));
97
+ assert_eq!(decimal("1.0") * decimal("2.1"), decimal("2.1"));
98
+ }
99
+
100
+ #[test]
101
+ #[ignore]
102
+ fn test_gt_positive_and_zero() {
103
+ assert!(decimal("1.0") > decimal("0.0"));
104
+ assert!(decimal("0.1") > decimal("0.0"));
105
+ }
106
+
107
+ #[test]
108
+ #[ignore]
109
+ fn test_gt_negative_and_zero() {
110
+ assert!(decimal("0.0") > decimal("-0.1"));
111
+ assert!(decimal("0.0") > decimal("-1.0"));
112
+ }
113
+
114
+
115
+ // tests of arbitrary precision behavior
116
+ #[test]
117
+ #[ignore]
118
+ fn test_add_uneven_position() {
119
+ assert_eq!(decimal("0.1") + decimal("0.02"), decimal("0.12"));
120
+ }
121
+
122
+ #[test]
123
+ #[ignore]
124
+ fn test_eq_vary_sig_digits() {
125
+ assert!(decimal("0") == decimal("0000000000000.0000000000000000000000"));
126
+ assert!(decimal("1") == decimal("00000000000000001.000000000000000000"));
127
+ }
128
+
129
+ #[test]
130
+ #[ignore]
131
+ fn test_add_vary_precision() {
132
+ assert_eq!(
133
+ decimal("100000000000000000000000000000000000000000000") +
134
+ decimal("0.00000000000000000000000000000000000000001"),
135
+ decimal(BIGS[0])
136
+ )
137
+ }
138
+
139
+ #[test]
140
+ #[ignore]
141
+ fn test_cleanup_precision() {
142
+ assert_eq!(
143
+ decimal(
144
+ "10000000000000000000000000000000000000000000000.999999999999999999999999998",
145
+ ) +
146
+ decimal(
147
+ "10000000000000000000000000000000000000000000000.000000000000000000000000002",
148
+ ),
149
+ decimal("20000000000000000000000000000000000000000000001")
150
+ )
151
+ }
152
+
153
+ #[test]
154
+ #[ignore]
155
+ fn test_gt_varying_positive_precisions() {
156
+ assert!(decimal("1.1") > decimal("1.01"));
157
+ assert!(decimal("1.01") > decimal("1.0"));
158
+ assert!(decimal("1.0") > decimal("0.1"));
159
+ assert!(decimal("0.1") > decimal("0.01"));
160
+ }
161
+
162
+ #[test]
163
+ #[ignore]
164
+ fn test_gt_positive_and_negative() {
165
+ assert!(decimal("1.0") > decimal("-1.0"));
166
+ assert!(decimal("1.1") > decimal("-1.1"));
167
+ assert!(decimal("0.1") > decimal("-0.1"));
168
+ }
169
+
170
+ #[test]
171
+ #[ignore]
172
+ fn test_gt_varying_negative_precisions() {
173
+ assert!(decimal("-0.01") > decimal("-0.1"));
174
+ assert!(decimal("-0.1") > decimal("-1.0"));
175
+ assert!(decimal("-1.0") > decimal("-1.01"));
176
+ assert!(decimal("-1.01") > decimal("-1.1"));
177
+ }
178
+
179
+ // test signed properties
180
+ #[test]
181
+ #[ignore]
182
+ fn test_negatives() {
183
+ assert!(Decimal::try_from("-1").is_some());
184
+ assert_eq!(decimal("0") - decimal("1"), decimal("-1"));
185
+ assert_eq!(decimal("5.5") + decimal("-6.5"), decimal("-1"));
186
+ }
187
+
188
+ #[test]
189
+ #[ignore]
190
+ fn test_explicit_positive() {
191
+ assert_eq!(decimal("+1"), decimal("1"));
192
+ assert_eq!(decimal("+2.0") - decimal("-0002.0"), decimal("4"));
193
+ }
194
+
195
+ #[test]
196
+ #[ignore]
197
+ fn test_multiply_by_negative() {
198
+ assert_eq!(decimal("5") * decimal("-0.2"), decimal("-1"));
199
+ assert_eq!(decimal("-20") * decimal("-0.2"), decimal("4"));
200
+ }
201
+
202
+ #[test]
203
+ #[ignore]
204
+ fn test_simple_partial_cmp() {
205
+ assert!(decimal("1.0") < decimal("1.1"));
206
+ assert!(decimal("0.00000000000000000000001") > decimal("-20000000000000000000000000000"));
207
+ }
208
+
209
+ // test carrying rules
210
+ // these tests are designed to ensure correctness of implementations for which the
211
+ // integer and fractional parts of the number are stored separately
212
+ #[test]
213
+ #[ignore]
214
+ fn test_carry_into_integer() {
215
+ assert_eq!(decimal("0.901") + decimal("0.1"), decimal("1.001"))
216
+ }
217
+
218
+ #[test]
219
+ #[ignore]
220
+ fn test_carry_into_fractional_with_digits_to_right() {
221
+ assert_eq!(decimal("0.0901") + decimal("0.01"), decimal("0.1001"))
222
+ }
223
+
224
+ #[test]
225
+ #[ignore]
226
+ fn test_add_carry_over_negative() {
227
+ assert_eq!(decimal("-1.99") + decimal("-0.01"), decimal("-2.0"))
228
+ }
229
+
230
+ #[test]
231
+ #[ignore]
232
+ fn test_sub_carry_over_negative() {
233
+ assert_eq!(decimal("-1.99") - decimal("0.01"), decimal("-2.0"))
234
+ }
235
+
236
+ #[test]
237
+ #[ignore]
238
+ fn test_add_carry_over_negative_with_fractional() {
239
+ assert_eq!(decimal("-1.99") + decimal("-0.02"), decimal("-2.01"))
240
+ }
241
+
242
+ #[test]
243
+ #[ignore]
244
+ fn test_sub_carry_over_negative_with_fractional() {
245
+ assert_eq!(decimal("-1.99") - decimal("0.02"), decimal("-2.01"))
246
+ }
247
+
248
+ #[test]
249
+ #[ignore]
250
+ fn test_carry_from_rightmost_one() {
251
+ assert_eq!(decimal("0.09") + decimal("0.01"), decimal("0.1"))
252
+ }
253
+
254
+ #[test]
255
+ #[ignore]
256
+ fn test_carry_from_rightmost_more() {
257
+ assert_eq!(decimal("0.099") + decimal("0.001"), decimal("0.1"))
258
+ }
259
+
260
+ #[test]
261
+ #[ignore]
262
+ fn test_carry_from_rightmost_into_integer() {
263
+ assert_eq!(decimal("0.999") + decimal("0.001"), decimal("1.0"))
264
+ }
265
+
266
+ // test arithmetic borrow rules
267
+ #[test]
268
+ #[ignore]
269
+ fn test_add_borrow() {
270
+ assert_eq!(decimal("0.01") + decimal("-0.0001"), decimal("0.0099"))
271
+ }
272
+
273
+ #[test]
274
+ #[ignore]
275
+ fn test_sub_borrow() {
276
+ assert_eq!(decimal("0.01") - decimal("0.0001"), decimal("0.0099"))
277
+ }
278
+
279
+ #[test]
280
+ #[ignore]
281
+ fn test_add_borrow_integral() {
282
+ assert_eq!(decimal("1.0") + decimal("-0.01"), decimal("0.99"))
283
+ }
284
+
285
+ #[test]
286
+ #[ignore]
287
+ fn test_sub_borrow_integral() {
288
+ assert_eq!(decimal("1.0") - decimal("0.01"), decimal("0.99"))
289
+ }
290
+
291
+ #[test]
292
+ #[ignore]
293
+ fn test_add_borrow_integral_zeroes() {
294
+ assert_eq!(decimal("1.0") + decimal("-0.99"), decimal("0.01"))
295
+ }
296
+
297
+ #[test]
298
+ #[ignore]
299
+ fn test_sub_borrow_integral_zeroes() {
300
+ assert_eq!(decimal("1.0") - decimal("0.99"), decimal("0.01"))
301
+ }
302
+
303
+ #[test]
304
+ #[ignore]
305
+ fn test_borrow_from_negative() {
306
+ assert_eq!(decimal("-1.0") + decimal("0.01"), decimal("-0.99"))
307
+ }
308
+
309
+ #[test]
310
+ #[ignore]
311
+ fn test_add_into_fewer_digits() {
312
+ assert_eq!(decimal("0.011") + decimal("-0.001"), decimal("0.01"))
313
+ }
314
+
315
+ // misc tests of arithmetic properties
316
+ #[test]
317
+ #[ignore]
318
+ fn test_sub_into_fewer_digits() {
319
+ assert_eq!(decimal("0.011") - decimal("0.001"), decimal("0.01"))
320
+ }
321
+
322
+ #[test]
323
+ #[ignore]
324
+ fn test_add_away_decimal() {
325
+ assert_eq!(decimal("1.1") + decimal("-0.1"), decimal("1.0"))
326
+ }
327
+
328
+ #[test]
329
+ #[ignore]
330
+ fn test_sub_away_decimal() {
331
+ assert_eq!(decimal("1.1") - decimal("0.1"), decimal("1.0"))
332
+ }