Skip to content

Commit c24e27f

Browse files
committed
feat: add sepby1 combinator
`sepby1` is a common reusable combinator in Haskell Parsec. This adds `mpc_sepby1(mpc_fold_t f, mpc_parser_t *sep, mpc_parser_t *a)` according to Haskell's implementation: https://hackage.haskell.org/package/parsec-3.1.16.1/docs/src/Text.Parsec.Combinator.html#sepBy1 Reuses existing `mpc_and`, `mpc_many`, and `mpcf_snd_free`.
1 parent 6912695 commit c24e27f

File tree

4 files changed

+29
-0
lines changed

4 files changed

+29
-0
lines changed

README.md

+8
Original file line numberDiff line numberDiff line change
@@ -356,6 +356,14 @@ Runs `a` one or more times until it fails. Results are combined with fold functi
356356
357357
* * *
358358
359+
```c
360+
mpc_parser_t *mpc_sepby1(mpc_fold_t f, mpc_parser_t *sep, mpc_parser_t *a);
361+
```
362+
363+
Runs `a` one or more times, separated by `sep`. Results are combined with fold function `f`.
364+
365+
* * *
366+
359367
```c
360368
mpc_parser_t *mpc_count(int n, mpc_fold_t f, mpc_parser_t *a, mpc_dtor_t da);
361369
```

mpc.c

+6
Original file line numberDiff line numberDiff line change
@@ -2120,6 +2120,12 @@ mpc_parser_t *mpc_tok_braces(mpc_parser_t *a, mpc_dtor_t ad) { return mpc_tok_
21202120
mpc_parser_t *mpc_tok_brackets(mpc_parser_t *a, mpc_dtor_t ad) { return mpc_tok_between(a, ad, "{", "}"); }
21212121
mpc_parser_t *mpc_tok_squares(mpc_parser_t *a, mpc_dtor_t ad) { return mpc_tok_between(a, ad, "[", "]"); }
21222122

2123+
mpc_parser_t *mpc_sepby1(mpc_fold_t f, mpc_parser_t *sep, mpc_parser_t *a) {
2124+
return mpc_and(2, f,
2125+
a, mpc_many(f, mpc_and(2, mpcf_snd_free, sep, mpc_copy(a), free)),
2126+
free);
2127+
}
2128+
21232129
/*
21242130
** Regular Expression Parsers
21252131
*/

mpc.h

+2
Original file line numberDiff line numberDiff line change
@@ -220,6 +220,8 @@ mpc_parser_t *mpc_tok_braces(mpc_parser_t *a, mpc_dtor_t ad);
220220
mpc_parser_t *mpc_tok_brackets(mpc_parser_t *a, mpc_dtor_t ad);
221221
mpc_parser_t *mpc_tok_squares(mpc_parser_t *a, mpc_dtor_t ad);
222222

223+
mpc_parser_t *mpc_sepby1(mpc_fold_t f, mpc_parser_t *sep, mpc_parser_t *a);
224+
223225
/*
224226
** Common Function Parameters
225227
*/

tests/core.c

+13
Original file line numberDiff line numberDiff line change
@@ -242,6 +242,18 @@ void test_eoi(void) {
242242

243243
}
244244

245+
void test_sepby(void) {
246+
mpc_parser_t* CommaSepIdent = mpc_sepby1(mpcf_strfold, mpc_char(','), mpc_ident());
247+
248+
PT_ASSERT(mpc_test_fail(CommaSepIdent, "", "", streq, free, strprint));
249+
PT_ASSERT(mpc_test_pass(CommaSepIdent, "one", "one", streq, free, strprint));
250+
PT_ASSERT(mpc_test_pass(CommaSepIdent, "one,", "one", streq, free, strprint));
251+
PT_ASSERT(mpc_test_pass(CommaSepIdent, "one,two,", "onetwo", streq, free, strprint));
252+
PT_ASSERT(mpc_test_pass(CommaSepIdent, "one,two,three", "onetwothree", streq, free, strprint));
253+
254+
mpc_delete(CommaSepIdent);
255+
}
256+
245257
void suite_core(void) {
246258
pt_add_test(test_ident, "Test Ident", "Suite Core");
247259
pt_add_test(test_maths, "Test Maths", "Suite Core");
@@ -251,4 +263,5 @@ void suite_core(void) {
251263
pt_add_test(test_reader, "Test Reader", "Suite Core");
252264
pt_add_test(test_tokens, "Test Tokens", "Suite Core");
253265
pt_add_test(test_eoi, "Test EOI", "Suite Core");
266+
pt_add_test(test_sepby, "Test SepBy", "Suite Core");
254267
}

0 commit comments

Comments
 (0)