2015-12-25 23:42:45 +08:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#define TOC_ENTRIES 512
|
|
|
|
#define TOC_ENTRY_SIZE 1024
|
|
|
|
|
2015-12-26 00:48:37 +08:00
|
|
|
|
2015-12-25 23:42:45 +08:00
|
|
|
char toc_start[] = "[tocstart]: # (toc start)";
|
|
|
|
char toc_end[] = "[tocend]: # (toc end)";
|
|
|
|
|
|
|
|
char toc[TOC_ENTRIES][TOC_ENTRY_SIZE];
|
|
|
|
int toc_level[TOC_ENTRIES];
|
|
|
|
int toc_cnt = 0;
|
|
|
|
|
2015-12-26 00:48:37 +08:00
|
|
|
char *getentry(const char *s)
|
|
|
|
{
|
|
|
|
static char buf[TOC_ENTRY_SIZE];
|
|
|
|
char *t = buf;
|
|
|
|
for(;;)
|
|
|
|
{
|
|
|
|
if ( *s == '\0' )
|
|
|
|
break;
|
|
|
|
if ( *s >= ' ' )
|
|
|
|
{
|
|
|
|
*t++ = *s++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*t = '\0';
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *getref(const char *s)
|
|
|
|
{
|
|
|
|
static char buf[TOC_ENTRY_SIZE];
|
|
|
|
char *t = buf;
|
|
|
|
for(;;)
|
|
|
|
{
|
|
|
|
if ( *s == '\0' )
|
|
|
|
break;
|
|
|
|
if ( *s == ' ' )
|
|
|
|
{
|
|
|
|
*t++ = '-';
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
else if ( *s >= '0' && *s <= '9' )
|
|
|
|
{
|
|
|
|
*t++ = *s++;
|
|
|
|
}
|
|
|
|
else if ( *s < 'A' )
|
|
|
|
{
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*t = tolower(*s);
|
|
|
|
t++;
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*t = '\0';
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2015-12-25 23:42:45 +08:00
|
|
|
void add_toc(int level, char *entry)
|
|
|
|
{
|
|
|
|
strncpy( toc[toc_cnt], entry, TOC_ENTRY_SIZE);
|
|
|
|
toc[toc_cnt][TOC_ENTRY_SIZE-1] = '\0';
|
|
|
|
toc_level[toc_cnt] = level;
|
|
|
|
toc_cnt++;
|
2015-12-26 00:48:37 +08:00
|
|
|
// printf("%d: %s [%s](#%s)\n", level, entry, getentry(entry), getref(entry));
|
2015-12-25 23:42:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void read_md_fp(FILE *in, FILE *out)
|
|
|
|
{
|
|
|
|
static char buf[TOC_ENTRY_SIZE];
|
|
|
|
char *s;
|
|
|
|
int is_inside_toc = 0;
|
|
|
|
int is_start_of_file = 1;
|
|
|
|
for(;;)
|
|
|
|
{
|
|
|
|
s = fgets(buf, TOC_ENTRY_SIZE, in);
|
|
|
|
if ( s == NULL )
|
|
|
|
return;
|
|
|
|
if ( is_inside_toc != 0 )
|
|
|
|
{
|
2015-12-26 00:48:37 +08:00
|
|
|
if ( strncmp(s, toc_end, strlen(toc_end)) == 0 )
|
2015-12-25 23:42:45 +08:00
|
|
|
{
|
|
|
|
is_inside_toc = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-12-26 00:48:37 +08:00
|
|
|
if ( buf[0] == '\0' || buf[0] == '\n' )
|
2015-12-25 23:42:45 +08:00
|
|
|
{
|
|
|
|
/* skip empty lines at the beginning of the file */
|
|
|
|
if ( is_start_of_file == 0 )
|
|
|
|
fputs(buf, out);
|
|
|
|
}
|
2015-12-26 00:48:37 +08:00
|
|
|
else if ( strncmp(s, toc_start, strlen(toc_start)) == 0 )
|
2015-12-25 23:42:45 +08:00
|
|
|
{
|
|
|
|
is_inside_toc = 1;
|
|
|
|
}
|
|
|
|
else if ( strncmp(s, "# ", 2) == 0 )
|
|
|
|
{
|
|
|
|
is_start_of_file = 0;
|
|
|
|
add_toc(1, s+2);
|
|
|
|
fputs(buf, out);
|
|
|
|
}
|
|
|
|
else if ( strncmp(s, "## ", 3) == 0 )
|
|
|
|
{
|
|
|
|
is_start_of_file = 0;
|
|
|
|
add_toc(2, s+3);
|
|
|
|
fputs(buf, out);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
is_start_of_file = 0;
|
|
|
|
fputs(buf, out);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void read_md(const char *in_name, const char *out_name)
|
|
|
|
{
|
|
|
|
FILE *in;
|
|
|
|
FILE *out;
|
|
|
|
|
|
|
|
in = fopen(in_name, "r");
|
|
|
|
if ( in == NULL )
|
|
|
|
{
|
|
|
|
perror(in_name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
out = fopen(out_name, "w");
|
|
|
|
if ( out == NULL )
|
|
|
|
{
|
|
|
|
perror(out_name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
read_md_fp(in, out);
|
|
|
|
|
|
|
|
fclose(in);
|
|
|
|
fclose(out);
|
|
|
|
}
|
|
|
|
|
2015-12-26 00:48:37 +08:00
|
|
|
void copy_to_md_fp(FILE *tmp, FILE *out)
|
|
|
|
{
|
|
|
|
static char buf[TOC_ENTRY_SIZE];
|
|
|
|
char *s;
|
|
|
|
int i, j;
|
|
|
|
fprintf(out, "\n");
|
|
|
|
fprintf(out, "%s\n", toc_start);
|
|
|
|
fprintf(out, "\n");
|
|
|
|
for( i = 0; i < toc_cnt; i++ )
|
|
|
|
{
|
|
|
|
for( j = 0; j < toc_level[i]; j++)
|
|
|
|
fprintf(out, " ");
|
|
|
|
fprintf(out, "* [%s](#%s)\n", getentry(toc[i]), getref(toc[i]));
|
|
|
|
}
|
|
|
|
fprintf(out, "\n");
|
|
|
|
fprintf(out, "%s\n", toc_end);
|
|
|
|
fprintf(out, "\n");
|
|
|
|
|
|
|
|
for(;;)
|
|
|
|
{
|
|
|
|
s = fgets(buf, TOC_ENTRY_SIZE, tmp);
|
|
|
|
if ( s == NULL )
|
|
|
|
return;
|
|
|
|
fputs(buf, out);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void copy_to_md(const char *tmp_name, const char *out_name)
|
|
|
|
{
|
|
|
|
FILE *tmp;
|
|
|
|
FILE *out;
|
|
|
|
|
|
|
|
tmp = fopen(tmp_name, "r");
|
|
|
|
if ( tmp == NULL )
|
|
|
|
{
|
|
|
|
perror(tmp_name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
out = fopen(out_name, "w");
|
|
|
|
if ( out == NULL )
|
|
|
|
{
|
|
|
|
perror(out_name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
copy_to_md_fp(tmp, out);
|
|
|
|
|
|
|
|
fclose(tmp);
|
|
|
|
fclose(out);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-12-25 23:42:45 +08:00
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
if ( argc <= 1 )
|
|
|
|
{
|
|
|
|
printf("%s [markdown files]\n", argv[0]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
argv++;
|
|
|
|
while( *argv != NULL )
|
|
|
|
{
|
|
|
|
toc_cnt = 0;
|
|
|
|
read_md(*argv, "tmp.md");
|
2015-12-26 00:48:37 +08:00
|
|
|
copy_to_md("tmp.md", *argv);
|
|
|
|
printf("%s: added %d toc entries\n", *argv, toc_cnt);
|
2015-12-25 23:42:45 +08:00
|
|
|
argv++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|