aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMatthew Hall <hallmatthew314@gmail.com>2023-03-11 00:02:12 +1300
committerMatthew Hall <hallmatthew314@gmail.com>2023-03-11 00:02:12 +1300
commit60be5fc8a6be8f60ed37af30ef18f02639259774 (patch)
treeedcc91e1cb0cadd35c2fe9003300ccc085ec2c02
parent46943966888b5ed9db2e0a454754b0f8f9947c48 (diff)
Implement FirstOf
-rw-r--r--spec/parcom_spec.cr33
-rw-r--r--src/parcom.cr26
2 files changed, 57 insertions, 2 deletions
diff --git a/spec/parcom_spec.cr b/spec/parcom_spec.cr
index fd4b470..77a14dd 100644
--- a/spec/parcom_spec.cr
+++ b/spec/parcom_spec.cr
@@ -614,7 +614,38 @@ end
pending StopIf do
end
-pending FirstOf do
+describe FirstOf do
+ tokens = TokenStream.from_string("abcd")
+ letter_a = Token.new('a')
+ f = Flunk(Char, Char).new
+
+ describe "#parse" do
+ it "cannot be instantiated with an empty Enumerable" do
+ expect_raises(ArgumentError) { FirstOf.new([] of Parser(Char, Char)) }
+ end
+
+ it "uses the result of the first successful parser" do
+ a1 = [letter_a, f, f, f] of Parser(Char, Char)
+ a2 = [f, letter_a, f, f] of Parser(Char, Char)
+ a3 = [f, f, letter_a, f] of Parser(Char, Char)
+ a4 = [f, f, f, letter_a] of Parser(Char, Char)
+
+ [a1, a2, a3, a4].each do |arr|
+ p = FirstOf.new(arr)
+ r = p.parse(tokens)
+ r.value.should eq('a')
+ r.tokens.should eq(tokens[1..])
+ end
+ end
+
+ it "only fails if no parsers are successful" do
+ x = Token.new('x')
+ y = Token.new('x')
+ z = Token.new('x')
+ p = FirstOf.new([x, y, z] of Parser(Char, Char))
+ expect_raises(ParserException) { p.parse(tokens) }
+ end
+ end
end
pending SepBy do
diff --git a/src/parcom.cr b/src/parcom.cr
index cb3408a..2a61c2d 100644
--- a/src/parcom.cr
+++ b/src/parcom.cr
@@ -405,7 +405,31 @@ module Parcom
class StopIf
end
- class FirstOf
+ class FirstOf(T, V) < Parser(T, V)
+ @p : Parser(T, V)
+
+ def initialize(ps : Iterable(Parser(T, V)))
+ ps_iter = ps.each
+ p = ps_iter.next
+
+ if p.is_a?(Iterator::Stop)
+ raise ArgumentError.new("FirstOf requires atleast one parser, got none")
+ end
+
+ @p = p
+ p = ps_iter.next
+
+ until p.is_a?(Iterator::Stop)
+ @p = @p | p
+ p = ps_iter.next
+ end
+ end
+
+ def parse(tokens : TokenStream(T)) : Result(T, V)
+ @p.parse(tokens)
+ rescue ex : ParserException
+ raise ParserException.new("FirstOf: #{ex.message}")
+ end
end
class SepBy