diff options
| author | Matthew Hall <hallmatthew314@gmail.com> | 2023-03-08 20:53:17 +1300 |
|---|---|---|
| committer | Matthew Hall <hallmatthew314@gmail.com> | 2023-03-08 20:53:17 +1300 |
| commit | b61aba7186cb7c616d6c8f9a3b9e6442173cbbf6 (patch) | |
| tree | b5be1d3315da3ac7afa8fd74973d224fcb6636a1 | |
| parent | 61a58de90c2781562aacc8006616e97a3ff12005 (diff) | |
Implement Optional (sort of)
| -rw-r--r-- | spec/parcom_spec.cr | 23 | ||||
| -rw-r--r-- | src/parcom.cr | 19 |
2 files changed, 39 insertions, 3 deletions
diff --git a/spec/parcom_spec.cr b/spec/parcom_spec.cr index d9af3b9..2f3f6ba 100644 --- a/spec/parcom_spec.cr +++ b/spec/parcom_spec.cr @@ -348,7 +348,7 @@ describe Recover do result.value.should eq('t') end - it "succeeds and returns the default value without modifying the input of the wrapped parser fails" do + it "succeeds and returns the default value without modifying the input if the wrapped parser fails" do tokens = TokenStream.from_string("_____") result = p.parse(tokens) @@ -358,7 +358,26 @@ describe Recover do end end -pending Optional do +describe Optional do + p = Optional.new(Token.new('t')) + + describe "#parse" do + it "succeeds and returns the wrapped parser's value if it succeeds" do + tokens = TokenStream.from_string("testing") + result = p.parse(tokens) + + result.tokens.should eq(tokens[1..]) + result.value.should eq('t') + end + + it "succeeds and returns a value of `nil` without modifying the input if the wrapped parser fails" do + tokens = TokenStream.from_string("_____") + result = p.parse(tokens) + + result.tokens.should eq(tokens) + result.value.should be_nil + end + end end pending Tokens do diff --git a/src/parcom.cr b/src/parcom.cr index f102728..87c73c5 100644 --- a/src/parcom.cr +++ b/src/parcom.cr @@ -201,7 +201,24 @@ module Parcom end end - class Optional + # TODO: find a way to make this use Recover on the back end + class Optional(T, V) < Parser(T, V?) + def initialize(@p : Parser(T, V)) + end + + def parse(tokens : TokenStream(T)) : Result(T, V?) + r = @p.parse?(tokens) + + if r.nil? + Result.new(tokens, nil) + else + Result.new(r.tokens, r.value || nil) + end + + new_tokens = r.nil? ? tokens : r.tokens + new_value = r.nil? ? nil : r.value + Result.new(new_tokens, new_value) + end end class Tokens |
