trackler 1.0.0 → 1.0.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (288) hide show
  1. checksums.yaml +4 -4
  2. data/.travis.yml +7 -0
  3. data/CHANGELOG.md +9 -1
  4. data/README.md +10 -0
  5. data/common/CONTRIBUTING.md +50 -2
  6. data/common/exercises/alphametics/canonical-data.json +10 -10
  7. data/common/exercises/markdown/canonical-data.json +4 -4
  8. data/common/exercises/pov/canonical-data.json +356 -0
  9. data/lib/trackler/version.rb +1 -1
  10. data/trackler.gemspec +1 -2
  11. data/tracks/c/config.json +11 -1
  12. data/tracks/c/exercises/binary-search/makefile +16 -0
  13. data/tracks/c/exercises/binary-search/src/example.c +21 -0
  14. data/tracks/c/exercises/binary-search/src/example.h +8 -0
  15. data/tracks/c/exercises/binary-search/test/test_binary_search.c +99 -0
  16. data/tracks/c/exercises/binary-search/test/vendor/unity.c +1300 -0
  17. data/tracks/c/exercises/binary-search/test/vendor/unity.h +274 -0
  18. data/tracks/c/exercises/binary-search/test/vendor/unity_internals.h +701 -0
  19. data/tracks/clojure/config.json +3 -2
  20. data/tracks/clojure/exercises/bank-account/project.clj +4 -0
  21. data/tracks/clojure/exercises/bank-account/src/example.clj +17 -0
  22. data/tracks/clojure/exercises/bank-account/test/bank_account_test.clj +44 -0
  23. data/tracks/clojure/exercises/isogram/project.clj +4 -0
  24. data/tracks/clojure/exercises/isogram/src/example.clj +5 -0
  25. data/tracks/clojure/exercises/isogram/test/isogram_test.clj +17 -0
  26. data/tracks/csharp/config.json +27 -87
  27. data/tracks/csharp/exercises/exercises.csproj +265 -1
  28. data/tracks/csharp/exercises/grep/Example.cs +114 -0
  29. data/tracks/csharp/exercises/grep/GrepTest.cs +376 -0
  30. data/tracks/csharp/exercises/paket.references +2 -1
  31. data/tracks/csharp/exercises/poker/Example.cs +96 -0
  32. data/tracks/csharp/exercises/poker/PokerTest.cs +171 -0
  33. data/tracks/csharp/exercises/sgf-parsing/Example.cs +61 -0
  34. data/tracks/csharp/exercises/sgf-parsing/SgfParsing.cs +13 -0
  35. data/tracks/csharp/exercises/sgf-parsing/SgfParsingTest.cs +158 -0
  36. data/tracks/csharp/paket.dependencies +2 -1
  37. data/tracks/csharp/paket.lock +72 -0
  38. data/tracks/ecmascript/exercises/simple-cipher/simple-cipher.spec.js +1 -1
  39. data/tracks/elisp/config.json +6 -1
  40. data/tracks/elisp/exercises/etl/etl-test.el +90 -0
  41. data/tracks/elisp/exercises/etl/etl.el +9 -0
  42. data/tracks/elisp/exercises/etl/example.el +22 -0
  43. data/tracks/elisp/exercises/nucleotide-count/example.el +19 -0
  44. data/tracks/elisp/exercises/nucleotide-count/nucleotide-count-test.el +34 -0
  45. data/tracks/elisp/exercises/nucleotide-count/nucleotide-count.el +9 -0
  46. data/tracks/elisp/exercises/perfect-numbers/example.el +28 -0
  47. data/tracks/elisp/exercises/perfect-numbers/perfect-numbers-test.el +24 -0
  48. data/tracks/elisp/exercises/perfect-numbers/perfect-numbers.el +10 -0
  49. data/tracks/elisp/exercises/roman-numerals/example.el +36 -0
  50. data/tracks/elisp/exercises/roman-numerals/roman-numerals-test.el +30 -0
  51. data/tracks/elisp/exercises/roman-numerals/roman-numerals.el +10 -0
  52. data/tracks/elisp/exercises/word-count/example.el +18 -0
  53. data/tracks/elisp/exercises/word-count/word-count-test.el +64 -0
  54. data/tracks/elisp/exercises/word-count/word-count.el +9 -0
  55. data/tracks/elixir/README.md +4 -4
  56. data/tracks/elixir/SETUP.md +3 -3
  57. data/tracks/elixir/docs/TESTS.md +3 -3
  58. data/tracks/elixir/exercises/accumulate/accumulate_test.exs +7 -7
  59. data/tracks/elixir/exercises/acronym/acronym_test.exs +5 -5
  60. data/tracks/elixir/exercises/allergies/allergies_test.exs +13 -13
  61. data/tracks/elixir/exercises/anagram/anagram_test.exs +12 -12
  62. data/tracks/elixir/exercises/atbash-cipher/atbash_cipher_test.exs +9 -9
  63. data/tracks/elixir/exercises/bank-account/bank_account_test.exs +5 -5
  64. data/tracks/elixir/exercises/beer-song/beer_song_test.exs +6 -6
  65. data/tracks/elixir/exercises/binary-search/binary_search_test.exs +7 -7
  66. data/tracks/elixir/exercises/binary/binary_test.exs +13 -13
  67. data/tracks/elixir/exercises/bob/bob_test.exs +15 -15
  68. data/tracks/elixir/exercises/bowling/bowling_test.exs +18 -18
  69. data/tracks/elixir/exercises/bracket-push/bracket_push_test.exs +12 -12
  70. data/tracks/elixir/exercises/change/change_test.exs +7 -7
  71. data/tracks/elixir/exercises/connect/connect_test.exs +9 -9
  72. data/tracks/elixir/exercises/crypto-square/crypto_square_test.exs +7 -7
  73. data/tracks/elixir/exercises/custom-set/custom_set_test.exs +11 -11
  74. data/tracks/elixir/exercises/diamond/diamond_test.exs +4 -4
  75. data/tracks/elixir/exercises/difference-of-squares/difference_of_squares_test.exs +10 -10
  76. data/tracks/elixir/exercises/dot-dsl/dot_dsl_test.exs +17 -17
  77. data/tracks/elixir/exercises/etl/etl_test.exs +5 -5
  78. data/tracks/elixir/exercises/flatten-array/flatten_array_test.exs +6 -6
  79. data/tracks/elixir/exercises/forth/forth_test.exs +16 -16
  80. data/tracks/elixir/exercises/gigasecond/gigasecond_test.exs +5 -5
  81. data/tracks/elixir/exercises/grade-school/grade_school_test.exs +6 -6
  82. data/tracks/elixir/exercises/grains/grains_test.exs +9 -9
  83. data/tracks/elixir/exercises/hamming/hamming_test.exs +6 -6
  84. data/tracks/elixir/exercises/hello-world/hello_world_test.exs +3 -3
  85. data/tracks/elixir/exercises/hexadecimal/hexadecimal_test.exs +11 -11
  86. data/tracks/elixir/exercises/isogram/isogram_test.exs +10 -10
  87. data/tracks/elixir/exercises/kindergarten-garden/garden_test.exs +7 -7
  88. data/tracks/elixir/exercises/largest-series-product/largest_series_product_test.exs +17 -17
  89. data/tracks/elixir/exercises/leap/leap_test.exs +5 -5
  90. data/tracks/elixir/exercises/list-ops/list_ops_test.exs +26 -26
  91. data/tracks/elixir/exercises/luhn/luhn_test.exs +7 -7
  92. data/tracks/elixir/exercises/markdown/markdown_test.exs +10 -10
  93. data/tracks/elixir/exercises/meetup/meetup_test.exs +92 -92
  94. data/tracks/elixir/exercises/minesweeper/minesweeper_test.exs +8 -8
  95. data/tracks/elixir/exercises/nth-prime/nth_prime_test.exs +6 -6
  96. data/tracks/elixir/exercises/nucleotide-count/example.exs +0 -5
  97. data/tracks/elixir/exercises/nucleotide-count/nucleotide_count_test.exs +7 -7
  98. data/tracks/elixir/exercises/palindrome-products/palindrome_products_test.exs +6 -6
  99. data/tracks/elixir/exercises/pangram/pangram_test.exs +11 -11
  100. data/tracks/elixir/exercises/parallel-letter-frequency/parallel_letter_frequency_test.exs +10 -10
  101. data/tracks/elixir/exercises/pascals-triangle/pascals_triangle_test.exs +7 -7
  102. data/tracks/elixir/exercises/phone-number/phone_number_test.exs +10 -10
  103. data/tracks/elixir/exercises/prime-factors/prime_factors_test.exs +13 -13
  104. data/tracks/elixir/exercises/pythagorean-triplet/pythagorean_triplet_test.exs +8 -8
  105. data/tracks/elixir/exercises/queen-attack/queen_attack_test.exs +16 -16
  106. data/tracks/elixir/exercises/rail-fence-cipher/rail_fence_cipher_test.exs +13 -13
  107. data/tracks/elixir/exercises/raindrops/raindrops_test.exs +17 -17
  108. data/tracks/elixir/exercises/rna-transcription/rna_transcription_test.exs +6 -6
  109. data/tracks/elixir/exercises/robot-simulator/robot_simulator_test.exs +6 -6
  110. data/tracks/elixir/exercises/roman-numerals/roman_numerals_test.exs +19 -19
  111. data/tracks/elixir/exercises/run-length-encoding/rle_test.exs +6 -6
  112. data/tracks/elixir/exercises/saddle-points/saddle_points_test.exs +11 -11
  113. data/tracks/elixir/exercises/scrabble-score/scrabble_score_test.exs +9 -9
  114. data/tracks/elixir/exercises/sieve/sieve_test.exs +3 -3
  115. data/tracks/elixir/exercises/space-age/space_age_test.exs +9 -9
  116. data/tracks/elixir/exercises/sublist/sublist_test.exs +20 -20
  117. data/tracks/elixir/exercises/sum-of-multiples/sum_of_multiples_test.exs +11 -11
  118. data/tracks/elixir/exercises/test_helper.exs +1 -1
  119. data/tracks/elixir/exercises/triangle/triangle_test.exs +16 -16
  120. data/tracks/elixir/exercises/word-count/word_count_test.exs +9 -9
  121. data/tracks/elixir/exercises/wordy/wordy_test.exs +16 -16
  122. data/tracks/elixir/exercises/zipper/zipper_test.exs +13 -13
  123. data/tracks/fsharp/build.fsx +2 -7
  124. data/tracks/fsharp/config.json +0 -104
  125. data/tracks/fsharp/exercises/grep/GrepTest.fs +7 -1
  126. data/tracks/go/config.json +1 -0
  127. data/tracks/go/exercises/acronym/acronym_test.go +36 -0
  128. data/tracks/go/exercises/acronym/example.go +21 -0
  129. data/tracks/go/exercises/palindrome-products/example.go +2 -0
  130. data/tracks/go/exercises/palindrome-products/palindrome_products_test.go +10 -0
  131. data/tracks/go/exercises/phone-number/example.go +2 -0
  132. data/tracks/go/exercises/phone-number/phone_number_test.go +44 -27
  133. data/tracks/go/exercises/robot-simulator/defs.go +6 -6
  134. data/tracks/go/exercises/robot-simulator/example.go +22 -22
  135. data/tracks/go/exercises/robot-simulator/robot_simulator_step2_test.go +28 -28
  136. data/tracks/go/exercises/robot-simulator/robot_simulator_step3_test.go +75 -76
  137. data/tracks/go/exercises/robot-simulator/robot_simulator_test.go +10 -4
  138. data/tracks/haskell/exercises/anagram/HINTS.md +22 -0
  139. data/tracks/haskell/exercises/anagram/package.yaml +3 -0
  140. data/tracks/haskell/exercises/anagram/src/Example.hs +10 -7
  141. data/tracks/haskell/exercises/anagram/test/Tests.hs +9 -3
  142. data/tracks/haskell/exercises/beer-song/HINTS.md +18 -0
  143. data/tracks/haskell/exercises/beer-song/src/Beer.hs +301 -3
  144. data/tracks/haskell/exercises/beer-song/src/Example.hs +5 -3
  145. data/tracks/haskell/exercises/beer-song/test/Tests.hs +332 -23
  146. data/tracks/haskell/exercises/crypto-square/src/CryptoSquare.hs +3 -17
  147. data/tracks/haskell/exercises/crypto-square/src/Example.hs +12 -20
  148. data/tracks/haskell/exercises/crypto-square/test/Tests.hs +46 -86
  149. data/tracks/haskell/exercises/leap/HINTS.md +11 -0
  150. data/tracks/haskell/exercises/leap/src/LeapYear.hs +1 -0
  151. data/tracks/java/exercises/pangram/src/test/java/PangramTest.java +34 -4
  152. data/tracks/lua/exercises/pov/pov_spec.lua +0 -13
  153. data/tracks/mips/docs/ABOUT.md +1 -1
  154. data/tracks/objective-c/config.json +225 -0
  155. data/tracks/objective-c/exercises/largest-series-product/LargestSeriesProductExample.h +8 -0
  156. data/tracks/objective-c/exercises/largest-series-product/LargestSeriesProductExample.m +63 -0
  157. data/tracks/objective-c/exercises/largest-series-product/LargestSeriesProductTest.m +84 -0
  158. data/tracks/objective-c/xcodeProject/ObjectiveC.xcodeproj/project.pbxproj +18 -0
  159. data/tracks/ocaml/.travis-ci.sh +23 -0
  160. data/tracks/ocaml/.travis.yml +5 -1
  161. data/tracks/ocaml/Makefile +41 -0
  162. data/tracks/ocaml/config.json +18 -0
  163. data/tracks/ocaml/exercises/anagram/Makefile +1 -0
  164. data/tracks/ocaml/exercises/beer-song/Makefile +1 -0
  165. data/tracks/ocaml/exercises/beer-song/{beer.mli → beer_song.mli} +0 -0
  166. data/tracks/ocaml/exercises/beer-song/example.ml +1 -1
  167. data/tracks/ocaml/exercises/beer-song/test.ml +7 -6
  168. data/tracks/ocaml/exercises/bob/Makefile +1 -0
  169. data/tracks/ocaml/exercises/bowling/Makefile +1 -0
  170. data/tracks/ocaml/exercises/custom-set/Makefile +1 -0
  171. data/tracks/ocaml/exercises/difference-of-squares/.merlin +5 -0
  172. data/tracks/ocaml/exercises/difference-of-squares/Makefile +11 -0
  173. data/tracks/ocaml/exercises/difference-of-squares/difference_of_squares.mli +8 -0
  174. data/tracks/ocaml/exercises/difference-of-squares/example.ml +21 -0
  175. data/tracks/ocaml/exercises/difference-of-squares/test.ml +32 -0
  176. data/tracks/ocaml/exercises/grade-school/Makefile +1 -0
  177. data/tracks/ocaml/exercises/grade-school/{school.mli → grade_school.mli} +0 -0
  178. data/tracks/ocaml/exercises/grade-school/test.ml +27 -27
  179. data/tracks/ocaml/exercises/hamming/Makefile +1 -0
  180. data/tracks/ocaml/exercises/hangman/Makefile +1 -0
  181. data/tracks/ocaml/exercises/hello-world/Makefile +1 -1
  182. data/tracks/ocaml/exercises/hello-world/{hello.ml → hello_world.ml} +0 -0
  183. data/tracks/ocaml/exercises/hello-world/{hello.mli → hello_world.mli} +0 -0
  184. data/tracks/ocaml/exercises/hello-world/test.ml +1 -1
  185. data/tracks/ocaml/exercises/hexadecimal/Makefile +1 -0
  186. data/tracks/ocaml/exercises/leap/Makefile +1 -1
  187. data/tracks/ocaml/exercises/list-ops/Makefile +1 -0
  188. data/tracks/ocaml/exercises/luhn/Makefile +1 -0
  189. data/tracks/ocaml/exercises/minesweeper/Makefile +1 -0
  190. data/tracks/ocaml/exercises/nucleotide-count/Makefile +1 -0
  191. data/tracks/ocaml/exercises/nucleotide-count/{dna.mli → nucleotide_count.mli} +0 -0
  192. data/tracks/ocaml/exercises/nucleotide-count/test.ml +7 -7
  193. data/tracks/ocaml/exercises/phone-number/Makefile +1 -0
  194. data/tracks/ocaml/exercises/phone-number/{phone.mli → phone_number.mli} +0 -0
  195. data/tracks/ocaml/exercises/phone-number/test.ml +10 -10
  196. data/tracks/ocaml/exercises/point-mutations/Makefile +8 -0
  197. data/tracks/ocaml/exercises/point-mutations/example.ml +0 -0
  198. data/tracks/ocaml/exercises/prime-factors/Makefile +1 -0
  199. data/tracks/ocaml/exercises/raindrops/.merlin +5 -0
  200. data/tracks/ocaml/exercises/raindrops/Makefile +11 -0
  201. data/tracks/ocaml/exercises/raindrops/example.ml +11 -0
  202. data/tracks/ocaml/exercises/raindrops/raindrops.mli +7 -0
  203. data/tracks/ocaml/exercises/raindrops/test.ml +48 -0
  204. data/tracks/ocaml/exercises/rna-transcription/Makefile +1 -0
  205. data/tracks/ocaml/exercises/rna-transcription/{dna.mli → rna_transcription.mli} +0 -0
  206. data/tracks/ocaml/exercises/rna-transcription/test.ml +6 -6
  207. data/tracks/ocaml/exercises/roman-numerals/Makefile +1 -0
  208. data/tracks/ocaml/exercises/say/.gitignore +1 -0
  209. data/tracks/ocaml/exercises/say/.merlin +5 -0
  210. data/tracks/ocaml/exercises/say/Makefile +11 -0
  211. data/tracks/ocaml/exercises/say/example.ml +52 -0
  212. data/tracks/ocaml/exercises/say/say.mli +8 -0
  213. data/tracks/ocaml/exercises/say/test.ml +65 -0
  214. data/tracks/ocaml/exercises/space-age/Makefile +1 -0
  215. data/tracks/ocaml/exercises/word-count/Makefile +1 -0
  216. data/tracks/ocaml/exercises/word-count/example.ml +1 -1
  217. data/tracks/ocaml/exercises/zipper/HINT.md +5 -6
  218. data/tracks/ocaml/exercises/zipper/Makefile +1 -0
  219. data/tracks/ocaml/exercises/zipper/example.ml +2 -2
  220. data/tracks/ocaml/exercises/zipper/tree.ml +1 -1
  221. data/tracks/php/config.json +7 -1
  222. data/tracks/php/exercises/variable-length-quantity/example.php +56 -0
  223. data/tracks/php/exercises/variable-length-quantity/variable-length-quantity_test.php +149 -0
  224. data/tracks/python/config.json +1 -1
  225. data/tracks/ruby/.rubocop.yml +1915 -26
  226. data/tracks/ruby/bin/generate-acronym +7 -0
  227. data/tracks/ruby/bin/generate-nth-prime +7 -0
  228. data/tracks/ruby/config.json +482 -0
  229. data/tracks/ruby/exercises/acronym/.version +1 -0
  230. data/tracks/ruby/exercises/acronym/acronym_test.rb +45 -21
  231. data/tracks/ruby/exercises/acronym/example.rb +1 -1
  232. data/tracks/ruby/exercises/acronym/example.tt +19 -0
  233. data/tracks/ruby/exercises/anagram/anagram_test.rb +1 -1
  234. data/tracks/ruby/exercises/nth-prime/.version +1 -0
  235. data/tracks/ruby/exercises/nth-prime/example.rb +4 -0
  236. data/tracks/ruby/exercises/nth-prime/example.tt +23 -0
  237. data/tracks/ruby/exercises/nth-prime/nth_prime_test.rb +33 -10
  238. data/tracks/ruby/exercises/raindrops/.version +1 -1
  239. data/tracks/ruby/exercises/raindrops/example.rb +1 -1
  240. data/tracks/ruby/exercises/raindrops/raindrops_test.rb +20 -5
  241. data/tracks/ruby/lib/acronym_cases.rb +19 -0
  242. data/tracks/ruby/lib/generator.rb +3 -2
  243. data/tracks/ruby/lib/nth_prime_cases.rb +23 -0
  244. data/tracks/ruby/lib/raindrop_cases.rb +1 -1
  245. data/tracks/rust/README.md +1 -1
  246. data/tracks/scala/exercises/leap/src/test/scala/leap_test.scala +10 -0
  247. data/tracks/scala/exercises/prime-factors/example.scala +2 -5
  248. data/tracks/scala/exercises/prime-factors/src/test/scala/primefactors_test.scala +11 -13
  249. data/tracks/scheme/{anagram → exercises/anagram}/anagram-test.scm +0 -0
  250. data/tracks/scheme/{anagram → exercises/anagram}/anagram.scm +0 -0
  251. data/tracks/scheme/{anagram → exercises/anagram}/example.scm +0 -0
  252. data/tracks/scheme/{bob → exercises/bob}/bob-test.scm +0 -0
  253. data/tracks/scheme/{bob → exercises/bob}/bob.scm +0 -0
  254. data/tracks/scheme/{bob → exercises/bob}/example.scm +0 -0
  255. data/tracks/scheme/{difference-of-squares → exercises/difference-of-squares}/difference-of-squares-test.scm +0 -0
  256. data/tracks/scheme/{difference-of-squares → exercises/difference-of-squares}/example.scm +0 -0
  257. data/tracks/scheme/{difference-of-squares → exercises/difference-of-squares}/squares.scm +0 -0
  258. data/tracks/scheme/{grains → exercises/grains}/example.scm +0 -0
  259. data/tracks/scheme/{grains → exercises/grains}/grains-test.scm +0 -0
  260. data/tracks/scheme/{grains → exercises/grains}/grains.scm +0 -0
  261. data/tracks/scheme/{hamming → exercises/hamming}/example.scm +0 -0
  262. data/tracks/scheme/{hamming → exercises/hamming}/hamming-test.scm +0 -0
  263. data/tracks/scheme/{hamming → exercises/hamming}/hamming.scm +0 -0
  264. data/tracks/scheme/{hello-world → exercises/hello-world}/example.scm +0 -0
  265. data/tracks/scheme/{hello-world → exercises/hello-world}/hello-world-test.scm +0 -0
  266. data/tracks/scheme/{hello-world → exercises/hello-world}/hello-world.scm +0 -0
  267. data/tracks/scheme/{leap → exercises/leap}/example.scm +0 -0
  268. data/tracks/scheme/{leap → exercises/leap}/leap-test.scm +0 -0
  269. data/tracks/scheme/{leap → exercises/leap}/leap-year.scm +0 -0
  270. data/tracks/scheme/{list-ops → exercises/list-ops}/example.scm +0 -0
  271. data/tracks/scheme/{list-ops → exercises/list-ops}/list-ops-test.scm +0 -0
  272. data/tracks/scheme/{list-ops → exercises/list-ops}/list-ops.scm +0 -0
  273. data/tracks/scheme/{nucleotide-count → exercises/nucleotide-count}/example.scm +0 -0
  274. data/tracks/scheme/{nucleotide-count → exercises/nucleotide-count}/nucleotide-count-test.scm +0 -0
  275. data/tracks/scheme/{nucleotide-count → exercises/nucleotide-count}/nucleotide-count.scm +0 -0
  276. data/tracks/scheme/{phone-number → exercises/phone-number}/example.scm +0 -0
  277. data/tracks/scheme/{phone-number → exercises/phone-number}/phone-number-test.scm +0 -0
  278. data/tracks/scheme/{phone-number → exercises/phone-number}/phone-number.scm +0 -0
  279. data/tracks/scheme/{raindrops → exercises/raindrops}/example.scm +0 -0
  280. data/tracks/scheme/{raindrops → exercises/raindrops}/raindrops-test.scm +0 -0
  281. data/tracks/scheme/{raindrops → exercises/raindrops}/raindrops.scm +0 -0
  282. data/tracks/scheme/{rna-transcription → exercises/rna-transcription}/dna.scm +0 -0
  283. data/tracks/scheme/{rna-transcription → exercises/rna-transcription}/example.scm +0 -0
  284. data/tracks/scheme/{rna-transcription → exercises/rna-transcription}/rna-transcription-test.scm +0 -0
  285. data/tracks/scheme/{robot-name → exercises/robot-name}/example.scm +0 -0
  286. data/tracks/scheme/{robot-name → exercises/robot-name}/robot-name-test.scm +0 -0
  287. data/tracks/scheme/{robot-name → exercises/robot-name}/robot.scm +0 -0
  288. metadata +125 -64
