diff options
author | Marvin Borner | 2024-11-27 02:12:12 +0100 |
---|---|---|
committer | Marvin Borner | 2024-11-27 02:27:48 +0100 |
commit | 6da602b0a29afcd2aa15725547375a80e30b3983 (patch) | |
tree | bb268dc7935696c2687c4ec151c2e0108536fa6f /src/parser.js |
initial commit
Diffstat (limited to 'src/parser.js')
-rw-r--r-- | src/parser.js | 28 |
1 files changed, 28 insertions, 0 deletions
diff --git a/src/parser.js b/src/parser.js new file mode 100644 index 0000000..a15c8e2 --- /dev/null +++ b/src/parser.js @@ -0,0 +1,28 @@ +import * as wrapper from "../src/wrapper.js" +import * as either from "../src/either.js" + +export const show = either => + either(v => "Error: " + v)(v => v(cur => rst => ({ cur, rst }))) + +export const unit = cur => rst => either.Right(s => s(cur)(rst)) +export const bind = p => f => s => either.bind(p(s))(right => right(cur => rst => f(cur)(rst))) + +export const DO = wrapper.DO(unit, bind) + +export const satisfy = pred => s => { + if (s === "") return either.Left("end of input") + const head = s[0] + const tail = s.slice(1) + return pred(head) ? either.Right(s => s(head)(tail)) + : either.Left("unexpected " + head) +} + +export const char = ch => satisfy(c => c == ch) + +export const string = str => DO(function* () { + const head = str[0] + const tail = str.slice(1) + yield char(head) + return yield tail === "" ? unit(str) + : bind(string(tail))(_ => unit(str)) +}) |