shortcode/shortcodes.rl

150 lines
3.8 KiB
Plaintext
Raw Normal View History

2023-07-11 09:58:58 -03:00
#include <string.h>
2023-07-12 16:22:54 -03:00
#include "shortcodes.h"
2023-07-10 18:11:54 -03:00
2023-07-10 16:39:01 -03:00
%%{
machine shortcode;
2023-07-10 18:11:54 -03:00
action mark {
mark = p;
}
spc = space*;
sep = space+;
2023-07-10 20:36:50 -03:00
path = (alnum | '/' )+;
2023-07-10 18:11:54 -03:00
2023-07-10 21:55:45 -03:00
name = (alpha+ path?)
2023-07-10 18:11:54 -03:00
> mark
2023-07-11 22:20:26 -03:00
%{ sc_list[c_sc].name.start = mark-start;
sc_list[c_sc].name.len = p-mark;
sc_list[c_sc].matching = 0;
};
2023-07-10 21:55:45 -03:00
argname = alpha+
2023-07-10 18:17:17 -03:00
> mark
2023-07-11 21:09:42 -03:00
% {
2023-07-11 22:20:26 -03:00
sc_list[c_sc].argnames[sc_list[c_sc].argcount].start = mark-start;
sc_list[c_sc].argnames[sc_list[c_sc].argcount].len = p-mark;
2023-07-11 21:09:42 -03:00
};
qvalue = ('"' [^"]* '"')
2023-07-10 18:11:54 -03:00
> mark
2023-07-11 21:09:42 -03:00
% {
2023-07-11 22:20:26 -03:00
sc_list[c_sc].argvals[sc_list[c_sc].argcount].start = mark-start+1;
sc_list[c_sc].argvals[sc_list[c_sc].argcount].len = p-mark-2;
sc_list[c_sc].argcount++;
2023-07-11 21:09:42 -03:00
};
value = alnum+
> mark
2023-07-11 21:09:42 -03:00
% {
2023-07-11 22:20:26 -03:00
sc_list[c_sc].argvals[sc_list[c_sc].argcount].start = mark-start;
sc_list[c_sc].argvals[sc_list[c_sc].argcount].len = p-mark;
sc_list[c_sc].argcount++;
2023-07-11 21:09:42 -03:00
};
arg = ((argname '=')? (value|qvalue));
2023-07-10 16:39:01 -03:00
start_p = '{{%';
end_p = '%}}';
2023-07-10 16:39:01 -03:00
start_b = '{{<';
end_b = '>}}';
start = start_p | start_b ;
2023-07-11 16:51:11 -03:00
end = end_p | end_b ;
shortcode = (start spc name (sep arg)* spc end)
2023-07-11 20:18:05 -03:00
> {
2023-07-11 22:20:26 -03:00
sc_list[c_sc].start = p-start-1;
2023-07-11 20:18:05 -03:00
}
2023-07-10 22:11:43 -03:00
@ {
2023-07-11 22:20:26 -03:00
sc_list[c_sc].len = p-start-sc_list[c_sc].start+1;
2023-07-11 20:18:05 -03:00
data_mark = p+1;
c_sc++;
2023-07-10 22:11:43 -03:00
};
2023-07-11 10:22:18 -03:00
closing_shortcode = (start spc '/' name spc end)
> {
2023-07-11 20:49:33 -03:00
// Starting a shortcode, close data (if previous shortcode is opening)
// If it's closing, then previous text can't be data.
2023-07-11 22:20:26 -03:00
if (sc_list[c_sc-1].matching == 0) {
sc_list[c_sc-1].data.start = data_mark-start;
sc_list[c_sc-1].data.len = p-data_mark-1;
2023-07-11 20:49:33 -03:00
}
2023-07-11 10:22:18 -03:00
};
2023-07-10 22:29:53 -03:00
matched_shortcode = (shortcode any* closing_shortcode)
2023-07-10 22:11:43 -03:00
@ {
2023-07-11 22:20:26 -03:00
sc_list[c_sc-1].matching = 1;
sc_list[c_sc-1].len = p-start-sc_list[c_sc-1].start + 1;
2023-07-11 20:49:33 -03:00
if (
2023-07-11 22:20:26 -03:00
sc_list[c_sc-1].name.len != sc_list[c_sc].name.len ||
2023-07-11 20:49:33 -03:00
strncmp(
2023-07-11 22:20:26 -03:00
start + sc_list[c_sc-1].name.start,
start + sc_list[c_sc].name.start,
sc_list[c_sc-1].name.len) !=0)
2023-07-11 20:49:33 -03:00
{
2023-07-12 16:22:54 -03:00
// printf("Mismatched tags!\n");
// str_copyb(&sc,start + sc_list[c_sc-1].name.start, sc_list[c_sc-1].name.len);
// printf("opened: %s\n", sc.s);
// str_copyb(&sc,start + sc_list[c_sc].name.start, sc_list[c_sc].name.len);
// printf("closed: %s\n", sc.s);
return NULL;
2023-07-10 22:26:19 -03:00
}
2023-07-11 20:49:33 -03:00
// Reuse this shortcode entry for next one
2023-07-11 22:20:26 -03:00
sc_list[c_sc].name.start = 0;
2023-07-10 22:11:43 -03:00
};
2023-07-10 20:36:50 -03:00
2023-07-10 22:29:53 -03:00
main := (any* (shortcode | matched_shortcode))*;
2023-07-10 16:39:01 -03:00
}%%
2023-07-11 20:18:05 -03:00
2023-07-12 16:22:54 -03:00
shortcode *parse(char *input) {
2023-07-11 20:18:05 -03:00
2023-07-11 21:09:42 -03:00
%%write data;
char *eof, *ts, *te = 0;
int cs, act = 0;
char *start = input;
char *p = input;
char *pe = p + strlen(input);
2023-07-11 22:20:26 -03:00
shortcode sc_list[1000];
2023-07-11 21:09:42 -03:00
int c_sc = 0;
int sc_start = 0;
int sc_end = 0;
char *mark = p;
char *data_mark = p;
%% write init;
%% write exec;
2023-07-12 16:22:54 -03:00
// for (int i=0; sc_list[i].name.start!=0; i++) {
// str_copyb(&sc, start + sc_list[i].name.start, sc_list[i].name.len);
// printf("sc_name: %s (%d)\n", sc.s, sc_list[i].matching );
// str_copyb(&sc, start + sc_list[i].start, sc_list[i].len);
// printf("full_sc: %s\n", sc.s);
// if (sc_list[i].matching) {
// str_copyb(&sc, start + sc_list[i].data.start, sc_list[i].data.len);
// printf("sc_data: %s\n", sc.s);
// }
// for (int j=0; j< sc_list[i].argcount; j++) {
// str_copyb(&sc, start + sc_list[i].argnames[j].start, sc_list[i].argnames[j].len);
// printf("argname %d: %s\n", j, sc.s);
// str_copyb(&sc, start + sc_list[i].argvals[j].start, sc_list[i].argvals[j].len);
// printf("argval %d: %s\n", j, sc.s);
// }
// }
// TODO: handle errors
return sc_list;
2023-07-11 20:18:05 -03:00
}
2023-07-11 10:22:18 -03:00
2023-07-12 16:09:17 -03:00
// int main(int argc, char **argv) {
// parse(
// "bbb{{% sarasa sar1 sar2 \"sar3\" %}}ccc"
// "{{< c1 arg2 >}}foobar{{% /c1%}}aaa{{% sarasa name=\"pepe\" %}}");
// return 0;
// };