aboutsummaryrefslogtreecommitdiff
path: root/src/template.c
blob: 5d400b99e02b8fcffd3b3e7cbf9555103b88ff16 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
#include <stdio.h>
#include <stdlib.h>

#include "vendor/mpc.h"
#include "template.h"

char *read_file(char *filename) {
    char *input = malloc(BUFSIZ);
    unsigned int size = 0;

    FILE *f = fopen(filename, "r");
    if (!f) {
        printf("Could not open \"%s\" for reading", filename);
        exit(1);
    }

    unsigned int read = 0;
    while ( (read = fread(input, 1, BUFSIZ, f)) > 0) {
        size += read;
        input = realloc(input, size + BUFSIZ);
    }

    fclose(f);

    input[size] = '\0';
    return input;
}

int eval(char *dest, mpc_ast_t* t, struct hashmap *ctx) {
    printf("%s: %s\n", t->tag, t->contents);

    if (strstr(t->tag, "content|text")) {
        strcat(dest, t->contents);
        return 0;
    }

    if (strstr(t->tag, "content|var")) {
        printf("Key: %s\n", t->contents);
        char *value = hashmap_get(ctx, t->children[1]->contents);
        if  (value == NULL) {
            return 1;
        }
        strcat(dest, value);
        return 0;
    }

    if (strstr(t->tag, "content|for")) {
        char *tmp_key = t->children[2]->contents;
        char *iterator_key = t->children[4]->contents;
        // TODO: Make this dynamic
        struct post **posts = hashmap_get(ctx, iterator_key);
        for (int i=0; i < 2; i++) {
            hashmap_insert(ctx, tmp_key, posts[i]);
            eval(dest, t->children[6], ctx);
        }
        return 0;
    }

    for (int i=0; i < t->children_num; i++) {
        eval(dest, t->children[i], ctx);
    }
    
    return 0;
}

mpc_parser_t *parser_init() {
    mpc_parser_t *Symbol = mpc_new("symbol");
    mpc_parser_t *Text = mpc_new("text");
    mpc_parser_t *Var_Open = mpc_new("var_open");
    mpc_parser_t *Var_Close = mpc_new("var_close");
    mpc_parser_t *Var = mpc_new("var");
    mpc_parser_t *Block_Open = mpc_new("block_open");
    mpc_parser_t *Block_Close = mpc_new("block_close");
    mpc_parser_t *For = mpc_new("for");
    mpc_parser_t *Content = mpc_new("content");
    mpc_parser_t *Template = mpc_new("template");
    mpca_lang(MPCA_LANG_WHITESPACE_SENSITIVE,
    " symbol    : /[a-zA-Z._]+/ ;"
    " var_open  : /\{{2} ?/ ;"
    " var_close : / ?}{2}/ ;"
    " var       : <var_open> <symbol> <var_close> ;"
    " block_open: /\{\% ?/ ;"
    " block_close: / ?\%}/ ;"
    " for       : <block_open> \"for \" <symbol> \" in \" <symbol> <block_close> <content>* <block_open> \"endfor\" <block_close> ;"
    " text      : /[^{][^{%]*/ ;"
    " content   : <var> | <for> | <text>;"
    " template  : /^/ <content>* /$/ ;",
    Symbol, Var_Open, Var_Close, Var, Block_Open, Block_Close, For, Text, Content, Template, NULL);
    return Template;
}

void template(char *tmpl, struct hashmap *ctx) {
    mpc_parser_t *parser = parser_init();
    mpc_result_t r;

    if (mpc_parse("input", tmpl, parser, &r)) {
        mpc_ast_print(r.output);
        
        // FIXME: Allocate precisely
        char *output = malloc(strlen(tmpl) * 2);
        output[0] = '\0';
        eval(output, r.output, ctx);
        printf("Template: \n%s", output);
        
        mpc_ast_delete(r.output);
        free(output);
    } else {
        mpc_err_print(r.error);
        mpc_err_delete(r.error);
    }
}