@@ -0,0 +1,114 @@
1
+ using System;
2
+ using System.Collections.Generic;
3
+ using System.IO;
4
+ using System.Linq;
5
+ using System.Text.RegularExpressions;
6
+
7
+ public static class Grep
8
+ {
9
+ private class Line
10
+ {
11
+ public int Number { get; set; }
12
+ public string Text { get; set; }
13
+ public string File { get; set; }
14
+ }
15
+
16
+ [Flags]
17
+ private enum Flags
18
+ {
19
+ None = 0,
20
+ PrintLineNumbers = 1,
21
+ PrintFileNames = 2,
22
+ CaseInsensitive = 4,
23
+ Invert = 8,
24
+ MatchEntireLines = 16
25
+ }
26
+
27
+ public static string Find(string pattern, string flags, string[] files)
28
+ {
29
+ var parsedFlags = ParseFlags(flags);
30
+
31
+ if (parsedFlags.HasFlag(Flags.PrintFileNames))
32
+ {
33
+ return FormatMatchingFiles(pattern, parsedFlags, files);
34
+ }
35
+
36
+ return FormatMatchingLines(pattern, parsedFlags, files);
37
+ }
38
+
39
+ private static Flags ParseFlags(string flags) => flags.Split(' ').Aggregate(Flags.None, (acc, flag) => acc | ParseFlag(flag));
40
+
41
+ private static Flags ParseFlag(string flag)
42
+ {
43
+ switch (flag)
44
+ {
45
+ case "-n": return Flags.PrintLineNumbers;
46
+ case "-l": return Flags.PrintFileNames;
47
+ case "-i": return Flags.CaseInsensitive;
48
+ case "-v": return Flags.Invert;
49
+ case "-x": return Flags.MatchEntireLines;
50
+ default: return Flags.None;
51
+ }
52
+ }
53
+
54
+ private static Func<Line, bool> IsMatch(string pattern, Flags flags)
55
+ {
56
+ var matchPattern = flags.HasFlag(Flags.MatchEntireLines) ? $"^{pattern}$" : pattern;
57
+ var options = flags.HasFlag(Flags.CaseInsensitive) ? RegexOptions.IgnoreCase : RegexOptions.None;
58
+ var regex = new Regex(matchPattern, options);
59
+
60
+ return line => regex.IsMatch(line.Text) != flags.HasFlag(Flags.Invert);
61
+ }
62
+
63
+ private static IEnumerable<Line> FindMatchingLines(string pattern, Flags flags, string file)
64
+ {
65
+ var isMatch = IsMatch(pattern, flags);
66
+
67
+ return File.ReadAllLines(file)
68
+ .Select((line, index) => CreateLine(file, index, line))
69
+ .Where(isMatch);
70
+ }
71
+
72
+ private static Line CreateLine(string file, int index, string text) => new Line { File = file, Number = index + 1, Text = text };
73
+
74
+ private static string FormatMatchingFile(string file) => $"{file}\n";
75
+
76
+ private static string FormatMatchingFiles(string pattern, Flags flags, string[] files)
77
+ {
78
+ var matchingFiles = files
79
+ .Where(file => FindMatchingLines(pattern, flags, file).Any())
80
+ .Select(FormatMatchingFile);
81
+
82
+ return string.Concat(matchingFiles);
83
+ }
84
+
85
+ private static string FormatMatchingLine(Flags flags, string[] files, Line line)
86
+ {
87
+ var printLineNumbers = flags.HasFlag(Flags.PrintLineNumbers);
88
+ var printFileName = files.Length > 1;
89
+
90
+ if (printLineNumbers && printFileName)
91
+ {
92
+ return $"{line.File}:{line.Number}:{line.Text}\n";
93
+ }
94
+ if (printLineNumbers && !printFileName)
95
+ {
96
+ return $"{line.Number}:{line.Text}\n";
97
+ }
98
+ if (!printLineNumbers && printFileName)
99
+ {
100
+ return $"{line.File}:{line.Text}\n";
101
+ }
102
+
103
+ return $"{line.Text}\n";
104
+ }
105
+
106
+ private static string FormatMatchingLines(string pattern, Flags flags, string[] files)
107
+ {
108
+ var matchingFiles = files
109
+ .SelectMany(file => FindMatchingLines(pattern, flags, file))
110
+ .Select(line => FormatMatchingLine(flags, files, line));
111
+
112
+ return string.Concat(matchingFiles);
113
+ }
114
+ }
@@ -0,0 +1,376 @@
1
+ using System.IO;
2
+ using NUnit.Framework;
3
+
4
+ public class GrepTest
5
+ {
6
+ private const string IliadFileName = "iliad.txt";
7
+ private const string IliadContents =
8
+ @"Achilles sing, O Goddess! Peleus' son;
9
+ His wrath pernicious, who ten thousand woes
10
+ Caused to Achaia's host, sent many a soul
11
+ Illustrious into Ades premature,
12
+ And Heroes gave (so stood the will of Jove)
13
+ To dogs and to all ravening fowls a prey,
14
+ When fierce dispute had separated once
15
+ The noble Chief Achilles from the son
16
+ Of Atreus, Agamemnon, King of men.
17
+ ";
18
+
19
+ private const string MidsummerNightFileName = "midsummer-night.txt";
20
+ private const string MidsummerNightContents =
21
+ @"I do entreat your grace to pardon me.
22
+ I know not by what power I am made bold,
23
+ Nor how it may concern my modesty,
24
+ In such a presence here to plead my thoughts;
25
+ But I beseech your grace that I may know
26
+ The worst that may befall me in this case,
27
+ If I refuse to wed Demetrius.
28
+ ";
29
+
30
+ private const string ParadiseLostFileName = "paradise-lost.txt";
31
+ private const string ParadiseLostContents =
32
+ @"Of Mans First Disobedience, and the Fruit
33
+ Of that Forbidden Tree, whose mortal tast
34
+ Brought Death into the World, and all our woe,
35
+ With loss of Eden, till one greater Man
36
+ Restore us, and regain the blissful Seat,
37
+ Sing Heav'nly Muse, that on the secret top
38
+ Of Oreb, or of Sinai, didst inspire
39
+ That Shepherd, who first taught the chosen Seed
40
+ ";
41
+
42
+ [OneTimeSetUp]
43
+ public void SetUp()
44
+ {
45
+ File.WriteAllText(IliadFileName, IliadContents);
46
+ File.WriteAllText(MidsummerNightFileName, MidsummerNightContents);
47
+ File.WriteAllText(ParadiseLostFileName, ParadiseLostContents);
48
+ }
49
+
50
+ [OneTimeTearDown]
51
+ public void TearDown()
52
+ {
53
+ File.Delete(IliadFileName);
54
+ File.Delete(MidsummerNightFileName);
55
+ File.Delete(ParadiseLostFileName);
56
+ }
57
+
58
+ [Test]
59
+ public void One_file_one_match_no_flags()
60
+ {
61
+ const string pattern = "Agamemnon";
62
+ const string flags = "";
63
+ var files = new[] { IliadFileName };
64
+
65
+ const string expected =
66
+ "Of Atreus, Agamemnon, King of men.\n";
67
+
68
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
69
+ }
70
+
71
+ [Ignore("Remove to run test")]
72
+ [Test]
73
+ public void One_file_one_match_print_line_numbers_flag()
74
+ {
75
+ const string pattern = "Forbidden";
76
+ const string flags = "-n";
77
+ var files = new[] { ParadiseLostFileName };
78
+
79
+ const string expected =
80
+ "2:Of that Forbidden Tree, whose mortal tast\n";
81
+
82
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
83
+ }
84
+
85
+ [Ignore("Remove to run test")]
86
+ [Test]
87
+ public void One_file_one_match_case_insensitive_flag()
88
+ {
89
+ const string pattern = "Forbidden";
90
+ const string flags = "-i";
91
+ var files = new[] { ParadiseLostFileName };
92
+
93
+ const string expected =
94
+ "Of that Forbidden Tree, whose mortal tast\n";
95
+
96
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
97
+ }
98
+
99
+ [Ignore("Remove to run test")]
100
+ [Test]
101
+ public void One_file_one_match_print_file_names_flag()
102
+ {
103
+ const string pattern = "Forbidden";
104
+ const string flags = "-l";
105
+ var files = new[] { ParadiseLostFileName };
106
+
107
+ var expected =
108
+ $"{ParadiseLostFileName}\n";
109
+
110
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
111
+ }
112
+
113
+ [Ignore("Remove to run test")]
114
+ [Test]
115
+ public void One_file_one_match_match_entire_lines_flag()
116
+ {
117
+ const string pattern = "With loss of Eden, till one greater Man";
118
+ const string flags = "-x";
119
+ var files = new[] { ParadiseLostFileName };
120
+
121
+ const string expected =
122
+ "With loss of Eden, till one greater Man\n";
123
+
124
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
125
+ }
126
+
127
+ [Ignore("Remove to run test")]
128
+ [Test]
129
+ public void One_file_one_match_multiple_flags()
130
+ {
131
+ const string pattern = "OF ATREUS, Agamemnon, KIng of MEN.";
132
+ var files = new[] { IliadFileName };
133
+ const string flags = "-n -i -x";
134
+ const string expected =
135
+ "9:Of Atreus, Agamemnon, King of men.\n";
136
+
137
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
138
+ }
139
+
140
+ [Ignore("Remove to run test")]
141
+ [Test]
142
+ public void One_file_several_matches_no_flags()
143
+ {
144
+ const string pattern = "may";
145
+ const string flags = "";
146
+ var files = new[] { MidsummerNightFileName };
147
+
148
+ const string expected =
149
+ "Nor how it may concern my modesty,\n" +
150
+ "But I beseech your grace that I may know\n" +
151
+ "The worst that may befall me in this case,\n";
152
+
153
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
154
+ }
155
+
156
+ [Ignore("Remove to run test")]
157
+ [Test]
158
+ public void One_file_several_matches_print_line_numbers_flag()
159
+ {
160
+ const string pattern = "may";
161
+ const string flags = "-n";
162
+ var files = new[] { MidsummerNightFileName };
163
+
164
+ const string expected =
165
+ "3:Nor how it may concern my modesty,\n" +
166
+ "5:But I beseech your grace that I may know\n" +
167
+ "6:The worst that may befall me in this case,\n";
168
+
169
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
170
+ }
171
+
172
+ [Ignore("Remove to run test")]
173
+ [Test]
174
+ public void One_file_several_matches_match_entire_lines_flag()
175
+ {
176
+ const string pattern = "may";
177
+ const string flags = "-x";
178
+ var files = new[] { MidsummerNightFileName };
179
+
180
+ const string expected = "";
181
+
182
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
183
+ }
184
+
185
+ [Ignore("Remove to run test")]
186
+ [Test]
187
+ public void One_file_several_matches_case_insensitive_flag()
188
+ {
189
+ const string pattern = "ACHILLES";
190
+ const string flags = "-i";
191
+ var files = new[] { IliadFileName };
192
+
193
+ const string expected =
194
+ "Achilles sing, O Goddess! Peleus' son;\n" +
195
+ "The noble Chief Achilles from the son\n";
196
+
197
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
198
+ }
199
+
200
+ [Ignore("Remove to run test")]
201
+ [Test]
202
+ public void One_file_several_matches_inverted_flag()
203
+ {
204
+ const string pattern = "Of";
205
+ const string flags = "-v";
206
+ var files = new[] { ParadiseLostFileName };
207
+
208
+ const string expected =
209
+ "Brought Death into the World, and all our woe,\n" +
210
+ "With loss of Eden, till one greater Man\n" +
211
+ "Restore us, and regain the blissful Seat,\n" +
212
+ "Sing Heav'nly Muse, that on the secret top\n" +
213
+ "That Shepherd, who first taught the chosen Seed\n";
214
+
215
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
216
+ }
217
+
218
+ [TestCase("", Ignore = "Remove to run test case")]
219
+ [TestCase("-n", Ignore = "Remove to run test case")]
220
+ [TestCase("-l", Ignore = "Remove to run test case")]
221
+ [TestCase("-x", Ignore = "Remove to run test case")]
222
+ [TestCase("-i", Ignore = "Remove to run test case")]
223
+ [TestCase("-n -l -x -i", Ignore = "Remove to run test case")]
224
+ public void One_file_no_matches_various_flags(string flags)
225
+ {
226
+ const string pattern = "Gandalf";
227
+ var files = new[] { IliadFileName };
228
+ const string expected = "";
229
+
230
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
231
+ }
232
+
233
+ [Ignore("Remove to run test")]
234
+ [Test]
235
+ public void Multiple_files_one_match_no_flags()
236
+ {
237
+ const string pattern = "Agamemnon";
238
+ const string flags = "";
239
+ var files = new[] { IliadFileName, MidsummerNightFileName, ParadiseLostFileName };
240
+
241
+ var expected =
242
+ $"{IliadFileName}:Of Atreus, Agamemnon, King of men.\n";
243
+
244
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
245
+ }
246
+
247
+ [Ignore("Remove to run test")]
248
+ [Test]
249
+ public void Multiple_files_several_matches_no_flags()
250
+ {
251
+ const string pattern = "may";
252
+ const string flags = "";
253
+ var files = new[] { IliadFileName, MidsummerNightFileName, ParadiseLostFileName };
254
+
255
+ var expected =
256
+ $"{MidsummerNightFileName}:Nor how it may concern my modesty,\n" +
257
+ $"{MidsummerNightFileName}:But I beseech your grace that I may know\n" +
258
+ $"{MidsummerNightFileName}:The worst that may befall me in this case,\n";
259
+
260
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
261
+ }
262
+
263
+ [Ignore("Remove to run test")]
264
+ [Test]
265
+ public void Multiple_files_several_matches_print_line_numbers_flag()
266
+ {
267
+ const string pattern = "that";
268
+ const string flags = "-n";
269
+ var files = new[] { IliadFileName, MidsummerNightFileName, ParadiseLostFileName };
270
+
271
+ var expected =
272
+ $"{MidsummerNightFileName}:5:But I beseech your grace that I may know\n" +
273
+ $"{MidsummerNightFileName}:6:The worst that may befall me in this case,\n" +
274
+ $"{ParadiseLostFileName}:2:Of that Forbidden Tree, whose mortal tast\n" +
275
+ $"{ParadiseLostFileName}:6:Sing Heav'nly Muse, that on the secret top\n";
276
+
277
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
278
+ }
279
+
280
+ [Ignore("Remove to run test")]
281
+ [Test]
282
+ public void Multiple_files_several_matches_print_file_names_flag()
283
+ {
284
+ const string pattern = "who";
285
+ const string flags = "-l";
286
+ var files = new[] { IliadFileName, MidsummerNightFileName, ParadiseLostFileName };
287
+
288
+ var expected =
289
+ $"{IliadFileName}\n" +
290
+ $"{ParadiseLostFileName}\n";
291
+
292
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
293
+ }
294
+
295
+ [Ignore("Remove to run test")]
296
+ [Test]
297
+ public void Multiple_files_several_matches_case_insensitive_flag()
298
+ {
299
+ const string pattern = "TO";
300
+ const string flags = "-i";
301
+ var files = new[] { IliadFileName, MidsummerNightFileName, ParadiseLostFileName };
302
+
303
+ var expected =
304
+ $"{IliadFileName}:Caused to Achaia's host, sent many a soul\n" +
305
+ $"{IliadFileName}:Illustrious into Ades premature,\n" +
306
+ $"{IliadFileName}:And Heroes gave (so stood the will of Jove)\n" +
307
+ $"{IliadFileName}:To dogs and to all ravening fowls a prey,\n" +
308
+ $"{MidsummerNightFileName}:I do entreat your grace to pardon me.\n" +
309
+ $"{MidsummerNightFileName}:In such a presence here to plead my thoughts;\n" +
310
+ $"{MidsummerNightFileName}:If I refuse to wed Demetrius.\n" +
311
+ $"{ParadiseLostFileName}:Brought Death into the World, and all our woe,\n" +
312
+ $"{ParadiseLostFileName}:Restore us, and regain the blissful Seat,\n" +
313
+ $"{ParadiseLostFileName}:Sing Heav'nly Muse, that on the secret top\n";
314
+
315
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
316
+ }
317
+
318
+ [Ignore("Remove to run test")]
319
+ [Test]
320
+ public void Multiple_files_several_matches_inverted_flag()
321
+ {
322
+ const string pattern = "a";
323
+ const string flags = "-v";
324
+ var files = new[] { IliadFileName, MidsummerNightFileName, ParadiseLostFileName };
325
+
326
+ var expected =
327
+ $"{IliadFileName}:Achilles sing, O Goddess! Peleus' son;\n" +
328
+ $"{IliadFileName}:The noble Chief Achilles from the son\n" +
329
+ $"{MidsummerNightFileName}:If I refuse to wed Demetrius.\n";
330
+
331
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
332
+ }
333
+
334
+ [Ignore("Remove to run test")]
335
+ [Test]
336
+ public void Multiple_files_one_match_match_entire_lines_flag()
337
+ {
338
+ const string pattern = "But I beseech your grace that I may know";
339
+ const string flags = "-x";
340
+ var files = new[] { IliadFileName, MidsummerNightFileName, ParadiseLostFileName };
341
+
342
+ var expected =
343
+ $"{MidsummerNightFileName}:But I beseech your grace that I may know\n";
344
+
345
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
346
+ }
347
+
348
+ [Ignore("Remove to run test")]
349
+ [Test]
350
+ public void Multiple_files_one_match_multiple_flags()
351
+ {
352
+ const string pattern = "WITH LOSS OF EDEN, TILL ONE GREATER MAN";
353
+ var files = new[] { IliadFileName, MidsummerNightFileName, ParadiseLostFileName };
354
+ const string flags = "-n -i -x";
355
+ var expected =
356
+ $"{ParadiseLostFileName}:4:With loss of Eden, till one greater Man\n";
357
+
358
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
359
+ }
360
+
361
+ [TestCase("", Ignore = "Remove to run test case")]
362
+ [TestCase("-n", Ignore = "Remove to run test case")]
363
+ [TestCase("-l", Ignore = "Remove to run test case")]
364
+ [TestCase("-x", Ignore = "Remove to run test case")]
365
+ [TestCase("-i", Ignore = "Remove to run test case")]
366
+ [TestCase("-n -l -x -i", Ignore = "Remove to run test case")]
367
+ public void Multiple_files_no_matches_various_flags(string flags)
368
+ {
369
+ const string pattern = "Frodo";
370
+ var files = new[] { IliadFileName, MidsummerNightFileName, ParadiseLostFileName };
371
+
372
+ const string expected = "";
373
+
374
+ Assert.That(Grep.Find(pattern, flags, files), Is.EqualTo(expected));
375
+ }
376
+ }