aboutsummaryrefslogtreecommitdiff
path: root/src/parser.js
diff options
context:
space:
mode:
authorMarvin Borner2024-11-27 02:12:12 +0100
committerMarvin Borner2024-11-27 02:27:48 +0100
commit6da602b0a29afcd2aa15725547375a80e30b3983 (patch)
treebb268dc7935696c2687c4ec151c2e0108536fa6f /src/parser.js
initial commit
Diffstat (limited to 'src/parser.js')
-rw-r--r--src/parser.js28
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))
+})