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.
- checksums.yaml +4 -4
- data/.travis.yml +7 -0
- data/CHANGELOG.md +9 -1
- data/README.md +10 -0
- data/common/CONTRIBUTING.md +50 -2
- data/common/exercises/alphametics/canonical-data.json +10 -10
- data/common/exercises/markdown/canonical-data.json +4 -4
- data/common/exercises/pov/canonical-data.json +356 -0
- data/lib/trackler/version.rb +1 -1
- data/trackler.gemspec +1 -2
- data/tracks/c/config.json +11 -1
- data/tracks/c/exercises/binary-search/makefile +16 -0
- data/tracks/c/exercises/binary-search/src/example.c +21 -0
- data/tracks/c/exercises/binary-search/src/example.h +8 -0
- data/tracks/c/exercises/binary-search/test/test_binary_search.c +99 -0
- data/tracks/c/exercises/binary-search/test/vendor/unity.c +1300 -0
- data/tracks/c/exercises/binary-search/test/vendor/unity.h +274 -0
- data/tracks/c/exercises/binary-search/test/vendor/unity_internals.h +701 -0
- data/tracks/clojure/config.json +3 -2
- data/tracks/clojure/exercises/bank-account/project.clj +4 -0
- data/tracks/clojure/exercises/bank-account/src/example.clj +17 -0
- data/tracks/clojure/exercises/bank-account/test/bank_account_test.clj +44 -0
- data/tracks/clojure/exercises/isogram/project.clj +4 -0
- data/tracks/clojure/exercises/isogram/src/example.clj +5 -0
- data/tracks/clojure/exercises/isogram/test/isogram_test.clj +17 -0
- data/tracks/csharp/config.json +27 -87
- data/tracks/csharp/exercises/exercises.csproj +265 -1
- data/tracks/csharp/exercises/grep/Example.cs +114 -0
- data/tracks/csharp/exercises/grep/GrepTest.cs +376 -0
- data/tracks/csharp/exercises/paket.references +2 -1
- data/tracks/csharp/exercises/poker/Example.cs +96 -0
- data/tracks/csharp/exercises/poker/PokerTest.cs +171 -0
- data/tracks/csharp/exercises/sgf-parsing/Example.cs +61 -0
- data/tracks/csharp/exercises/sgf-parsing/SgfParsing.cs +13 -0
- data/tracks/csharp/exercises/sgf-parsing/SgfParsingTest.cs +158 -0
- data/tracks/csharp/paket.dependencies +2 -1
- data/tracks/csharp/paket.lock +72 -0
- data/tracks/ecmascript/exercises/simple-cipher/simple-cipher.spec.js +1 -1
- data/tracks/elisp/config.json +6 -1
- data/tracks/elisp/exercises/etl/etl-test.el +90 -0
- data/tracks/elisp/exercises/etl/etl.el +9 -0
- data/tracks/elisp/exercises/etl/example.el +22 -0
- data/tracks/elisp/exercises/nucleotide-count/example.el +19 -0
- data/tracks/elisp/exercises/nucleotide-count/nucleotide-count-test.el +34 -0
- data/tracks/elisp/exercises/nucleotide-count/nucleotide-count.el +9 -0
- data/tracks/elisp/exercises/perfect-numbers/example.el +28 -0
- data/tracks/elisp/exercises/perfect-numbers/perfect-numbers-test.el +24 -0
- data/tracks/elisp/exercises/perfect-numbers/perfect-numbers.el +10 -0
- data/tracks/elisp/exercises/roman-numerals/example.el +36 -0
- data/tracks/elisp/exercises/roman-numerals/roman-numerals-test.el +30 -0
- data/tracks/elisp/exercises/roman-numerals/roman-numerals.el +10 -0
- data/tracks/elisp/exercises/word-count/example.el +18 -0
- data/tracks/elisp/exercises/word-count/word-count-test.el +64 -0
- data/tracks/elisp/exercises/word-count/word-count.el +9 -0
- data/tracks/elixir/README.md +4 -4
- data/tracks/elixir/SETUP.md +3 -3
- data/tracks/elixir/docs/TESTS.md +3 -3
- data/tracks/elixir/exercises/accumulate/accumulate_test.exs +7 -7
- data/tracks/elixir/exercises/acronym/acronym_test.exs +5 -5
- data/tracks/elixir/exercises/allergies/allergies_test.exs +13 -13
- data/tracks/elixir/exercises/anagram/anagram_test.exs +12 -12
- data/tracks/elixir/exercises/atbash-cipher/atbash_cipher_test.exs +9 -9
- data/tracks/elixir/exercises/bank-account/bank_account_test.exs +5 -5
- data/tracks/elixir/exercises/beer-song/beer_song_test.exs +6 -6
- data/tracks/elixir/exercises/binary-search/binary_search_test.exs +7 -7
- data/tracks/elixir/exercises/binary/binary_test.exs +13 -13
- data/tracks/elixir/exercises/bob/bob_test.exs +15 -15
- data/tracks/elixir/exercises/bowling/bowling_test.exs +18 -18
- data/tracks/elixir/exercises/bracket-push/bracket_push_test.exs +12 -12
- data/tracks/elixir/exercises/change/change_test.exs +7 -7
- data/tracks/elixir/exercises/connect/connect_test.exs +9 -9
- data/tracks/elixir/exercises/crypto-square/crypto_square_test.exs +7 -7
- data/tracks/elixir/exercises/custom-set/custom_set_test.exs +11 -11
- data/tracks/elixir/exercises/diamond/diamond_test.exs +4 -4
- data/tracks/elixir/exercises/difference-of-squares/difference_of_squares_test.exs +10 -10
- data/tracks/elixir/exercises/dot-dsl/dot_dsl_test.exs +17 -17
- data/tracks/elixir/exercises/etl/etl_test.exs +5 -5
- data/tracks/elixir/exercises/flatten-array/flatten_array_test.exs +6 -6
- data/tracks/elixir/exercises/forth/forth_test.exs +16 -16
- data/tracks/elixir/exercises/gigasecond/gigasecond_test.exs +5 -5
- data/tracks/elixir/exercises/grade-school/grade_school_test.exs +6 -6
- data/tracks/elixir/exercises/grains/grains_test.exs +9 -9
- data/tracks/elixir/exercises/hamming/hamming_test.exs +6 -6
- data/tracks/elixir/exercises/hello-world/hello_world_test.exs +3 -3
- data/tracks/elixir/exercises/hexadecimal/hexadecimal_test.exs +11 -11
- data/tracks/elixir/exercises/isogram/isogram_test.exs +10 -10
- data/tracks/elixir/exercises/kindergarten-garden/garden_test.exs +7 -7
- data/tracks/elixir/exercises/largest-series-product/largest_series_product_test.exs +17 -17
- data/tracks/elixir/exercises/leap/leap_test.exs +5 -5
- data/tracks/elixir/exercises/list-ops/list_ops_test.exs +26 -26
- data/tracks/elixir/exercises/luhn/luhn_test.exs +7 -7
- data/tracks/elixir/exercises/markdown/markdown_test.exs +10 -10
- data/tracks/elixir/exercises/meetup/meetup_test.exs +92 -92
- data/tracks/elixir/exercises/minesweeper/minesweeper_test.exs +8 -8
- data/tracks/elixir/exercises/nth-prime/nth_prime_test.exs +6 -6
- data/tracks/elixir/exercises/nucleotide-count/example.exs +0 -5
- data/tracks/elixir/exercises/nucleotide-count/nucleotide_count_test.exs +7 -7
- data/tracks/elixir/exercises/palindrome-products/palindrome_products_test.exs +6 -6
- data/tracks/elixir/exercises/pangram/pangram_test.exs +11 -11
- data/tracks/elixir/exercises/parallel-letter-frequency/parallel_letter_frequency_test.exs +10 -10
- data/tracks/elixir/exercises/pascals-triangle/pascals_triangle_test.exs +7 -7
- data/tracks/elixir/exercises/phone-number/phone_number_test.exs +10 -10
- data/tracks/elixir/exercises/prime-factors/prime_factors_test.exs +13 -13
- data/tracks/elixir/exercises/pythagorean-triplet/pythagorean_triplet_test.exs +8 -8
- data/tracks/elixir/exercises/queen-attack/queen_attack_test.exs +16 -16
- data/tracks/elixir/exercises/rail-fence-cipher/rail_fence_cipher_test.exs +13 -13
- data/tracks/elixir/exercises/raindrops/raindrops_test.exs +17 -17
- data/tracks/elixir/exercises/rna-transcription/rna_transcription_test.exs +6 -6
- data/tracks/elixir/exercises/robot-simulator/robot_simulator_test.exs +6 -6
- data/tracks/elixir/exercises/roman-numerals/roman_numerals_test.exs +19 -19
- data/tracks/elixir/exercises/run-length-encoding/rle_test.exs +6 -6
- data/tracks/elixir/exercises/saddle-points/saddle_points_test.exs +11 -11
- data/tracks/elixir/exercises/scrabble-score/scrabble_score_test.exs +9 -9
- data/tracks/elixir/exercises/sieve/sieve_test.exs +3 -3
- data/tracks/elixir/exercises/space-age/space_age_test.exs +9 -9
- data/tracks/elixir/exercises/sublist/sublist_test.exs +20 -20
- data/tracks/elixir/exercises/sum-of-multiples/sum_of_multiples_test.exs +11 -11
- data/tracks/elixir/exercises/test_helper.exs +1 -1
- data/tracks/elixir/exercises/triangle/triangle_test.exs +16 -16
- data/tracks/elixir/exercises/word-count/word_count_test.exs +9 -9
- data/tracks/elixir/exercises/wordy/wordy_test.exs +16 -16
- data/tracks/elixir/exercises/zipper/zipper_test.exs +13 -13
- data/tracks/fsharp/build.fsx +2 -7
- data/tracks/fsharp/config.json +0 -104
- data/tracks/fsharp/exercises/grep/GrepTest.fs +7 -1
- data/tracks/go/config.json +1 -0
- data/tracks/go/exercises/acronym/acronym_test.go +36 -0
- data/tracks/go/exercises/acronym/example.go +21 -0
- data/tracks/go/exercises/palindrome-products/example.go +2 -0
- data/tracks/go/exercises/palindrome-products/palindrome_products_test.go +10 -0
- data/tracks/go/exercises/phone-number/example.go +2 -0
- data/tracks/go/exercises/phone-number/phone_number_test.go +44 -27
- data/tracks/go/exercises/robot-simulator/defs.go +6 -6
- data/tracks/go/exercises/robot-simulator/example.go +22 -22
- data/tracks/go/exercises/robot-simulator/robot_simulator_step2_test.go +28 -28
- data/tracks/go/exercises/robot-simulator/robot_simulator_step3_test.go +75 -76
- data/tracks/go/exercises/robot-simulator/robot_simulator_test.go +10 -4
- data/tracks/haskell/exercises/anagram/HINTS.md +22 -0
- data/tracks/haskell/exercises/anagram/package.yaml +3 -0
- data/tracks/haskell/exercises/anagram/src/Example.hs +10 -7
- data/tracks/haskell/exercises/anagram/test/Tests.hs +9 -3
- data/tracks/haskell/exercises/beer-song/HINTS.md +18 -0
- data/tracks/haskell/exercises/beer-song/src/Beer.hs +301 -3
- data/tracks/haskell/exercises/beer-song/src/Example.hs +5 -3
- data/tracks/haskell/exercises/beer-song/test/Tests.hs +332 -23
- data/tracks/haskell/exercises/crypto-square/src/CryptoSquare.hs +3 -17
- data/tracks/haskell/exercises/crypto-square/src/Example.hs +12 -20
- data/tracks/haskell/exercises/crypto-square/test/Tests.hs +46 -86
- data/tracks/haskell/exercises/leap/HINTS.md +11 -0
- data/tracks/haskell/exercises/leap/src/LeapYear.hs +1 -0
- data/tracks/java/exercises/pangram/src/test/java/PangramTest.java +34 -4
- data/tracks/lua/exercises/pov/pov_spec.lua +0 -13
- data/tracks/mips/docs/ABOUT.md +1 -1
- data/tracks/objective-c/config.json +225 -0
- data/tracks/objective-c/exercises/largest-series-product/LargestSeriesProductExample.h +8 -0
- data/tracks/objective-c/exercises/largest-series-product/LargestSeriesProductExample.m +63 -0
- data/tracks/objective-c/exercises/largest-series-product/LargestSeriesProductTest.m +84 -0
- data/tracks/objective-c/xcodeProject/ObjectiveC.xcodeproj/project.pbxproj +18 -0
- data/tracks/ocaml/.travis-ci.sh +23 -0
- data/tracks/ocaml/.travis.yml +5 -1
- data/tracks/ocaml/Makefile +41 -0
- data/tracks/ocaml/config.json +18 -0
- data/tracks/ocaml/exercises/anagram/Makefile +1 -0
- data/tracks/ocaml/exercises/beer-song/Makefile +1 -0
- data/tracks/ocaml/exercises/beer-song/{beer.mli → beer_song.mli} +0 -0
- data/tracks/ocaml/exercises/beer-song/example.ml +1 -1
- data/tracks/ocaml/exercises/beer-song/test.ml +7 -6
- data/tracks/ocaml/exercises/bob/Makefile +1 -0
- data/tracks/ocaml/exercises/bowling/Makefile +1 -0
- data/tracks/ocaml/exercises/custom-set/Makefile +1 -0
- data/tracks/ocaml/exercises/difference-of-squares/.merlin +5 -0
- data/tracks/ocaml/exercises/difference-of-squares/Makefile +11 -0
- data/tracks/ocaml/exercises/difference-of-squares/difference_of_squares.mli +8 -0
- data/tracks/ocaml/exercises/difference-of-squares/example.ml +21 -0
- data/tracks/ocaml/exercises/difference-of-squares/test.ml +32 -0
- data/tracks/ocaml/exercises/grade-school/Makefile +1 -0
- data/tracks/ocaml/exercises/grade-school/{school.mli → grade_school.mli} +0 -0
- data/tracks/ocaml/exercises/grade-school/test.ml +27 -27
- data/tracks/ocaml/exercises/hamming/Makefile +1 -0
- data/tracks/ocaml/exercises/hangman/Makefile +1 -0
- data/tracks/ocaml/exercises/hello-world/Makefile +1 -1
- data/tracks/ocaml/exercises/hello-world/{hello.ml → hello_world.ml} +0 -0
- data/tracks/ocaml/exercises/hello-world/{hello.mli → hello_world.mli} +0 -0
- data/tracks/ocaml/exercises/hello-world/test.ml +1 -1
- data/tracks/ocaml/exercises/hexadecimal/Makefile +1 -0
- data/tracks/ocaml/exercises/leap/Makefile +1 -1
- data/tracks/ocaml/exercises/list-ops/Makefile +1 -0
- data/tracks/ocaml/exercises/luhn/Makefile +1 -0
- data/tracks/ocaml/exercises/minesweeper/Makefile +1 -0
- data/tracks/ocaml/exercises/nucleotide-count/Makefile +1 -0
- data/tracks/ocaml/exercises/nucleotide-count/{dna.mli → nucleotide_count.mli} +0 -0
- data/tracks/ocaml/exercises/nucleotide-count/test.ml +7 -7
- data/tracks/ocaml/exercises/phone-number/Makefile +1 -0
- data/tracks/ocaml/exercises/phone-number/{phone.mli → phone_number.mli} +0 -0
- data/tracks/ocaml/exercises/phone-number/test.ml +10 -10
- data/tracks/ocaml/exercises/point-mutations/Makefile +8 -0
- data/tracks/ocaml/exercises/point-mutations/example.ml +0 -0
- data/tracks/ocaml/exercises/prime-factors/Makefile +1 -0
- data/tracks/ocaml/exercises/raindrops/.merlin +5 -0
- data/tracks/ocaml/exercises/raindrops/Makefile +11 -0
- data/tracks/ocaml/exercises/raindrops/example.ml +11 -0
- data/tracks/ocaml/exercises/raindrops/raindrops.mli +7 -0
- data/tracks/ocaml/exercises/raindrops/test.ml +48 -0
- data/tracks/ocaml/exercises/rna-transcription/Makefile +1 -0
- data/tracks/ocaml/exercises/rna-transcription/{dna.mli → rna_transcription.mli} +0 -0
- data/tracks/ocaml/exercises/rna-transcription/test.ml +6 -6
- data/tracks/ocaml/exercises/roman-numerals/Makefile +1 -0
- data/tracks/ocaml/exercises/say/.gitignore +1 -0
- data/tracks/ocaml/exercises/say/.merlin +5 -0
- data/tracks/ocaml/exercises/say/Makefile +11 -0
- data/tracks/ocaml/exercises/say/example.ml +52 -0
- data/tracks/ocaml/exercises/say/say.mli +8 -0
- data/tracks/ocaml/exercises/say/test.ml +65 -0
- data/tracks/ocaml/exercises/space-age/Makefile +1 -0
- data/tracks/ocaml/exercises/word-count/Makefile +1 -0
- data/tracks/ocaml/exercises/word-count/example.ml +1 -1
- data/tracks/ocaml/exercises/zipper/HINT.md +5 -6
- data/tracks/ocaml/exercises/zipper/Makefile +1 -0
- data/tracks/ocaml/exercises/zipper/example.ml +2 -2
- data/tracks/ocaml/exercises/zipper/tree.ml +1 -1
- data/tracks/php/config.json +7 -1
- data/tracks/php/exercises/variable-length-quantity/example.php +56 -0
- data/tracks/php/exercises/variable-length-quantity/variable-length-quantity_test.php +149 -0
- data/tracks/python/config.json +1 -1
- data/tracks/ruby/.rubocop.yml +1915 -26
- data/tracks/ruby/bin/generate-acronym +7 -0
- data/tracks/ruby/bin/generate-nth-prime +7 -0
- data/tracks/ruby/config.json +482 -0
- data/tracks/ruby/exercises/acronym/.version +1 -0
- data/tracks/ruby/exercises/acronym/acronym_test.rb +45 -21
- data/tracks/ruby/exercises/acronym/example.rb +1 -1
- data/tracks/ruby/exercises/acronym/example.tt +19 -0
- data/tracks/ruby/exercises/anagram/anagram_test.rb +1 -1
- data/tracks/ruby/exercises/nth-prime/.version +1 -0
- data/tracks/ruby/exercises/nth-prime/example.rb +4 -0
- data/tracks/ruby/exercises/nth-prime/example.tt +23 -0
- data/tracks/ruby/exercises/nth-prime/nth_prime_test.rb +33 -10
- data/tracks/ruby/exercises/raindrops/.version +1 -1
- data/tracks/ruby/exercises/raindrops/example.rb +1 -1
- data/tracks/ruby/exercises/raindrops/raindrops_test.rb +20 -5
- data/tracks/ruby/lib/acronym_cases.rb +19 -0
- data/tracks/ruby/lib/generator.rb +3 -2
- data/tracks/ruby/lib/nth_prime_cases.rb +23 -0
- data/tracks/ruby/lib/raindrop_cases.rb +1 -1
- data/tracks/rust/README.md +1 -1
- data/tracks/scala/exercises/leap/src/test/scala/leap_test.scala +10 -0
- data/tracks/scala/exercises/prime-factors/example.scala +2 -5
- data/tracks/scala/exercises/prime-factors/src/test/scala/primefactors_test.scala +11 -13
- data/tracks/scheme/{anagram → exercises/anagram}/anagram-test.scm +0 -0
- data/tracks/scheme/{anagram → exercises/anagram}/anagram.scm +0 -0
- data/tracks/scheme/{anagram → exercises/anagram}/example.scm +0 -0
- data/tracks/scheme/{bob → exercises/bob}/bob-test.scm +0 -0
- data/tracks/scheme/{bob → exercises/bob}/bob.scm +0 -0
- data/tracks/scheme/{bob → exercises/bob}/example.scm +0 -0
- data/tracks/scheme/{difference-of-squares → exercises/difference-of-squares}/difference-of-squares-test.scm +0 -0
- data/tracks/scheme/{difference-of-squares → exercises/difference-of-squares}/example.scm +0 -0
- data/tracks/scheme/{difference-of-squares → exercises/difference-of-squares}/squares.scm +0 -0
- data/tracks/scheme/{grains → exercises/grains}/example.scm +0 -0
- data/tracks/scheme/{grains → exercises/grains}/grains-test.scm +0 -0
- data/tracks/scheme/{grains → exercises/grains}/grains.scm +0 -0
- data/tracks/scheme/{hamming → exercises/hamming}/example.scm +0 -0
- data/tracks/scheme/{hamming → exercises/hamming}/hamming-test.scm +0 -0
- data/tracks/scheme/{hamming → exercises/hamming}/hamming.scm +0 -0
- data/tracks/scheme/{hello-world → exercises/hello-world}/example.scm +0 -0
- data/tracks/scheme/{hello-world → exercises/hello-world}/hello-world-test.scm +0 -0
- data/tracks/scheme/{hello-world → exercises/hello-world}/hello-world.scm +0 -0
- data/tracks/scheme/{leap → exercises/leap}/example.scm +0 -0
- data/tracks/scheme/{leap → exercises/leap}/leap-test.scm +0 -0
- data/tracks/scheme/{leap → exercises/leap}/leap-year.scm +0 -0
- data/tracks/scheme/{list-ops → exercises/list-ops}/example.scm +0 -0
- data/tracks/scheme/{list-ops → exercises/list-ops}/list-ops-test.scm +0 -0
- data/tracks/scheme/{list-ops → exercises/list-ops}/list-ops.scm +0 -0
- data/tracks/scheme/{nucleotide-count → exercises/nucleotide-count}/example.scm +0 -0
- data/tracks/scheme/{nucleotide-count → exercises/nucleotide-count}/nucleotide-count-test.scm +0 -0
- data/tracks/scheme/{nucleotide-count → exercises/nucleotide-count}/nucleotide-count.scm +0 -0
- data/tracks/scheme/{phone-number → exercises/phone-number}/example.scm +0 -0
- data/tracks/scheme/{phone-number → exercises/phone-number}/phone-number-test.scm +0 -0
- data/tracks/scheme/{phone-number → exercises/phone-number}/phone-number.scm +0 -0
- data/tracks/scheme/{raindrops → exercises/raindrops}/example.scm +0 -0
- data/tracks/scheme/{raindrops → exercises/raindrops}/raindrops-test.scm +0 -0
- data/tracks/scheme/{raindrops → exercises/raindrops}/raindrops.scm +0 -0
- data/tracks/scheme/{rna-transcription → exercises/rna-transcription}/dna.scm +0 -0
- data/tracks/scheme/{rna-transcription → exercises/rna-transcription}/example.scm +0 -0
- data/tracks/scheme/{rna-transcription → exercises/rna-transcription}/rna-transcription-test.scm +0 -0
- data/tracks/scheme/{robot-name → exercises/robot-name}/example.scm +0 -0
- data/tracks/scheme/{robot-name → exercises/robot-name}/robot-name-test.scm +0 -0
- data/tracks/scheme/{robot-name → exercises/robot-name}/robot.scm +0 -0
- 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
|
+
}
|