580 lines
17 KiB
Lua
580 lines
17 KiB
Lua
---------------
|
|
-- ldoc, a Lua documentation generator.
|
|
-- Compatible with luadoc-style annoations, but providing
|
|
-- easier customization options. C/C++ support is provided.
|
|
-- Steve Donovan, 2011
|
|
|
|
require 'pl'
|
|
|
|
local append = table.insert
|
|
local template = require 'pl.template'
|
|
local lapp = require 'pl.lapp'
|
|
|
|
-- so we can find our private modules
|
|
app.require_here()
|
|
|
|
local args = lapp [[
|
|
ldoc, a Lua documentation generator, vs 0.1 Beta
|
|
-d,--dir (default docs) output directory
|
|
-o (default 'index') output name
|
|
-v,--verbose verbose
|
|
-q,--quiet suppress output
|
|
-m,--module module docs as text
|
|
-s,--style (default !) directory for templates and style
|
|
-p,--project (default ldoc) project name
|
|
-t,--title (default Reference) page title
|
|
-f,--format (default plain) formatting - can be markdown
|
|
-b,--package (default .) top-level package basename (needed for module(...))
|
|
--dump debug output dump
|
|
<file> (string) source file or directory containing source
|
|
]]
|
|
|
|
local lexer = require 'lexer'
|
|
local doc = require 'doc'
|
|
local Item,File,Module = doc.Item,doc.File,doc.Module
|
|
local tools = require 'tools'
|
|
local KindMap = tools.KindMap
|
|
|
|
class.ModuleMap(KindMap)
|
|
|
|
function ModuleMap:_init ()
|
|
self.klass = ModuleMap
|
|
end
|
|
|
|
ModuleMap:add_kind('function','Functions','Parameters')
|
|
ModuleMap:add_kind('table','Tables','Fields')
|
|
ModuleMap:add_kind('field','Fields')
|
|
|
|
class.ProjectMap(KindMap)
|
|
ProjectMap.project_level = true
|
|
|
|
function ProjectMap:_init ()
|
|
self.klass = ProjectMap
|
|
end
|
|
|
|
ProjectMap:add_kind('module','Modules')
|
|
ProjectMap:add_kind('script','Scripts')
|
|
|
|
------- ldoc external API ------------
|
|
|
|
-- the ldoc table represents the API available in `config.ld`.
|
|
local ldoc = {}
|
|
|
|
-- aliases to existing tags can be defined. E.g. just 'p' for 'param'
|
|
function ldoc.alias (a,tag)
|
|
doc.add_alias(a,tag)
|
|
end
|
|
|
|
-- new tags can be added, which can be on a project level.
|
|
function ldoc.new_type (tag,header,project_level)
|
|
doc.add_tag(tag,doc.TAG_TYPE,project_level)
|
|
if project_level then
|
|
ProjectMap:add_kind(tag,header)
|
|
else
|
|
ModuleMap:add_kind(tag,header)
|
|
end
|
|
end
|
|
|
|
-- any file called 'config.ld' found in the source tree will be
|
|
-- handled specially. It will be loaded using 'ldoc' as the environment.
|
|
local function read_ldoc_config (fname)
|
|
local directory = path.dirname(fname)
|
|
print('reading configuration from '..fname)
|
|
local txt = utils.readfile(fname)
|
|
-- Penlight defines loadin for Lua 5.1 as well
|
|
local chunk,err = loadin(ldoc,txt)
|
|
if chunk then
|
|
local ok
|
|
ok,err = pcall(chunk)
|
|
end
|
|
if err then print('error loading config file '..fname..': '..err) end
|
|
return directory
|
|
end
|
|
|
|
------ Parsing the Source --------------
|
|
-- This uses the lexer from PL, but it should be possible to use Peter Odding's
|
|
-- excellent Lpeg based lexer instead.
|
|
|
|
local tnext = lexer.skipws
|
|
|
|
-- a pattern particular to LuaDoc tag lines: the line must begin with @TAG,
|
|
-- followed by the value, which may extend over several lines.
|
|
local luadoc_tag = '^%s*@(%a+)%s(.+)'
|
|
|
|
-- assumes that the doc comment consists of distinct tag lines
|
|
function parse_tags(text)
|
|
local lines = stringio.lines(text)
|
|
local preamble, line = tools.grab_while_not(lines,luadoc_tag)
|
|
local tag_items = {}
|
|
local follows
|
|
while line do
|
|
local tag,rest = line:match(luadoc_tag)
|
|
follows, line = tools.grab_while_not(lines,luadoc_tag)
|
|
append(tag_items,{tag, rest .. '\n' .. follows})
|
|
end
|
|
return preamble,tag_items
|
|
end
|
|
|
|
-- This takes the collected comment block, and uses the docstyle to
|
|
-- extract tags and values. Assume that the summary ends in a period or a question
|
|
-- mark, and everything else in the preamble is the description.
|
|
-- If a tag appears more than once, then its value becomes a list of strings.
|
|
-- Alias substitution and @TYPE NAME shortcutting is handled by Item.check_tag
|
|
local function extract_tags (s)
|
|
if s:match '^%s*$' then return {} end
|
|
local preamble,tag_items = parse_tags(s)
|
|
|
|
local strip = tools.strip
|
|
local summary,description = preamble:match('^(.-)[%.?]%s(.+)')
|
|
if not summary then summary = preamble end
|
|
summary = summary .. '.'
|
|
local tags = {summary=summary and strip(summary),description=description and strip(description)}
|
|
for _,item in ipairs(tag_items) do
|
|
local tag,value = item[1],item[2]
|
|
tag = Item.check_tag(tags,tag)
|
|
value = strip(value)
|
|
local old_value = tags[tag]
|
|
if old_value then
|
|
if type(old_value)=='string' then tags[tag] = List{old_value} end
|
|
tags[tag]:append(value)
|
|
else
|
|
tags[tag] = value
|
|
end
|
|
end
|
|
return Map(tags)
|
|
end
|
|
|
|
local quit = utils.quit
|
|
|
|
|
|
class.Lang()
|
|
|
|
function Lang:trim_comment (s)
|
|
return s:gsub(self.line_comment,'')
|
|
end
|
|
|
|
function Lang:start_comment (v)
|
|
local line = v:match (self.start_comment_)
|
|
local block = v:match(self.block_comment)
|
|
return line or block, block
|
|
end
|
|
|
|
function Lang:empty_comment (v)
|
|
return v:match(self.empty_comment_)
|
|
end
|
|
|
|
function Lang:grab_block_comment(v,tok)
|
|
v = v:gsub(self.block_comment,'')
|
|
return tools.grab_block_comment(v,tok,self.end_block1,self.end_block2)
|
|
end
|
|
|
|
function Lang:find_module(tok,t,v)
|
|
return '...',t,v
|
|
end
|
|
|
|
function Lang:function_follows(t,v)
|
|
return false
|
|
end
|
|
|
|
function Lang:finalize()
|
|
self.empty_comment_ = self.start_comment_..'%s*$'
|
|
end
|
|
|
|
class.Lua(Lang)
|
|
|
|
function Lua:_init()
|
|
self.line_comment = '^%-%-+' -- used for stripping
|
|
self.start_comment_ = '^%-%-%-+' -- used for doc comment line start
|
|
self.block_comment = '^%-%-%[%[%-+' -- used for block doc comments
|
|
self.end_block1 = ']'
|
|
self.end_block2 = ']'
|
|
self:finalize()
|
|
end
|
|
|
|
function Lua.lexer(fname)
|
|
local f,e = io.open(fname)
|
|
if not f then quit(e) end
|
|
return lexer.lua(f,{}),f
|
|
end
|
|
|
|
-- If a module name was not provided, then we look for an explicit module()
|
|
-- call. However, we should not try too hard; if we hit a doc comment then
|
|
-- we should go back and process it. Likewise, module(...) also means
|
|
-- that we must infer the module name.
|
|
function Lua:find_module(tok,t,v)
|
|
while t and not (t == 'iden' and v == 'module') do
|
|
if t == 'comment' and self:start_comment(v) then return nil,t,v end
|
|
--print(t,v)
|
|
t,v = tnext(tok)
|
|
end
|
|
if not t then return nil end
|
|
t,v = tnext(tok)
|
|
if t == '(' then t,v = tnext(tok) end
|
|
if t == 'string' then -- explicit name, cool
|
|
return v,t,v
|
|
elseif t == '...' then -- we have to guess!
|
|
return '...',t,v
|
|
end
|
|
end
|
|
|
|
function Lua:function_follows(t,v)
|
|
return t == 'keyword' and v == 'function'
|
|
end
|
|
|
|
|
|
local lua = Lua()
|
|
|
|
class.CC(Lang)
|
|
|
|
function CC:_init()
|
|
self.line_comment = '^//+'
|
|
self.start_comment_ = '^///+'
|
|
self.block_comment = '^/%*%*+'
|
|
self:finalize()
|
|
end
|
|
|
|
function CC.lexer(f)
|
|
f,err = utils.readfile(f)
|
|
if not f then quit(err) end
|
|
return lexer.cpp(f,{})
|
|
end
|
|
|
|
function CC:grab_block_comment(v,tok)
|
|
v = v:gsub(self.block_comment,'')
|
|
return 'comment',v:sub(1,-3)
|
|
end
|
|
|
|
local cc = CC()
|
|
|
|
|
|
-- parses a Lua file, looking for ldoc comments. These are like LuaDoc comments;
|
|
-- they start with multiple '-'. If they don't define a name tag, then by default
|
|
-- it is assumed that a function definition follows. If it is the first comment
|
|
-- encountered, then ldoc looks for a call to module() to find the name of the
|
|
-- module.
|
|
local function parse_file(fname,lang)
|
|
local line,f = 1
|
|
local F = File(fname)
|
|
local module_found, first_comment = false,true
|
|
|
|
local tok,f = lang.lexer(fname)
|
|
local toks = tools.space_skip_getter(tok)
|
|
|
|
function lineno () return lexer.lineno(tok) end
|
|
function filename () return fname end
|
|
|
|
function F:warning (msg,kind)
|
|
kind = kind or 'warning'
|
|
lineno() -- why is this necessary?
|
|
lineno()
|
|
io.stderr:write(kind..' '..fname..':'..lineno()..' '..msg,'\n')
|
|
end
|
|
|
|
function F:error (msg)
|
|
self:warning(msg,'error')
|
|
os.exit(1)
|
|
end
|
|
|
|
local function add_module(tags,module_found,old_style)
|
|
tags.name = module_found
|
|
tags.class = 'module'
|
|
local item = F:new_item(tags,lineno())
|
|
item.old_style = old_style
|
|
end
|
|
|
|
local t,v = tok()
|
|
while t do
|
|
if t == 'comment' then
|
|
local comment = {}
|
|
local ldoc_comment,block = lang:start_comment(v)
|
|
if ldoc_comment and block then
|
|
t,v = lang:grab_block_comment(v,tok)
|
|
end
|
|
if lang:empty_comment(v) then -- ignore rest of empty start comments
|
|
t,v = tok()
|
|
end
|
|
while t and t == 'comment' do
|
|
v = lang:trim_comment(v)
|
|
append(comment,v)
|
|
t,v = tok()
|
|
end
|
|
if not t then break end -- no more file!
|
|
|
|
if t == 'space' then t,v = tnext(tok) end
|
|
|
|
local fun_follows,tags
|
|
if ldoc_comment or first_comment then
|
|
comment = table.concat(comment)
|
|
fun_follows = lang:function_follows(t,v)
|
|
if fun_follows or comment:find '@' or first_comment then
|
|
tags = extract_tags(comment)
|
|
-- handle the special case where the initial module comment was not
|
|
-- an ldoc style comment
|
|
if not ldoc_comment and first_comment and not tags.class then
|
|
tags.class = 'module'
|
|
ldoc_comment = true
|
|
F:warning 'Module doc comment assumed'
|
|
end
|
|
if doc.project_level(tags.class) then
|
|
module_found = tags.name
|
|
end
|
|
end
|
|
first_comment = false
|
|
end
|
|
-- some hackery necessary to find the module() call
|
|
if not module_found and ldoc_comment then
|
|
local old_style
|
|
module_found,t,v = lang:find_module(tok,t,v)
|
|
-- right, we can add the module object ...
|
|
old_style = module_found ~= nil
|
|
if not module_found or module_found == '...' then
|
|
if not t then quit(fname..": end of file") end -- run out of file!
|
|
-- we have to guess the module name
|
|
module_found = tools.this_module_name(args.package,fname)
|
|
end
|
|
if not tags then tags = extract_tags(comment) end
|
|
add_module(tags,module_found,old_style)
|
|
tags = nil
|
|
-- if we did bump into a doc comment, then we can continue parsing it
|
|
end
|
|
|
|
-- end of a group of comments (may be just one)
|
|
if ldoc_comment and tags then
|
|
-- ldoc block
|
|
if fun_follows then -- parse the function definition
|
|
tags.name = tools.get_fun_name(tok)
|
|
tags.formal_args = tools.get_parameters(toks)
|
|
tags.class = 'function'
|
|
end
|
|
if tags.name then
|
|
F:new_item(tags,lineno()).inferred = fun_follows
|
|
end
|
|
end
|
|
end
|
|
if t ~= 'comment' then t,v = tok() end
|
|
end
|
|
if f then f:close() end
|
|
return F
|
|
end
|
|
|
|
function read_file(name,lang)
|
|
local F = parse_file(name,lang)
|
|
F:finish()
|
|
return F
|
|
end
|
|
|
|
--- processing command line and preparing for output ---
|
|
|
|
local F
|
|
local file_list,module_list = List(),List()
|
|
module_list.by_name = {}
|
|
local multiple_files
|
|
local config_dir
|
|
|
|
local function extract_modules (F)
|
|
for mod in F.modules:iter() do
|
|
module_list:append(mod)
|
|
module_list.by_name[mod.name] = mod
|
|
end
|
|
end
|
|
|
|
-- ldoc -m is expecting a Lua package; this converts this to a file path
|
|
if args.module then
|
|
local fullpath,lua = path.package_path(args.file)
|
|
if not fullpath then
|
|
local mpath,name = tools.split_dotted_name(args.file)
|
|
fullpath,lua = path.package_path(mpath)
|
|
if not fullpath then
|
|
quit("module "..args.file.." not found on module path")
|
|
else
|
|
args.module = name
|
|
end
|
|
end
|
|
if not lua then quit("module "..args.file.." is a binary extension") end
|
|
args.file = fullpath
|
|
end
|
|
|
|
if args.file == '.' then
|
|
args.file = lfs.currentdir()
|
|
else
|
|
args.file = path.abspath(args.file)
|
|
end
|
|
|
|
local source_dir = args.file
|
|
if path.isfile(args.file) then
|
|
source_dir = path.splitpath(source_dir)
|
|
end
|
|
|
|
if args.package == '.' then
|
|
args.package = source_dir
|
|
elseif args.package == '..' then
|
|
args.package = path.splitpath(source_dir)
|
|
end
|
|
|
|
local file_types = {
|
|
['.lua'] = lua,
|
|
['.ldoc'] = lua,
|
|
['.luadoc'] = lua,
|
|
['.c'] = cc,
|
|
['.cpp'] = cc,
|
|
['.cxx'] = cc,
|
|
['.C'] = cc
|
|
}
|
|
|
|
local CONFIG_NAME = 'config.ld'
|
|
|
|
if path.isdir(args.file) then
|
|
local files = List(dir.getallfiles(args.file,'*.*'))
|
|
local config_files = files:filter(function(f)
|
|
return path.basename(f) == CONFIG_NAME
|
|
end)
|
|
|
|
-- finding more than one should probably be a warning...
|
|
if #config_files > 0 then
|
|
config_dir = read_ldoc_config(config_files[1])
|
|
end
|
|
|
|
for f in files:iter() do
|
|
local ext = path.extension(f)
|
|
local ftype = file_types[ext]
|
|
if ftype then
|
|
if args.verbose then print(path.basename(f)) end
|
|
local F = read_file(f,ftype)
|
|
file_list:append(F)
|
|
end
|
|
end
|
|
for F in file_list:iter() do
|
|
extract_modules(F)
|
|
end
|
|
multiple_files = true
|
|
elseif path.isfile(args.file) then
|
|
-- a single file may be accompanied by a config.ld in the same dir
|
|
local config_dir = path.dirname(args.file)
|
|
if config_dir == '' then config_dir = '.' end
|
|
local config = path.join(config_dir,CONFIG_NAME)
|
|
if path.isfile(config) then
|
|
read_ldoc_config(config)
|
|
end
|
|
local ext = path.extension(args.file)
|
|
local ftype = file_types[ext]
|
|
if not ftype then quit "unsupported extension" end
|
|
F = read_file(args.file,ftype)
|
|
extract_modules(F)
|
|
else
|
|
quit ("file or directory does not exist")
|
|
end
|
|
|
|
local project = ProjectMap()
|
|
|
|
for mod in module_list:iter() do
|
|
mod:resolve_references(module_list)
|
|
project:add(mod,module_list)
|
|
end
|
|
|
|
table.sort(module_list,function(m1,m2)
|
|
return m1.name < m2.name
|
|
end)
|
|
|
|
-- ldoc -m will give a quick & dirty dump of the module's documentation;
|
|
-- using -v will make it more verbose
|
|
if args.module then
|
|
if #module_list == 0 then quit("no modules found") end
|
|
if args.module == true then
|
|
F:dump(args.verbose)
|
|
else
|
|
local fun = module_list[1].items.by_name[args.module]
|
|
if not fun then quit(args.module.." is not part of this module") end
|
|
fun:dump(true)
|
|
end
|
|
return
|
|
end
|
|
|
|
if args.dump then
|
|
for mod in module_list:iter() do
|
|
mod:dump(true)
|
|
end
|
|
os.exit()
|
|
end
|
|
|
|
local css, templ = 'ldoc.css','ldoc.ltp'
|
|
|
|
-- the style directory for template and stylesheet can be specified
|
|
-- either by command-line 'style' argument or by 'style' field in
|
|
-- config.ld. Then it is relative to the location of that file.
|
|
if ldoc.style then args.style = path.join(config_dir,ldoc.style) end
|
|
|
|
-- '!' here means 'use same directory as the ldoc.lua script'
|
|
if args.style == '!' then
|
|
args.style = arg[0]:gsub('[^/\\]+$','')
|
|
end
|
|
|
|
local module_template,err = utils.readfile (path.join(args.style,templ))
|
|
if not module_template then quit(err) end
|
|
|
|
-- can specify formatter in config.ld
|
|
if ldoc.format then args.format = ldoc.format end
|
|
|
|
if args.format ~= 'plain' then
|
|
local ok,markup = pcall(require,args.format)
|
|
if not ok then quit("cannot load formatter: "..args.format) end
|
|
function ldoc.markup(txt)
|
|
txt = markup(txt)
|
|
return (txt:gsub('^%s*<p>(.+)</p>%s*$','%1'))
|
|
end
|
|
else
|
|
function ldoc.markup(txt)
|
|
return txt
|
|
end
|
|
end
|
|
|
|
function generate_output()
|
|
-- ldoc.single = not multiple_files
|
|
local check_directory, check_file, writefile = tools.check_directory, tools.check_file, tools.writefile
|
|
ldoc.log = print
|
|
ldoc.kinds = project
|
|
ldoc.css = css
|
|
ldoc.modules = module_list
|
|
ldoc.title = ldoc.title or args.title
|
|
ldoc.project = ldoc.project or args.project
|
|
|
|
local out,err = template.substitute(module_template,{ ldoc = ldoc })
|
|
if not out then quit(err) end
|
|
|
|
check_directory(args.dir)
|
|
|
|
args.dir = args.dir .. path.sep
|
|
|
|
check_file(args.dir..css, path.join(args.style,css))
|
|
|
|
-- write out the module index
|
|
writefile(args.dir..'index.html',out)
|
|
|
|
-- write out the per-module documentation
|
|
ldoc.css = '../'..css
|
|
for kind, modules in project() do
|
|
kind = kind:lower()
|
|
check_directory(args.dir..kind)
|
|
for m in modules() do
|
|
out,err = template.substitute(module_template,{
|
|
module=m,
|
|
ldoc = ldoc
|
|
})
|
|
if not out then
|
|
quit('template failed for '..m.name..': '..err)
|
|
else
|
|
writefile(args.dir..kind..'/'..m.name..'.html',out)
|
|
end
|
|
end
|
|
end
|
|
if not args.quiet then print('output written to '..args.dir) end
|
|
end
|
|
|
|
generate_output()
|
|
|
|
if args.verbose then
|
|
print 'modules'
|
|
for k in pairs(module_list.by_name) do print(k) end
|
|
end
|
|
|
|
|