2022-02-13 01:35:02 -05:00
|
|
|
#include "lex.h"
|
|
|
|
#include <unistd.h> /* chdir (POSIX) */
|
|
|
|
#include <sys/types.h> /* mode_t (POSIX) */
|
|
|
|
#include <sys/stat.h> /* umask (POSIX) */
|
|
|
|
#include <dirent.h> /* opendir readdir closedir */
|
2022-02-10 23:30:11 -05:00
|
|
|
#include <limits.h>
|
2022-02-13 01:35:02 -05:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2022-02-10 23:30:11 -05:00
|
|
|
|
2022-02-11 04:46:41 -05:00
|
|
|
#if INT_MAX >= 100000000000
|
|
|
|
#error int_to_string requires truncation on this compiler.
|
|
|
|
#endif
|
|
|
|
static void int_to_string(const int *const n,
|
|
|
|
char (*const a)[12]) { sprintf(*a, "%d", *n); }
|
|
|
|
#define ARRAY_NAME int
|
|
|
|
#define ARRAY_TYPE int
|
|
|
|
#define ARRAY_EXPECT_TRAIT
|
2022-02-15 16:05:15 -05:00
|
|
|
#include "array.h"
|
2022-02-11 04:46:41 -05:00
|
|
|
#define ARRAY_TO_STRING &int_to_string
|
2022-02-15 16:05:15 -05:00
|
|
|
#include "array.h"
|
2022-02-11 04:46:41 -05:00
|
|
|
static int int_cmp(const int *const a, const int *const b)
|
|
|
|
{ return (*b < *a) - (*a < *b); }
|
|
|
|
static int void_int_cmp(const void *const a, const void *const b)
|
|
|
|
{ return int_cmp(a, b); }
|
2022-02-10 23:30:11 -05:00
|
|
|
|
2022-02-13 23:45:38 -05:00
|
|
|
#define ARRAY_NAME char
|
|
|
|
#define ARRAY_TYPE char
|
|
|
|
#include "array.h"
|
|
|
|
/** Append a text file, `fn`, to `c`, and add a '\0'.
|
|
|
|
@return Success. A partial read is failure. @throws[fopen, fread, malloc]
|
|
|
|
@throws[EISEQ] The text file has embedded nulls.
|
|
|
|
@throws[ERANGE] If the standard library does not follow POSIX. */
|
|
|
|
static int append_file(struct char_array *c, const char *const fn) {
|
|
|
|
FILE *fp = 0;
|
|
|
|
const size_t granularity = 1024;
|
|
|
|
size_t nread;
|
|
|
|
char *cursor;
|
|
|
|
int success = 0;
|
|
|
|
assert(c && fn);
|
|
|
|
if(!(fp = fopen(fn, "r"))) goto catch;
|
|
|
|
/* Read entire file in chunks. */
|
|
|
|
do if(!(cursor = char_array_buffer(c, granularity))
|
|
|
|
|| (nread = fread(cursor, 1, granularity, fp), ferror(fp))
|
|
|
|
|| !char_array_append(c, nread)) goto catch;
|
|
|
|
while(nread == granularity);
|
|
|
|
/* File to `C` string. */
|
|
|
|
if(!(cursor = char_array_new(c))) goto catch;
|
|
|
|
*cursor = '\0';
|
|
|
|
/* Binary files with embedded '\0' are not allowed. */
|
|
|
|
if(strchr(c->data, '\0') != cursor) { errno = EILSEQ; goto catch; }
|
|
|
|
{ success = 1; goto finally; }
|
|
|
|
catch:
|
|
|
|
if(!errno) errno = EILSEQ; /* Will never be true on POSIX. */
|
|
|
|
finally:
|
|
|
|
if(fp && fclose(fp)) success = 0;
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
2022-02-10 23:30:11 -05:00
|
|
|
int main(int argc, char **argv) {
|
|
|
|
int success = EXIT_FAILURE;
|
2022-02-13 01:17:45 -05:00
|
|
|
DIR *dir = 0;
|
|
|
|
struct dirent *de;
|
|
|
|
struct int_array years = ARRAY_IDLE, months = ARRAY_IDLE, days = ARRAY_IDLE;
|
2022-02-13 23:45:38 -05:00
|
|
|
struct char_array entry = ARRAY_IDLE;
|
2022-02-13 01:17:45 -05:00
|
|
|
int *y, *y_end;
|
2022-02-11 03:28:28 -05:00
|
|
|
|
2022-02-11 04:46:41 -05:00
|
|
|
/* Get the years list as directories matching a year in order. */
|
2022-02-11 03:28:28 -05:00
|
|
|
errno = 0;
|
2022-02-10 23:30:11 -05:00
|
|
|
if(argc != 2) { fprintf(stderr, "Needs journal location.\n"
|
|
|
|
"(should contain <year>/<month>/<day>.txt)\n"); goto finally; }
|
2022-02-13 01:17:45 -05:00
|
|
|
if(chdir(argv[1]) == -1 || !(dir = opendir("."))) goto catch;
|
|
|
|
while((de = readdir(dir))) {
|
2022-02-11 03:28:28 -05:00
|
|
|
struct stat st;
|
2022-02-11 04:46:41 -05:00
|
|
|
int year, *p;
|
2022-02-13 23:45:38 -05:00
|
|
|
if(!lex_looks_like_year(de->d_name, &year)) continue;
|
2022-02-13 01:17:45 -05:00
|
|
|
if(stat(de->d_name, &st)) goto catch;
|
2022-02-11 04:46:41 -05:00
|
|
|
if(!S_ISDIR(st.st_mode)) continue;
|
|
|
|
if(!(p = int_array_new(&years))) goto catch;
|
|
|
|
*p = year;
|
2022-02-11 03:28:28 -05:00
|
|
|
}
|
2022-02-13 01:17:45 -05:00
|
|
|
closedir(dir), dir = 0;
|
2022-02-11 04:46:41 -05:00
|
|
|
qsort(years.data, years.size, sizeof *years.data, &void_int_cmp);
|
2022-02-13 01:17:45 -05:00
|
|
|
fprintf(stderr, "%s: %s.\n", argv[1], int_array_to_string(&years));
|
|
|
|
|
|
|
|
/* Go though each year. */
|
|
|
|
for(y = years.data, y_end = y + years.size; y < y_end; y++) {
|
|
|
|
char temp[64];
|
|
|
|
int *m, *m_end;
|
|
|
|
sprintf(temp, "%d", *y);
|
|
|
|
|
|
|
|
/* Get the months as directories. */
|
|
|
|
if(chdir(temp) == -1 || !(dir = opendir("."))) goto catch;
|
|
|
|
while((de = readdir(dir))) {
|
|
|
|
struct stat st;
|
|
|
|
int month, *p;
|
2022-02-13 23:45:38 -05:00
|
|
|
if(!(month = lex_looks_like_month(de->d_name))) continue;
|
2022-02-13 01:17:45 -05:00
|
|
|
if(stat(de->d_name, &st)) goto catch;
|
|
|
|
if(!S_ISDIR(st.st_mode)) continue;
|
|
|
|
if(!(p = int_array_new(&months))) goto catch;
|
|
|
|
*p = month;
|
|
|
|
}
|
|
|
|
closedir(dir), dir = 0;
|
|
|
|
qsort(months.data, months.size, sizeof *months.data, &void_int_cmp);
|
|
|
|
fprintf(stderr, "%s: %s.\n", temp, int_array_to_string(&months));
|
|
|
|
|
|
|
|
/* Go though each month. */
|
|
|
|
for(m = months.data, m_end = m + months.size; m < m_end; m++) {
|
|
|
|
int *d, *d_end;
|
|
|
|
sprintf(temp, "%.2d", *m);
|
|
|
|
|
|
|
|
/* Get the days as files. */
|
|
|
|
if(chdir(temp) == -1 || !(dir = opendir("."))) goto catch;
|
|
|
|
while((de = readdir(dir))) {
|
|
|
|
struct stat st;
|
|
|
|
int day, *p;
|
|
|
|
/* fixme: Have yyyy-mm-dd to figure out how many days. */
|
2022-02-13 23:45:38 -05:00
|
|
|
if(!(day = lex_looks_like_day(de->d_name))) continue;
|
2022-02-13 01:17:45 -05:00
|
|
|
if(stat(de->d_name, &st)) goto catch;
|
|
|
|
if(S_ISDIR(st.st_mode)) continue;
|
|
|
|
if(!(p = int_array_new(&days))) goto catch;
|
|
|
|
*p = day;
|
|
|
|
}
|
|
|
|
closedir(dir), dir = 0;
|
|
|
|
qsort(days.data, days.size, sizeof *days.data, &void_int_cmp);
|
|
|
|
fprintf(stderr, "%s: %s.\n", temp, int_array_to_string(&days));
|
|
|
|
|
|
|
|
for(d = days.data, d_end = d + days.size; d < d_end; d++) {
|
2022-02-13 23:45:38 -05:00
|
|
|
struct lex lex;
|
2022-02-13 01:17:45 -05:00
|
|
|
printf("%d-%.2d-%.2d\n", *y, *m, *d);
|
2022-02-13 23:45:38 -05:00
|
|
|
sprintf(temp, "%.2d.txt", *d);
|
|
|
|
if(!append_file(&entry, temp)) goto catch;
|
|
|
|
printf("%s", entry.data);
|
|
|
|
printf("Lexing:\n");
|
|
|
|
lex_reset(entry.data);
|
|
|
|
while(lex_next(&lex)) {
|
|
|
|
printf("%lu: %s",
|
|
|
|
(unsigned long)lex.line, lex_symbols[lex.symbol]);
|
2022-02-15 01:02:32 -05:00
|
|
|
if(lex.symbol == TEXT || lex.symbol == ARG_KEYWORD
|
|
|
|
|| lex.symbol == ARG_DATE
|
|
|
|
|| lex.symbol == ARG_FREEFORM) {
|
2022-02-13 23:45:38 -05:00
|
|
|
if(lex.s0 + INT_MAX < lex.s1)
|
|
|
|
{ errno = EILSEQ; goto catch; }
|
2022-02-15 01:02:32 -05:00
|
|
|
printf(" <<%.*s>>", (int)(lex.s1 - lex.s0), lex.s0);
|
2022-02-13 23:45:38 -05:00
|
|
|
}
|
|
|
|
printf(".\n");
|
|
|
|
}
|
2022-02-14 00:07:51 -05:00
|
|
|
printf("Lexing finished: %s on %lu.\n",
|
|
|
|
lex_symbols[lex.symbol], lex.line);
|
2022-02-13 23:45:38 -05:00
|
|
|
char_array_clear(&entry);
|
|
|
|
break; /* fixme */
|
2022-02-13 01:17:45 -05:00
|
|
|
}
|
2022-02-10 23:30:11 -05:00
|
|
|
|
2022-02-13 01:17:45 -05:00
|
|
|
int_array_clear(&days);
|
|
|
|
if(chdir("..") == -1) goto catch;
|
2022-02-13 23:45:38 -05:00
|
|
|
break; /* fixme */
|
2022-02-13 01:17:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
int_array_clear(&months);
|
|
|
|
if(chdir("..") == -1) goto catch;
|
2022-02-13 23:45:38 -05:00
|
|
|
break; /* fixme */
|
2022-02-13 01:17:45 -05:00
|
|
|
}
|
2022-02-10 23:30:11 -05:00
|
|
|
{ success = EXIT_SUCCESS; goto finally; }
|
|
|
|
catch:
|
|
|
|
perror("interpret");
|
|
|
|
finally:
|
2022-02-13 23:45:38 -05:00
|
|
|
if(dir && closedir(dir)) success = EXIT_FAILURE, perror("dir");
|
2022-02-11 04:46:41 -05:00
|
|
|
int_array_(&years);
|
2022-02-13 01:17:45 -05:00
|
|
|
int_array_(&months);
|
2022-02-10 23:30:11 -05:00
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|