mirror of
https://github.com/urbit/shrub.git
synced 2025-01-07 05:26:56 +03:00
202 lines
5.1 KiB
Lua
Executable File
202 lines
5.1 KiB
Lua
Executable File
#!/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)
|