discount/main.c
2025-12-13 00:40:57 +04:00

379 lines
8.1 KiB
C

/*
* markdown: convert a single markdown document into html
*/
/*
* Copyright (C) 2007 Jessica L Parsons.
* The redistribution terms are provided in the COPYRIGHT file that must
* be distributed with this source code.
*/
#include <stdio.h>
#include <stdlib.h>
#include <limits.h>
#include <mkdio.h>
#include <errno.h>
#include <string.h>
#include <stdarg.h>
#include <ctype.h>
#include <unistd.h>
#include <sys/wait.h>
#include "config.h"
#include "amalloc.h"
#include "pgm_options.h"
#include "tags.h"
#include "gethopt.h"
#if HAVE_LIBGEN_H
#include <libgen.h>
#endif
#ifndef HAVE_BASENAME
char*
basename(char *p)
{
char *ret = strrchr(p, '/');
return ret ? (1+ret) : p;
}
#endif
char *pgm = "markdown";
char *
e_flags(const char *text, const int size, void *context)
{
return (char*)context;
}
void
complain(char *fmt, ...)
{
va_list ptr;
fprintf(stderr, "%s: ", pgm);
va_start(ptr, fmt);
vfprintf(stderr, fmt, ptr);
va_end(ptr);
fputc('\n', stderr);
fflush(stderr);
}
void
callback_free(char *input, int size, void *ctx)
{
if (input)
free(input);
}
char *
anchor_format(char *input, void *ctx)
{
int i, j, size;
char* ret;
if ( !input )
return NULL;
size = strlen(input);
ret = malloc(1+size);
if ( !ret )
return NULL;
while ( size && isspace(input[size-1]) )
--size;
for ( j=i=0; i < size; i++ ) {
if (isalnum(input[i]) || strchr("-_+", input[i]) )
ret[j++] = input[i];
else if ( input[i] == ' ' )
ret[j++] = '-';
}
ret[j++] = 0;
return ret;
}
char *external_formatter = 0;
#define RECEIVER 0
#define SENDER 1
char *
external_codefmt(char *src, int len, char *lang)
{
int child_status;
int size, bufsize, curr;
char *res;
pid_t child;
int tochild[2], toparent[2];
if ( pipe(tochild) != 0 || pipe(toparent) != 0 ) {
perror("external_codefmt (pipe)");
res = malloc(len+1);
strcpy(res, src);
return res;
}
if ( (child = fork()) > 0 ) {
close(tochild[RECEIVER]);
close(toparent[SENDER]);
bufsize = 1000;
res = malloc(1+bufsize);
curr = 0;
/* parent */
write(tochild[SENDER], src, len);
close(tochild[SENDER]);
while ( (size = read(toparent[RECEIVER], res+curr, 1000)) > 0 ) {
curr += size;
res = realloc(res, bufsize += 1000);
}
res[curr] = 0;
waitpid(child, &child_status, 0);
close(toparent[RECEIVER]);
if ( WIFEXITED(child_status) )
return res;
else
free(res); /* something failed; just return the original string */
}
else if ( child == 0 ) {
close(tochild[SENDER]);
close(toparent[RECEIVER]);
close(1); dup2(toparent[SENDER], 1);
close(0); dup2(tochild[RECEIVER], 0);
system(external_formatter);
close(0); close(tochild[RECEIVER]);
close(1); close(toparent[SENDER]);
exit(0);
}
res = malloc(len+1);
strcpy(res, src);
return res;
}
struct h_opt opts[] = {
{ 0, "html5", '5', 0, "recognise html5 block elements" },
{ 0, "base", 'b', "url-base", "URL prefix" },
{ 0, "debug", 'd', 0, "debugging" },
{ 0, "version",'V', 0, "show version info" },
{ 0, 0, 'E', "flags", "url flags" },
{ 0, 0, 'F', "bitmap", "set/show hex flags" },
{ 0, 0, 'f', "{+-}flags", "set/show named flags" },
{ 0, 0, 'G', 0, "github flavoured markdown" },
{ 0, 0, 'n', 0, "don't write generated html" },
{ 0, 0, 's', "text", "format `text`" },
{ 0, "style", 'S', 0, "output <style> blocks" },
{ 0, 0, 't', "text", "format `text` with mkd_line()" },
{ 0, "toc", 'T', 0, "output a TOC" },
{ 0, 0, 'C', "prefix", "prefix for markdown extra footnotes" },
{ 0, 0, 'o', "file", "write output to file" },
{ 0, "squash", 'x', 0, "squash toc labels to be more like github" },
{ 0, "codefmt",'X', "command", "use an external code formatter" },
{ 0, "help", '?', 0, "print a detailed usage message" },
};
#define NROPTS (sizeof opts/sizeof opts[0])
int
main(int argc, char **argv)
{
int rc;
int debug = 0;
int toc = 0;
int content = 1;
int version = 0;
int styles = 0;
int use_mkd_line = 0;
int use_e_codefmt = 0;
int github_flavoured = 0;
int squash = 0;
char *extra_footnote_prefix = 0;
char *urlflags = 0;
char *text = 0;
char *ofile = 0;
char *urlbase = 0;
char *q;
MMIOT *doc;
struct h_context blob;
struct h_opt *opt;
mkd_flag_t *flags = mkd_flags();
if ( !flags )
perror("new_flags");
hoptset(&blob, argc, argv);
hopterr(&blob, 1);
pgm = basename(argv[0]);
if ( q = getenv("MARKDOWN_FLAGS") )
mkd_set_flag_bitmap(flags, strtol(q,0,0));
while ( opt=gethopt(&blob, opts, NROPTS) ) {
if ( opt == HOPTERR ) {
hoptusage(pgm, opts, NROPTS, "[file]");
exit(1);
}
switch (opt->optchar) {
case '5': mkd_set_flag_num(flags, MKD_HTML5);
break;
case 'b': urlbase = hoptarg(&blob);
break;
case 'd': debug++;
break;
case 'V': version++;
break;
case 'E': urlflags = hoptarg(&blob);
break;
case 'f': q = hoptarg(&blob);
if ( strcmp(q, "?") == 0 ) {
show_flags(1, version, 0);
exit(0);
}
else if ( strcmp(q, "??") == 0 ) {
show_flags(1, version, flags);
exit(0);
}
else if ( q=mkd_set_flag_string(flags, hoptarg(&blob)) )
complain("unknown option <%s>", q);
break;
case 'F': q = hoptarg(&blob);
if ( strcmp(q, "?") == 0 ) {
show_flags(0, 0, 0);
exit(0);
}
else if ( strcmp(q, "??") == 0 ) {
show_flags(0, version, flags);
exit(0);
}
else
mkd_set_flag_bitmap(flags,strtol(q, 0, 0));
break;
case 'G': github_flavoured = 1;
break;
case 'n': content = 0;
break;
case 's': text = hoptarg(&blob);
break;
case 'S': styles = 1;
break;
case 't': text = hoptarg(&blob);
use_mkd_line = 1;
break;
case 'T': mkd_set_flag_num(flags, MKD_TOC);
toc = 1;
break;
case 'C': extra_footnote_prefix = hoptarg(&blob);
break;
case 'o': if ( ofile ) {
complain("Too many -o options");
exit(1);
}
if ( !freopen(ofile = hoptarg(&blob), "w", stdout) ) {
perror(ofile);
exit(1);
}
break;
case 'x': squash = 1;
break;
case 'X': use_e_codefmt = 1;
mkd_set_flag_num(flags, MKD_FENCEDCODE);
external_formatter = hoptarg(&blob);
fprintf(stderr, "selected external formatter (%s)\n", external_formatter);
break;
case '?': hoptdescribe(pgm, opts, NROPTS, "[file]", 1);
return 0;
}
}
if ( version ) {
printf("%s: discount %s", pgm, markdown_version);
if ( version == 2 )
mkd_flags_are(stdout, flags, 0);
putchar('\n');
exit(0);
}
argc -= hoptind(&blob);
argv += hoptind(&blob);
if ( use_mkd_line )
rc = mkd_generateline( text, strlen(text), stdout, flags);
else {
if ( text ) {
doc = github_flavoured ? gfm_string(text, strlen(text), flags)
: mkd_string(text, strlen(text), flags) ;
if ( !doc ) {
perror(text);
exit(1);
}
}
else {
if ( argc && !freopen(argv[0], "r", stdin) ) {
perror(argv[0]);
exit(1);
}
doc = github_flavoured ? gfm_in(stdin,flags)
: mkd_in(stdin,flags);
if ( !doc ) {
perror(argc ? argv[0] : "stdin");
exit(1);
}
}
if ( urlbase )
mkd_basename(doc, urlbase);
if ( urlflags )
mkd_e_flags(doc, e_flags, NULL, urlflags);
if ( squash )
mkd_e_anchor(doc, (mkd_callback_t) anchor_format, callback_free, 0);
if ( use_e_codefmt )
mkd_e_code_format(doc, (mkd_callback_t)external_codefmt, callback_free, 0);
if ( extra_footnote_prefix )
mkd_ref_prefix(doc, extra_footnote_prefix);
if ( debug ) {
rc = mkd_dump(doc, stdout, flags, argc ? basename(argv[0]) : "stdin");
}
else {
rc = 1;
if ( mkd_compile(doc, flags) ) {
rc = 0;
if ( styles )
mkd_generatecss(doc, stdout);
if ( toc )
mkd_generatetoc(doc, stdout);
if ( content )
mkd_generatehtml(doc, stdout);
}
}
mkd_cleanup(doc);
}
mkd_free_flags(flags);
adump();
exit( (rc == 0) ? 0 : errno );
}