shrub/outside/commonmark/wrapper.lua

202 lines
5.1 KiB
Lua
Raw Normal View History

#!/usr/bin/env luajit
local ffi = require("ffi")
cmark = ffi.load("libcmark")
ffi.cdef[[
char *cmark_markdown_to_html(const char *text, int len);
typedef enum {
/* Block */
CMARK_NODE_DOCUMENT,
CMARK_NODE_BLOCK_QUOTE,
CMARK_NODE_LIST,
CMARK_NODE_LIST_ITEM,
CMARK_NODE_CODE_BLOCK,
CMARK_NODE_HTML,
CMARK_NODE_PARAGRAPH,
CMARK_NODE_HEADER,
CMARK_NODE_HRULE,
CMARK_NODE_REFERENCE_DEF,
CMARK_NODE_FIRST_BLOCK = CMARK_NODE_DOCUMENT,
CMARK_NODE_LAST_BLOCK = CMARK_NODE_REFERENCE_DEF,
/* Inline */
CMARK_NODE_TEXT,
CMARK_NODE_SOFTBREAK,
CMARK_NODE_LINEBREAK,
CMARK_NODE_INLINE_CODE,
CMARK_NODE_INLINE_HTML,
CMARK_NODE_EMPH,
CMARK_NODE_STRONG,
CMARK_NODE_LINK,
CMARK_NODE_IMAGE,
CMARK_NODE_FIRST_INLINE = CMARK_NODE_TEXT,
CMARK_NODE_LAST_INLINE = CMARK_NODE_IMAGE,
} cmark_node_type;
typedef enum {
CMARK_NO_LIST,
CMARK_BULLET_LIST,
CMARK_ORDERED_LIST
} cmark_list_type;
typedef enum {
CMARK_PERIOD_DELIM,
CMARK_PAREN_DELIM
} cmark_delim_type;
typedef struct cmark_node cmark_node;
typedef struct cmark_parser cmark_parser;
cmark_node* cmark_node_new(cmark_node_type type);
void
cmark_node_free(cmark_node *node);
cmark_node* cmark_node_next(cmark_node *node);
cmark_node* cmark_node_previous(cmark_node *node);
cmark_node* cmark_node_parent(cmark_node *node);
cmark_node* cmark_node_first_child(cmark_node *node);
cmark_node* cmark_node_last_child(cmark_node *node);
cmark_node_type cmark_node_get_type(cmark_node *node);
const char* cmark_node_get_string_content(cmark_node *node);
int cmark_node_set_string_content(cmark_node *node, const char *content);
int cmark_node_get_header_level(cmark_node *node);
int cmark_node_set_header_level(cmark_node *node, int level);
cmark_list_type cmark_node_get_list_type(cmark_node *node);
int cmark_node_set_list_type(cmark_node *node, cmark_list_type type);
int cmark_node_get_list_start(cmark_node *node);
int cmark_node_set_list_start(cmark_node *node, int start);
int cmark_node_get_list_tight(cmark_node *node);
int cmark_node_set_list_tight(cmark_node *node, int tight);
const char* cmark_node_get_fence_info(cmark_node *node);
int cmark_node_set_fence_info(cmark_node *node, const char *info);
const char* cmark_node_get_url(cmark_node *node);
int cmark_node_set_url(cmark_node *node, const char *url);
const char* cmark_node_get_title(cmark_node *node);
int cmark_node_set_title(cmark_node *node, const char *title);
int cmark_node_get_start_line(cmark_node *node);
int cmark_node_get_start_column(cmark_node *node);
int cmark_node_get_end_line(cmark_node *node);
void cmark_node_unlink(cmark_node *node);
int cmark_node_insert_before(cmark_node *node, cmark_node *sibling);
int cmark_node_insert_after(cmark_node *node, cmark_node *sibling);
int cmark_node_prepend_child(cmark_node *node, cmark_node *child);
int cmark_node_append_child(cmark_node *node, cmark_node *child);
cmark_parser *cmark_parser_new();
void cmark_parser_free(cmark_parser *parser);
cmark_node *cmark_parser_finish(cmark_parser *parser);
void cmark_parser_feed(cmark_parser *parser, const char *buffer, size_t len);
cmark_node *cmark_parse_document(const char *buffer, size_t len);
char *cmark_render_ast(cmark_node *root);
char *cmark_render_html(cmark_node *root);
]]
local inp = io.read("*all")
local doc = cmark.cmark_parse_document(inp, string.len(inp))
local cur = doc
local next
local child
local walk = function(action)
level = 0
while cur ~= nil do
action(cur, level)
child = cmark.cmark_node_first_child(cur)
if child == nil then
next = cmark.cmark_node_next(cur)
while next == nil do
cur = cmark.cmark_node_parent(cur)
level = level - 1
if cur == nil then
break
else
next = cmark.cmark_node_next(cur)
end
end
cur = next
else
level = level + 1
cur = child
end
end
end
local type_table = {
'BLOCK_QUOTE',
'LIST',
'LIST_ITEM',
'CODE_BLOCK',
'HTML',
'PARAGRAPH',
'HEADER',
'HRULE',
'REFERENCE_DEF',
'TEXT',
'SOFTBREAK',
'LINEBREAK',
'INLINE_CODE',
'INLINE_HTML',
'EMPH',
'STRONG',
'LINK',
'IMAGE',
}
type_table[0] = 'DOCUMENT'
local function print_type(node, level)
local t = tonumber(cmark.cmark_node_get_type(node))
io.write(string.rep(' ', level) .. type_table[t])
if t == cmark.CMARK_NODE_TEXT then
io.write(' ' .. ffi.string(cmark.cmark_node_get_string_content(node)))
end
io.write('\n')
end
walk(print_type)
-- local html = ffi.string(cmark.cmark_render_html(doc))
-- print(html)