aboutsummaryrefslogtreecommitdiff
path: root/tools/make_spec.lua
diff options
context:
space:
mode:
authorJohn MacFarlane <jgm@berkeley.edu>2016-01-08 12:15:31 -0800
committerJohn MacFarlane <jgm@berkeley.edu>2016-01-10 21:32:21 -0800
commit2413d436ed98866e9a70fbd4070a5da52dd97cd5 (patch)
tree0cacf842df8115e93e3de1ce9a1613cee13474be /tools/make_spec.lua
parent1357f2859ecb128636ea7a764b70407dca4e4015 (diff)
New format for spec tests, new lua formatter for specs.
The format for the spec examples has changed from . markdown . html . to ```````````````````````````````` example markdown . html ```````````````````````````````` One advantage of this is that `spec.txt` becomes a valid Markdown file. `tests/spec_test.py` has been changed to use the new format. The old `tools/makespec.py` has been replaced by a lua program, `tools/make_spec.lua`, which uses the `lcmark` rock (and indirectly libcmark). It can generate html, latex, and commonmark versions of the spec. Pandoc is no longer needed for the latex/PDF version. And, since the new program uses the cmark API and operates directly on the parse tree, we avoid certain bad results we got with the regex replacements done by the python script.
Diffstat (limited to 'tools/make_spec.lua')
-rw-r--r--tools/make_spec.lua292
1 files changed, 292 insertions, 0 deletions
diff --git a/tools/make_spec.lua b/tools/make_spec.lua
new file mode 100644
index 0000000..3425f35
--- /dev/null
+++ b/tools/make_spec.lua
@@ -0,0 +1,292 @@
+local lcmark = require('lcmark')
+local cmark = require('cmark')
+
+local format = arg[1] or 'html'
+
+local trim = function(s)
+ return s:gsub("^%s+",""):gsub("%s+$","")
+end
+
+local warn = function(s)
+ io.stderr:write('WARNING: ' .. s .. '\n')
+end
+
+local to_identifier = function(s)
+ return trim(s):lower():gsub('[^%w]+', ' '):gsub('[%s]+', '-')
+end
+
+local render_number = function(tbl)
+ local buf = {}
+ for i,x in ipairs(tbl) do
+ buf[i] = tostring(x)
+ end
+ return table.concat(buf, '.')
+end
+
+local extract_references = function(doc)
+ local cur, entering, node_type
+ local refs = {}
+ for cur, entering, node_type in cmark.walk(doc) do
+ if not entering and
+ ((node_type == cmark.NODE_LINK and cmark.node_get_url(cur) == '@') or
+ node_type == cmark.NODE_HEADING) then
+ local children = cmark.node_first_child(cur)
+ local label = trim(cmark.render_commonmark(children, OPT_DEFAULT, 0))
+ local ident = to_identifier(label)
+ if refs[label] then
+ warn("duplicate reference " .. label)
+ end
+ refs[label] = ident
+ if not refs[label .. 's'] then
+ -- plural too
+ refs[label .. 's'] = ident
+ end
+ end
+ end
+ -- check for duplicate IDs
+ local idents = {}
+ for _,id in ipairs(refs) do
+ if idents[id] then
+ warn("duplicate identifier " .. id)
+ end
+ idents[#idents + 1] = id
+ end
+ return refs
+end
+
+local make_toc = function(toc)
+ -- we create a commonmark string, then parse it
+ local toclines = {}
+ for _,entry in ipairs(toc) do
+ if entry.level <= 2 then
+ local indent = string.rep(' ', entry.level - 1)
+ toclines[#toclines + 1] = indent .. '* [' ..
+ (entry.number == '' and ''
+ or '<span class="number">' .. entry.number .. '</span>') ..
+ entry.label .. '](#' .. entry.ident .. ')'
+ end
+ end
+ -- now parse our cm list and return the resulting list node:
+ local doc = cmark.parse_string(table.concat(toclines, '\n'), cmark.OPT_SMART)
+ return cmark.node_first_child(doc)
+end
+
+local make_html_element = function(block, tagname, attrs)
+ local div = cmark.node_new(block and cmark.NODE_CUSTOM_BLOCK or
+ cmark.NODE_CUSTOM_INLINE)
+ local attribs = {}
+ for _,attr in ipairs(attrs) do
+ attribs[#attribs + 1] = ' ' .. attr[1] .. '="' .. attr[2] .. '"'
+ end
+ local opentag = '<' .. tagname .. table.concat(attribs, '') .. '>'
+ local closetag = '</' .. tagname .. '>'
+ cmark.node_set_on_enter(div, opentag)
+ cmark.node_set_on_exit(div, closetag)
+ return div
+end
+
+local make_html_block = function(tagname, attrs)
+ return make_html_element(true, tagname, attrs)
+end
+
+local make_html_inline = function(tagname, attrs)
+ return make_html_element(false, tagname, attrs)
+end
+
+local make_latex = function(spec)
+ local latex = cmark.node_new(spec.block and cmark.NODE_CUSTOM_BLOCK or
+ cmark.NODE_CUSTOM_INLINE)
+ cmark.node_set_on_enter(latex, spec.start)
+ cmark.node_set_on_exit(latex, spec.stop)
+ return latex
+end
+
+local make_text = function(s)
+ local text = cmark.node_new(cmark.NODE_TEXT)
+ cmark.node_set_literal(text, s)
+ return text
+end
+
+local create_anchors = function(doc, meta, to)
+ local cur, entering, node_type
+ local toc = {}
+ local number = {0}
+ local example = 0
+ for cur, entering, node_type in cmark.walk(doc) do
+ if not entering and
+ ((node_type == cmark.NODE_LINK and cmark.node_get_url(cur) == '@') or
+ node_type == cmark.NODE_HEADING) then
+
+ local anchor
+ local children = cmark.node_first_child(cur)
+ local label = trim(cmark.render_commonmark(children, OPT_DEFAULT, 0))
+ local ident = to_identifier(label)
+ if node_type == cmark.NODE_LINK then
+ if format == 'latex' then
+ anchor = make_latex({start="\\hypertarget{" .. ident .. "}{",
+ stop="\\label{" .. ident .. "}}",
+ block = true})
+ else
+ anchor = make_html_inline('a', {{'id', ident}, {'href', '#'..ident},
+ {'class', 'definition'}})
+ end
+
+ else -- NODE_HEADING
+
+ local level = cmark.node_get_heading_level(cur)
+ local last_level = #toc == 0 and 1 or toc[#toc].level
+ if #number > 0 then
+ if level > last_level then -- subhead
+ number[level] = 1
+ else
+ while last_level > level do
+ number[last_level] = nil
+ last_level = last_level - 1
+ end
+ number[level] = number[level] + 1
+ end
+ end
+ table.insert(toc, { label = label, ident = ident, level = level, number = render_number(number) })
+ local num = render_number(number)
+ local section_cmds = {"\\section", "\\subsection",
+ "\\subsubsection", "\\chapter"}
+ if format == 'latex' then
+ anchor = make_latex({start="\\hypertarget{" .. ident .. "}{" ..
+ section_cmds[level] .. "{",
+ stop="}\\label{" .. ident .. "}}",
+ block = true})
+ else
+ anchor = make_html_block('h' .. tostring(level),
+ {{'id', ident},
+ {'href', '#'..ident},
+ {'class', 'definition'}})
+ if num ~= '' then
+ local numspan = make_html_inline('span', {{'class','number'}})
+ node_append_child(numspan, make_text(num))
+ node_append_child(anchor, numspan)
+ end
+ end
+ end
+ while children do
+ node_append_child(anchor, children)
+ children = cmark.node_next(children)
+ end
+ cmark.node_insert_before(cur, anchor)
+ cmark.node_unlink(cur)
+ elseif entering and node_type == cmark.NODE_CODE_BLOCK and
+ cmark.node_get_fence_info(cur) == 'example' then
+ example = example + 1
+ -- split into two code blocks
+ local code = cmark.node_get_literal(cur)
+ local sepstart, sepend = code:find("[\n\r]+%.[\n\r]+")
+ if not sepstart then
+ warn("Could not find separator in:\n" .. contents)
+ end
+ local markdown_code = cmark.node_new(cmark.NODE_CODE_BLOCK)
+ local html_code = cmark.node_new(cmark.NODE_CODE_BLOCK)
+ -- note: we replace the ␣ with a special span after rendering
+ local markdown_code_string = code:sub(1, sepstart):gsub(' ', '␣')
+ local html_code_string = code:sub(sepend + 1):gsub(' ', '␣')
+ cmark.node_set_literal(markdown_code, markdown_code_string)
+ cmark.node_set_fence_info(markdown_code, 'markdown')
+ cmark.node_set_literal(html_code, html_code_string)
+ cmark.node_set_fence_info(html_code, 'html')
+
+ local example_div, leftcol_div, rightcol_div
+ if format == 'latex' then
+ example_div = make_latex({start = '\\begin{minipage}[t]{\\textwidth}\n{\\scriptsize Example ' .. tostring(example) .. '}\n\n\\vspace{-0.4em}\n', stop = '\\end{minipage}', block = true})
+ leftcol_div = make_latex({start = "\\begin{minipage}[t]{0.49\\textwidth}\n\\definecolor{shadecolor}{gray}{0.85}\n\\begin{snugshade}\\small\n", stop = "\\end{snugshade}\n\\end{minipage}\n\\hfill", block = true})
+ rightcol_div = make_latex({start = "\\begin{minipage}[t]{0.49\\textwidth}\n\\definecolor{shadecolor}{gray}{0.95}\n\\begin{snugshade}\\small\n", stop = "\\end{snugshade}\n\\end{minipage}\n\\vspace{0.8em}", block = true})
+ cmark.node_append_child(leftcol_div, markdown_code)
+ cmark.node_append_child(rightcol_div, html_code)
+ cmark.node_append_child(example_div, leftcol_div)
+ cmark.node_append_child(example_div, rightcol_div)
+ else
+ leftcol_div = make_html_block('div', {{'class','column'}})
+ rightcol_div = make_html_block('div', {{'class', 'column'}})
+ cmark.node_append_child(leftcol_div, markdown_code)
+ cmark.node_append_child(rightcol_div, html_code)
+ local examplenum_div = make_html_block('div', {{'class', 'examplenum'}})
+ local interact_link = make_html_inline('a', {{'class', 'dingus'},
+ {'title', 'open in interactive dingus'}})
+ cmark.node_append_child(interact_link, make_text("(interact)"))
+ local examplenum_link = cmark.node_new(cmark.NODE_LINK)
+ cmark.node_set_url(examplenum_link, '#example-' .. tostring(example))
+ cmark.node_append_child(examplenum_link,
+ make_text("Example " .. tostring(example)))
+ cmark.node_append_child(examplenum_div, examplenum_link)
+ if format == 'html' then
+ cmark.node_append_child(examplenum_div, interact_link)
+ end
+ example_div = make_html_block('div', {{'class', 'example'},
+ {'id','example-' .. tostring(example)}})
+ cmark.node_append_child(example_div, examplenum_div)
+ cmark.node_append_child(example_div, leftcol_div)
+ cmark.node_append_child(example_div, rightcol_div)
+ end
+ cmark.node_insert_before(cur, example_div)
+ cmark.node_unlink(cur)
+ cmark.node_free(cur)
+ elseif node_type == cmark.NODE_HTML_BLOCK and
+ cmark.node_get_literal(cur) == '<!-- END TESTS -->\n' then
+ -- change numbering
+ number = {}
+ if format ~= 'latex' then
+ local appendices = make_html_block('div', {{'class','appendices'}})
+ cmark.node_insert_after(cur, appendices)
+ -- put the remaining sections in an appendix
+ local tmp = cmark.node_next(appendices)
+ while tmp do
+ cmark.node_append_child(appendices, tmp)
+ tmp = cmark.node_next(tmp)
+ end
+ end
+ end
+ end
+ meta.toc = make_toc(toc)
+end
+
+local to_ref = function(ref)
+ return '[' .. ref.label .. ']: #' .. ref.indent .. '\n'
+end
+
+local inp = io.read("*a")
+local doc1 = cmark.parse_string(inp, cmark.OPT_DEFAULT)
+local refs = extract_references(doc1)
+local refblock = '\n'
+for lab,ident in pairs(refs) do
+ refblock = refblock .. '[' .. lab .. ']: #' .. ident .. '\n'
+ -- refblock = refblock .. '[' .. lab .. 's]: #' .. ident .. '\n'
+end
+-- append references and parse again
+local contents, meta, msg = lcmark.convert(inp .. refblock, format,
+ { smart = true,
+ yaml_metadata = true,
+ safe = false,
+ filters = { create_anchors }
+ })
+
+if contents then
+ local f = io.open("tools/template." .. format, 'r')
+ if not f then
+ io.stderr:write("Could not find template!")
+ os.exit(1)
+ end
+ local template = f:read("*a")
+
+ if format == 'html' then
+ contents = contents:gsub('␣', '<span class="space"> </span>')
+ end
+ meta.body = contents
+ local rendered, msg = lcmark.render_template(template, meta)
+ if not rendered then
+ io.stderr:write(msg)
+ os.exit(1)
+ end
+ io.write(rendered)
+ os.exit(0)
+else
+ io.stderr:write(msg)
+ os.exit(1)
+end
+