aboutsummaryrefslogtreecommitdiffstats
path: root/src/blogc/loader.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/blogc/loader.c')
-rw-r--r--src/blogc/loader.c212
1 files changed, 212 insertions, 0 deletions
diff --git a/src/blogc/loader.c b/src/blogc/loader.c
new file mode 100644
index 0000000..90f2401
--- /dev/null
+++ b/src/blogc/loader.c
@@ -0,0 +1,212 @@
+/*
+ * blogc: A blog compiler.
+ * Copyright (C) 2015-2016 Rafael G. Martins <rafael@rafaelmartins.eng.br>
+ *
+ * This program can be distributed under the terms of the BSD License.
+ * See the file LICENSE.
+ */
+
+#include <math.h>
+#include <stdbool.h>
+#include <stddef.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include "file.h"
+#include "source-parser.h"
+#include "template-parser.h"
+#include "loader.h"
+#include "error.h"
+#include "../common/utils.h"
+
+
+char*
+blogc_get_filename(const char *f)
+{
+ if (f == NULL)
+ return NULL;
+
+ if (strlen(f) == 0)
+ return NULL;
+
+ char *filename = sb_strdup(f);
+
+ // keep a pointer to original string
+ char *tmp = filename;
+
+ bool removed_dot = false;
+ for (int i = strlen(tmp); i >= 0 ; i--) {
+
+ // remove last extension
+ if (!removed_dot && tmp[i] == '.') {
+ tmp[i] = '\0';
+ removed_dot = true;
+ continue;
+ }
+
+ if (tmp[i] == '/' || tmp[i] == '\\') {
+ tmp += i + 1;
+ break;
+ }
+ }
+
+ char *final_filename = sb_strdup(tmp);
+ free(filename);
+
+ return final_filename;
+}
+
+
+sb_slist_t*
+blogc_template_parse_from_file(const char *f, blogc_error_t **err)
+{
+ if (err == NULL || *err != NULL)
+ return NULL;
+ size_t len;
+ char *s = blogc_file_get_contents(f, &len, err);
+ if (s == NULL)
+ return NULL;
+ sb_slist_t *rv = blogc_template_parse(s, len, err);
+ free(s);
+ return rv;
+}
+
+
+sb_trie_t*
+blogc_source_parse_from_file(const char *f, blogc_error_t **err)
+{
+ if (err == NULL || *err != NULL)
+ return NULL;
+ size_t len;
+ char *s = blogc_file_get_contents(f, &len, err);
+ if (s == NULL)
+ return NULL;
+ sb_trie_t *rv = blogc_source_parse(s, len, err);
+
+ // set FILENAME variable
+ if (rv != NULL) {
+ char *filename = blogc_get_filename(f);
+ if (filename != NULL)
+ sb_trie_insert(rv, "FILENAME", filename);
+ }
+
+ free(s);
+ return rv;
+}
+
+
+sb_slist_t*
+blogc_source_parse_from_files(sb_trie_t *conf, sb_slist_t *l, blogc_error_t **err)
+{
+ blogc_error_t *tmp_err = NULL;
+ sb_slist_t *rv = NULL;
+ unsigned int with_date = 0;
+
+ const char *filter_tag = sb_trie_lookup(conf, "FILTER_TAG");
+ const char *filter_page = sb_trie_lookup(conf, "FILTER_PAGE");
+ const char *filter_per_page = sb_trie_lookup(conf, "FILTER_PER_PAGE");
+
+ long page = strtol(filter_page != NULL ? filter_page : "", NULL, 10);
+ if (page <= 0)
+ page = 1;
+ long per_page = strtol(filter_per_page != NULL ? filter_per_page : "10",
+ NULL, 10);
+ if (per_page <= 0)
+ per_page = 10;
+
+ // poor man's pagination
+ unsigned int start = (page - 1) * per_page;
+ unsigned int end = start + per_page;
+ unsigned int counter = 0;
+
+ for (sb_slist_t *tmp = l; tmp != NULL; tmp = tmp->next) {
+ char *f = tmp->data;
+ sb_trie_t *s = blogc_source_parse_from_file(f, &tmp_err);
+ if (s == NULL) {
+ *err = blogc_error_new_printf(BLOGC_ERROR_LOADER,
+ "An error occurred while parsing source file: %s\n\n%s",
+ f, tmp_err->msg);
+ blogc_error_free(tmp_err);
+ tmp_err = NULL;
+ sb_slist_free_full(rv, (sb_free_func_t) sb_trie_free);
+ rv = NULL;
+ break;
+ }
+ if (filter_tag != NULL) {
+ const char *tags_str = sb_trie_lookup(s, "TAGS");
+ // if user wants to filter by tag and no tag is provided, skip it
+ if (tags_str == NULL) {
+ sb_trie_free(s);
+ continue;
+ }
+ char **tags = sb_str_split(tags_str, ' ', 0);
+ bool found = false;
+ for (unsigned int i = 0; tags[i] != NULL; i++) {
+ if (tags[i][0] == '\0')
+ continue;
+ if (0 == strcmp(tags[i], filter_tag))
+ found = true;
+ }
+ sb_strv_free(tags);
+ if (!found) {
+ sb_trie_free(s);
+ continue;
+ }
+ }
+ if (filter_page != NULL) {
+ if (counter < start || counter >= end) {
+ counter++;
+ sb_trie_free(s);
+ continue;
+ }
+ counter++;
+ }
+ if (sb_trie_lookup(s, "DATE") != NULL)
+ with_date++;
+ rv = sb_slist_append(rv, s);
+ }
+
+ if (with_date > 0 && with_date < sb_slist_length(rv))
+ // fatal error, maybe?
+ blogc_fprintf(stderr,
+ "blogc: warning: 'DATE' variable provided for at least one source "
+ "file, but not for all source files. This means that you may get "
+ "wrong values for 'DATE_FIRST' and 'DATE_LAST' variables.\n");
+
+ bool first = true;
+ for (sb_slist_t *tmp = rv; tmp != NULL; tmp = tmp->next) {
+ sb_trie_t *s = tmp->data;
+ if (first) {
+ const char *val = sb_trie_lookup(s, "DATE");
+ if (val != NULL)
+ sb_trie_insert(conf, "DATE_FIRST", sb_strdup(val));
+ val = sb_trie_lookup(s, "FILENAME");
+ if (val != NULL)
+ sb_trie_insert(conf, "FILENAME_FIRST", sb_strdup(val));
+ first = false;
+ }
+ if (tmp->next == NULL) { // last
+ const char *val = sb_trie_lookup(s, "DATE");
+ if (val != NULL)
+ sb_trie_insert(conf, "DATE_LAST", sb_strdup(val));
+ val = sb_trie_lookup(s, "FILENAME");
+ if (val != NULL)
+ sb_trie_insert(conf, "FILENAME_LAST", sb_strdup(val));
+ }
+ }
+
+ if (filter_page != NULL) {
+ unsigned int last_page = ceilf(((float) counter) / per_page);
+ sb_trie_insert(conf, "CURRENT_PAGE", sb_strdup_printf("%ld", page));
+ if (page > 1)
+ sb_trie_insert(conf, "PREVIOUS_PAGE", sb_strdup_printf("%ld", page - 1));
+ if (page < last_page)
+ sb_trie_insert(conf, "NEXT_PAGE", sb_strdup_printf("%ld", page + 1));
+ if (sb_slist_length(rv) > 0)
+ sb_trie_insert(conf, "FIRST_PAGE", sb_strdup("1"));
+ if (last_page > 0)
+ sb_trie_insert(conf, "LAST_PAGE", sb_strdup_printf("%d", last_page));
+ }
+
+ return rv;
+}