Compare commits
8 Commits
main
...
34a16bd6bd
Author | SHA1 | Date | |
---|---|---|---|
34a16bd6bd | |||
4dcd067ca7 | |||
35acfee0b6 | |||
f6e775474d | |||
96f08fcaa0 | |||
47ee73edbe | |||
f471642f43 | |||
5350dbd126 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -54,3 +54,4 @@ dkms.conf
|
||||
shortcodes.c
|
||||
shortcodes
|
||||
tests
|
||||
.cgreen*
|
||||
|
25
Makefile
25
Makefile
@@ -1,16 +1,13 @@
|
||||
CC=tcc
|
||||
all: tests
|
||||
run: shortcodes
|
||||
./shortcodes
|
||||
CC=gcc
|
||||
all: test
|
||||
shortcodes.c: shortcodes.rl
|
||||
ragel -G2 shortcodes.rl -o shortcodes.c
|
||||
shortcodes: shortcodes.c
|
||||
$(CC) shortcodes.c -g -o shortcodes
|
||||
tests: shortcodes.c shortcodes.h tests.c
|
||||
$(CC) tests.c shortcodes.c -lbg -lcgreen -g -o tests
|
||||
test: tests
|
||||
./tests
|
||||
ragel -G2 -L shortcodes.rl -o shortcodes.c
|
||||
tests.so: shortcodes.c tests.c
|
||||
$(CC) -fPIC -shared -g -o $@ $^ -lbg -lcgreen
|
||||
clean:
|
||||
rm -f shortcodes shortcodes.c
|
||||
|
||||
.PHONY: test run
|
||||
rm -f shortcodes.c *.o *.so tests
|
||||
test: tests.so
|
||||
cgreen-runner $^
|
||||
debug:
|
||||
cgreen-debug tests.so
|
||||
.PHONY: test debug
|
||||
|
2
TODO.md
2
TODO.md
@@ -11,5 +11,5 @@
|
||||
* Add self-closing shortcodes `{{< foo />}}`
|
||||
(also, how TF do they work?)
|
||||
* Add inline shortcodes `{{< time.inline >}}{{ now }}{{< /time.inline >}}`
|
||||
|
||||
* Start integrating with Crystal
|
||||
* ~~Detect mismatched start/end like {{% foo >}}~~
|
||||
|
33
shortcodes.h
33
shortcodes.h
@@ -1,5 +1,4 @@
|
||||
#ifndef SHORTCODES_H
|
||||
#define SHORTCODES_H
|
||||
#pragma once
|
||||
|
||||
// A chunk is a reference to a piece of string
|
||||
// from "start" relative to its start
|
||||
@@ -10,6 +9,14 @@ struct chunk
|
||||
};
|
||||
typedef struct chunk chunk;
|
||||
|
||||
// An error
|
||||
struct sc_error
|
||||
{
|
||||
int position;
|
||||
int code;
|
||||
};
|
||||
typedef struct sc_error sc_error;
|
||||
|
||||
// Describes a parsed shortcode
|
||||
struct shortcode
|
||||
{
|
||||
@@ -23,5 +30,23 @@ struct shortcode
|
||||
};
|
||||
typedef struct shortcode shortcode;
|
||||
|
||||
shortcode *parse(char *);
|
||||
#endif
|
||||
struct sc_result
|
||||
{
|
||||
shortcode sc[100];
|
||||
sc_error errors[10];
|
||||
int errcount;
|
||||
};
|
||||
typedef struct sc_result sc_result;
|
||||
|
||||
// Error codes
|
||||
|
||||
/* You are closing the wrong shortcode.
|
||||
Example:
|
||||
|
||||
{{% foo %}} {{% /bar %}}
|
||||
*/
|
||||
#define ERR_MISMATCHED_CLOSING_TAG 1
|
||||
#define ERR_MISMATCHED_BRACKET 2
|
||||
|
||||
|
||||
sc_result parse(char *);
|
||||
|
@@ -1,7 +1,6 @@
|
||||
#include <string.h>
|
||||
#include "shortcodes.h"
|
||||
|
||||
|
||||
%%{
|
||||
machine shortcode;
|
||||
|
||||
@@ -59,6 +58,9 @@
|
||||
// Since it's mismatched, remove the name
|
||||
sc_list[c_sc].name.start = 0;
|
||||
sc_list[c_sc].name.len=0;
|
||||
result.errors[result.errcount].position = p-start-2;
|
||||
result.errors[result.errcount].code = ERR_MISMATCHED_BRACKET;
|
||||
result.errcount++;
|
||||
};
|
||||
|
||||
shortcode = ((start_p content end_p) | (start_b content end_b))
|
||||
@@ -93,8 +95,10 @@
|
||||
start + sc_list[c_sc-1].name.start,
|
||||
start + sc_list[c_sc].name.start,
|
||||
sc_list[c_sc-1].name.len) !=0)
|
||||
{
|
||||
return NULL;
|
||||
{
|
||||
result.errors[result.errcount].position = p-start;
|
||||
result.errors[result.errcount].code = ERR_MISMATCHED_CLOSING_TAG;
|
||||
result.errcount++;
|
||||
}
|
||||
// Reuse this shortcode entry for next one
|
||||
sc_list[c_sc].name.start = 0;
|
||||
@@ -105,7 +109,7 @@
|
||||
}%%
|
||||
|
||||
|
||||
shortcode *parse(char *input) {
|
||||
sc_result parse(char *input) {
|
||||
|
||||
%%write data;
|
||||
char *eof, *ts, *te = 0;
|
||||
@@ -115,17 +119,15 @@ shortcode *parse(char *input) {
|
||||
char *p = input;
|
||||
char *pe = p + strlen(input);
|
||||
|
||||
shortcode sc_list[1000];
|
||||
sc_result result;
|
||||
shortcode *sc_list = result.sc;
|
||||
int c_sc = 0;
|
||||
|
||||
int sc_start = 0;
|
||||
int sc_end = 0;
|
||||
|
||||
char *mark = p;
|
||||
char *data_mark = p;
|
||||
|
||||
%% write init;
|
||||
%% write exec;
|
||||
|
||||
return sc_list;
|
||||
return result;
|
||||
}
|
||||
|
131
tests.c
Executable file → Normal file
131
tests.c
Executable file → Normal file
@@ -6,10 +6,11 @@ Describe(parse);
|
||||
BeforeEach(parse) {}
|
||||
AfterEach(parse) {}
|
||||
|
||||
shortcode *result;
|
||||
sc_result result;
|
||||
str s;
|
||||
|
||||
chunk_s(char *buffer, chunk c){
|
||||
void chunk_s(char *buffer, chunk c)
|
||||
{
|
||||
str_copyb(&s, buffer + c.start, c.len);
|
||||
}
|
||||
|
||||
@@ -18,7 +19,7 @@ Ensure(parse, empty_string)
|
||||
char *input = "";
|
||||
result = parse(input);
|
||||
// This means no shortcodes in it
|
||||
assert_that(result[0].name.len, is_equal_to(0));
|
||||
assert_that(result.sc[0].name.len, is_equal_to(0));
|
||||
}
|
||||
|
||||
Ensure(parse, simple_shortcode)
|
||||
@@ -26,15 +27,15 @@ Ensure(parse, simple_shortcode)
|
||||
char *input = "foobar {{% shortcode %}}blah";
|
||||
result = parse(input);
|
||||
// Only 1 shortcode
|
||||
assert_that(result[1].name.len, is_equal_to(0));
|
||||
assert_that(result.sc[1].name.len, is_equal_to(0));
|
||||
|
||||
// It's a simple one called shortcode, no args
|
||||
chunk_s(input, result[0].name);
|
||||
chunk_s(input, result.sc[0].name);
|
||||
assert_that(s.s, is_equal_to_string("shortcode"));
|
||||
assert_that(result[0].matching, is_equal_to(0));
|
||||
assert_that(result[0].argcount, is_equal_to(0));
|
||||
assert_that(result.sc[0].matching, is_equal_to(0));
|
||||
assert_that(result.sc[0].argcount, is_equal_to(0));
|
||||
// The whole shortcode is the whole thing
|
||||
chunk_s(input, result[0].whole);
|
||||
chunk_s(input, result.sc[0].whole);
|
||||
assert_that(s.s, is_equal_to_string("{{% shortcode %}}"));
|
||||
}
|
||||
|
||||
@@ -43,7 +44,11 @@ Ensure(parse, mismatched_brackets)
|
||||
char *input = "foobar {{% shortcode >}}blah";
|
||||
result = parse(input);
|
||||
// No shortcodes
|
||||
assert_that(result[0].name.len, is_equal_to(0));
|
||||
assert_that(result.sc[0].name.len, is_equal_to(0));
|
||||
assert_that(result.errcount, is_equal_to(1));
|
||||
assert_that(result.errors[0].code, is_equal_to(ERR_MISMATCHED_BRACKET));
|
||||
str_copyb(&s, input + result.errors[0].position, 3);
|
||||
assert_that(s.s, is_equal_to_string(">}}"));
|
||||
}
|
||||
|
||||
Ensure(parse, mismatched_brackets_inside_data_are_ok)
|
||||
@@ -51,10 +56,10 @@ Ensure(parse, mismatched_brackets_inside_data_are_ok)
|
||||
char *input = "foobar {{% sc %}} >}}blah {{% /sc %}} ";
|
||||
result = parse(input);
|
||||
// 1 shortcode
|
||||
assert_that(result[1].name.len, is_equal_to(0));
|
||||
chunk_s(input, result[0].whole);
|
||||
assert_that(result.sc[1].name.len, is_equal_to(0));
|
||||
chunk_s(input, result.sc[0].whole);
|
||||
assert_that(s.s, is_equal_to_string("{{% sc %}} >}}blah {{% /sc %}}"));
|
||||
chunk_s(input, result[0].data);
|
||||
chunk_s(input, result.sc[0].data);
|
||||
assert_that(s.s, is_equal_to_string(" >}}blah "));
|
||||
}
|
||||
|
||||
@@ -63,10 +68,10 @@ Ensure(parse, mismatched_brackets_in_qval_are_ok)
|
||||
char *input = "foobar {{% sc \">}}blah\" %}} {{% /sc %}}";
|
||||
result = parse(input);
|
||||
// 1 shortcode
|
||||
assert_that(result[1].name.len, is_equal_to(0));
|
||||
chunk_s(input, result[0].whole);
|
||||
assert_that(result.sc[1].name.len, is_equal_to(0));
|
||||
chunk_s(input, result.sc[0].whole);
|
||||
assert_that(s.s, is_equal_to_string("{{% sc \">}}blah\" %}} {{% /sc %}}"));
|
||||
chunk_s(input, result[0].argvals[0]);
|
||||
chunk_s(input, result.sc[0].argvals[0]);
|
||||
assert_that(s.s, is_equal_to_string(">}}blah"));
|
||||
}
|
||||
|
||||
@@ -75,15 +80,15 @@ Ensure(parse, inner_spaces_optional)
|
||||
char *input = "foobar {{% shortcode%}}blah";
|
||||
result = parse(input);
|
||||
// Only 1 shortcode
|
||||
assert_that(result[1].name.len, is_equal_to(0));
|
||||
assert_that(result.sc[1].name.len, is_equal_to(0));
|
||||
|
||||
// It's a simple one called shortcode, no args
|
||||
chunk_s(input, result[0].name);
|
||||
chunk_s(input, result.sc[0].name);
|
||||
assert_that(s.s, is_equal_to_string("shortcode"));
|
||||
assert_that(result[0].matching, is_equal_to(0));
|
||||
assert_that(result[0].argcount, is_equal_to(0));
|
||||
assert_that(result.sc[0].matching, is_equal_to(0));
|
||||
assert_that(result.sc[0].argcount, is_equal_to(0));
|
||||
// The whole shortcode is the whole thing
|
||||
chunk_s(input, result[0].whole);
|
||||
chunk_s(input, result.sc[0].whole);
|
||||
assert_that(s.s, is_equal_to_string("{{% shortcode%}}"));
|
||||
}
|
||||
|
||||
@@ -92,13 +97,13 @@ Ensure(parse, name_can_be_path)
|
||||
char *input = "foobar {{% shortcode/foo/bar %}}blah";
|
||||
result = parse(input);
|
||||
// Only 1 shortcode
|
||||
assert_that(result[1].name.len, is_equal_to(0));
|
||||
assert_that(result.sc[1].name.len, is_equal_to(0));
|
||||
|
||||
// It's a simple one called shortcode, no args
|
||||
chunk_s(input, result[0].name);
|
||||
chunk_s(input, result.sc[0].name);
|
||||
assert_that(s.s, is_equal_to_string("shortcode/foo/bar"));
|
||||
assert_that(result[0].matching, is_equal_to(0));
|
||||
assert_that(result[0].argcount, is_equal_to(0));
|
||||
assert_that(result.sc[0].matching, is_equal_to(0));
|
||||
assert_that(result.sc[0].argcount, is_equal_to(0));
|
||||
}
|
||||
|
||||
Ensure(parse, multiple_shortcodes)
|
||||
@@ -106,23 +111,23 @@ Ensure(parse, multiple_shortcodes)
|
||||
char *input = "foobar {{% shortcode %}}blah {{<sc2 >}}blahblah";
|
||||
result = parse(input);
|
||||
// 2 shortcodes
|
||||
assert_that(result[2].name.len, is_equal_to(0));
|
||||
assert_that(result.sc[2].name.len, is_equal_to(0));
|
||||
|
||||
// It's a simple one called shortcode, no args
|
||||
chunk_s(input, result[0].name);
|
||||
chunk_s(input, result.sc[0].name);
|
||||
assert_that(s.s, is_equal_to_string("shortcode"));
|
||||
assert_that(result[0].matching, is_equal_to(0));
|
||||
assert_that(result[0].argcount, is_equal_to(0));
|
||||
assert_that(result.sc[0].matching, is_equal_to(0));
|
||||
assert_that(result.sc[0].argcount, is_equal_to(0));
|
||||
// The whole shortcode is the whole thing
|
||||
chunk_s(input, result[0].whole);
|
||||
chunk_s(input, result.sc[0].whole);
|
||||
assert_that(s.s, is_equal_to_string("{{% shortcode %}}"));
|
||||
// It's a simple one called sc2, no args
|
||||
chunk_s(input, result[1].name);
|
||||
chunk_s(input, result.sc[1].name);
|
||||
assert_that(s.s, is_equal_to_string("sc2"));
|
||||
assert_that(result[1].matching, is_equal_to(0));
|
||||
assert_that(result[1].argcount, is_equal_to(0));
|
||||
assert_that(result.sc[1].matching, is_equal_to(0));
|
||||
assert_that(result.sc[1].argcount, is_equal_to(0));
|
||||
// The whole shortcode is the whole thing
|
||||
chunk_s(input, result[1].whole);
|
||||
chunk_s(input, result.sc[1].whole);
|
||||
assert_that(s.s, is_equal_to_string("{{<sc2 >}}"));
|
||||
}
|
||||
|
||||
@@ -132,18 +137,18 @@ Ensure(parse, matching_shortcode)
|
||||
result = parse(input);
|
||||
|
||||
// Only 1 shortcode
|
||||
assert_that(result[1].name.len, is_equal_to(0));
|
||||
assert_that(result.sc[1].name.len, is_equal_to(0));
|
||||
|
||||
// It's a matching one called shortcode, no args
|
||||
chunk_s(input, result[0].name);
|
||||
chunk_s(input, result.sc[0].name);
|
||||
assert_that(s.s, is_equal_to_string("shortcode"));
|
||||
assert_that(result[0].matching, is_equal_to(1));
|
||||
assert_that(result[0].argcount, is_equal_to(0));
|
||||
assert_that(result.sc[0].matching, is_equal_to(1));
|
||||
assert_that(result.sc[0].argcount, is_equal_to(0));
|
||||
// data is the stuff between the shortcode tags
|
||||
chunk_s(input, result[0].data);
|
||||
chunk_s(input, result.sc[0].data);
|
||||
assert_that(s.s, is_equal_to_string("foo bar"));
|
||||
// The whole shortcode is the whole thing
|
||||
chunk_s(input, result[0].whole);
|
||||
chunk_s(input, result.sc[0].whole);
|
||||
assert_that(s.s, is_equal_to_string("{{% shortcode %}}foo bar{{% /shortcode %}}"));
|
||||
}
|
||||
|
||||
@@ -152,53 +157,54 @@ Ensure(parse, shortcode_args)
|
||||
char *input = "foobar {{% shortcode foo \"bar\" 42 bat=v1 baz=\"v2\" %}}blah";
|
||||
result = parse(input);
|
||||
// Only 1 shortcode
|
||||
assert_that(result[1].name.len, is_equal_to(0));
|
||||
assert_that(result.sc[1].name.len, is_equal_to(0));
|
||||
|
||||
// The whole shortcode is the whole thing
|
||||
chunk_s(input, result[0].whole);
|
||||
chunk_s(input, result.sc[0].whole);
|
||||
assert_that(s.s, is_equal_to_string("{{% shortcode foo \"bar\" 42 bat=v1 baz=\"v2\" %}}"));
|
||||
|
||||
// Name is shortcode
|
||||
chunk_s(input, result[0].name);
|
||||
chunk_s(input, result.sc[0].name);
|
||||
assert_that(s.s, is_equal_to_string("shortcode"));
|
||||
assert_that(result[0].matching, is_equal_to(0));
|
||||
assert_that(result.sc[0].matching, is_equal_to(0));
|
||||
|
||||
// Has 5 args
|
||||
assert_that(result[0].argcount, is_equal_to(5));
|
||||
assert_that(result.sc[0].argcount, is_equal_to(5));
|
||||
|
||||
// Arg1 is foo, no name
|
||||
assert_that(result[0].argnames[0].len, is_equal_to(0));
|
||||
chunk_s(input, result[0].argvals[0]);
|
||||
assert_that(result.sc[0].argnames[0].len, is_equal_to(0));
|
||||
chunk_s(input, result.sc[0].argvals[0]);
|
||||
assert_that(s.s, is_equal_to_string("foo"));
|
||||
// Arg2 is bar, no name
|
||||
assert_that(result[0].argnames[1].len, is_equal_to(0));
|
||||
chunk_s(input, result[0].argvals[1]);
|
||||
assert_that(result.sc[0].argnames[1].len, is_equal_to(0));
|
||||
chunk_s(input, result.sc[0].argvals[1]);
|
||||
assert_that(s.s, is_equal_to_string("bar"));
|
||||
// Arg3 is 42, no name
|
||||
assert_that(result[0].argnames[2].len, is_equal_to(0));
|
||||
chunk_s(input, result[0].argvals[2]);
|
||||
assert_that(result.sc[0].argnames[2].len, is_equal_to(0));
|
||||
chunk_s(input, result.sc[0].argvals[2]);
|
||||
assert_that(s.s, is_equal_to_string("42"));
|
||||
// Arg4 is bat=v1
|
||||
chunk_s(input, result[0].argnames[3]);
|
||||
chunk_s(input, result.sc[0].argnames[3]);
|
||||
assert_that(s.s, is_equal_to_string("bat"));
|
||||
chunk_s(input, result[0].argvals[3]);
|
||||
chunk_s(input, result.sc[0].argvals[3]);
|
||||
assert_that(s.s, is_equal_to_string("v1"));
|
||||
// Arg5 is baz=v2
|
||||
chunk_s(input, result[0].argnames[4]);
|
||||
chunk_s(input, result.sc[0].argnames[4]);
|
||||
assert_that(s.s, is_equal_to_string("baz"));
|
||||
chunk_s(input, result[0].argvals[4]);
|
||||
chunk_s(input, result.sc[0].argvals[4]);
|
||||
assert_that(s.s, is_equal_to_string("v2"));
|
||||
}
|
||||
|
||||
Ensure(parse, escaped_shortcode)
|
||||
{
|
||||
char *input = "foobar \\{{% shortcode %}}";
|
||||
result = parse(input);
|
||||
// No shortcodes
|
||||
assert_that(result[0].name.len, is_equal_to(0));
|
||||
}
|
||||
|
||||
// BUG?
|
||||
// Ensure(parse, escaped_shortcode)
|
||||
// {
|
||||
// char *input = "foobar \\{{% shortcode %}}";
|
||||
// result = parse(input);
|
||||
// // No shortcodes
|
||||
// assert_that(result.sc[0].name.len, is_equal_to(0));
|
||||
// }
|
||||
|
||||
#ifdef MAIN
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
str_init(&s);
|
||||
@@ -218,3 +224,4 @@ int main(int argc, char **argv)
|
||||
// add_test_with_context(suite, parse, escaped_shortcode);
|
||||
return run_test_suite(suite, create_text_reporter());
|
||||
}
|
||||
#endif
|
Reference in New Issue
Block a user