mirror of
https://github.com/elua/elua.git
synced 2025-01-08 20:56:17 +08:00
380 lines
13 KiB
Lua
380 lines
13 KiB
Lua
-------------------------------------------------------------------------------
|
|
-- eLua doc builder module (for the eluadoc/ directory)
|
|
|
|
module( ..., package.seeall )
|
|
|
|
local sf = string.format
|
|
|
|
-------------------------------------------------------------------------------
|
|
-- Data structure declarations
|
|
|
|
-- List here all the sections for which we're generating the documentation
|
|
local doc_sections = { "arch_platform", "refman_gen", "refman_ps_lm3s", "refman_ps_str9", "refman_ps_stm32", "refman_ps_stm32f4", "refman_ps_mbed", "refman_ps_mizar32" }
|
|
|
|
-- List here all the components of each section
|
|
local components =
|
|
{
|
|
arch_platform = { "ll", "pio", "spi", "uart", "timers", "pwm", "cpu", "eth", "adc", "i2c", "can", "flash" },
|
|
refman_gen = { "bit", "pd", "cpu", "pack", "adc", "term", "pio", "uart", "spi", "tmr", "pwm", "net", "can", "rpc", "elua", "i2c" },
|
|
refman_ps_lm3s = { "disp" },
|
|
refman_ps_str9 = { "pio" },
|
|
refman_ps_mbed = { "pio" },
|
|
refman_ps_stm32 = { "enc" },
|
|
refman_ps_stm32f4 = { "enc" },
|
|
refman_ps_mizar32 = { "lcd", "rtc" },
|
|
}
|
|
|
|
-------------------------------------------------------------------------------
|
|
-- Generic helpers and doc text formatting functions
|
|
|
|
-- Format a name to a link by changing all the spaces to "_" and
|
|
-- making all letters lowercase
|
|
local function name2link( str )
|
|
str = str:gsub( " ", "_" )
|
|
return str:lower()
|
|
end
|
|
|
|
-- Returns the part of the string enclosed between two '#' chars
|
|
-- Used for parsing function sig.
|
|
local function namefromsig( str )
|
|
local _, _, name = str:find( "#(.*)#" )
|
|
return name
|
|
end
|
|
|
|
-- Adds a "." to the end of the string if it's not already present
|
|
local function dot( str )
|
|
-- return str:sub( -1 ) == "." and str or str .. "."
|
|
return str
|
|
end
|
|
|
|
--[[ Process the given string as follows:
|
|
- $string$ becomes <b>string</b>
|
|
- %string% becomes <i>string</i>
|
|
- @ref@text@ becomes <a href="ref">text</a>
|
|
- ^ref^text^ also becomes <a href="ref">text</a>
|
|
- $$, %%, @@, ^^ become $, %, @, ^ respectively
|
|
- the string "eLua" becomes <b>eLua</b>
|
|
- strings between two tildas (~~) get special code-like formatting
|
|
- newlines are changed to ' ' if 'keepnl' isn't true
|
|
- '&' is translated to its corresponding HTML code.
|
|
- '<<' and '>>" are also translated to the corresponding HTML codes (note the repetition).
|
|
--]]
|
|
local function format_string( str, keepnl )
|
|
-- replace double "special chars" with "temps" for later use
|
|
str = str:gsub( "%$%$", "\001" )
|
|
str = str:gsub( "%%%%", "\002" )
|
|
str = str:gsub( "@@", "\003" )
|
|
str = str:gsub( "%^%^", "\004" )
|
|
str = str:gsub( "~~", "\005" )
|
|
|
|
-- Translate 'special' HTML chars to their equivalents
|
|
local tr_table =
|
|
{
|
|
[ "%&" ] = "&",
|
|
}
|
|
for char, rep in pairs( tr_table ) do
|
|
str = str:gsub( char, rep )
|
|
end
|
|
|
|
-- some double chars are replaced directly with their HTML codes
|
|
str = str:gsub( "<<", "<" )
|
|
str = str:gsub( ">>", ">" )
|
|
|
|
-- replace eLua with <b>eLua</b>
|
|
str = str:gsub( "eLua", "<b>eLua</b>" )
|
|
|
|
-- $string$ becomes <b>string></b>
|
|
str = str:gsub( "%$(.-)%$", "<b>%1</b>" )
|
|
|
|
-- %string% becomes <i>string</i>
|
|
str = str:gsub( "%%(.-)%%", "<i>%1</i>" )
|
|
|
|
-- @ref@text@ becomes <a href="ref">text</a>
|
|
str = str:gsub( "@(.-)@(.-)@", '<a href="%1">%2</a>' )
|
|
|
|
-- ^ref^text^ becomes <a href="ref">text</a>
|
|
str = str:gsub( "%^(.-)%^(.-)%^", '<a href="%1">%2</a>' )
|
|
|
|
-- strings between two tildas (~~) get special code-like formatting
|
|
-- must keep '\n', so replace it with "temps" for now
|
|
str = str:gsub( "~(.-)~", function( data ) return '<pre class="code">' .. data:gsub( "\n", "\006" ) .. "</pre>" end )
|
|
str = str:gsub( "~~", "~" )
|
|
|
|
-- other "\n" chars should dissapear now
|
|
if not keepnl then str = str:gsub( "\n", " " ) end
|
|
|
|
-- put back the "temps"
|
|
str = str:gsub( "\001", "%$" )
|
|
str = str:gsub( "\002", "%%" )
|
|
str = str:gsub( "\003", "@" )
|
|
str = str:gsub( "\004", "%^" )
|
|
str = str:gsub( "\005", "~" )
|
|
str = str:gsub( "\006", "\n" )
|
|
|
|
-- all done
|
|
return str
|
|
end
|
|
|
|
-------------------------------------------------------------------------------
|
|
-- Content generation
|
|
|
|
-- Build the documentation starting from the given file
|
|
local function build_file( fname )
|
|
dofile( fname )
|
|
local res = {}
|
|
|
|
for _, lang in pairs( languages ) do
|
|
res[ lang ] = {}
|
|
res[ lang ].menu = {}
|
|
local menu = res[ lang ].menu
|
|
|
|
-- we need english always
|
|
-- the other languages will be substituted with english if not found
|
|
local resname = string.format( "data_%s", lang )
|
|
local r = _G[ resname ]
|
|
if not r then
|
|
if lang == "en" then
|
|
return false, "data_en must exist in the description"
|
|
else
|
|
print( string.format( "'%s': data for language '%s' not found, defaulting to english", fname, lang ) )
|
|
r = _G.data_en
|
|
end
|
|
end
|
|
|
|
-- process names
|
|
if not r.menu_name then
|
|
return false, "menu_names not found"
|
|
end
|
|
menu.name = r.menu_name
|
|
|
|
-- process title
|
|
if not r.title then
|
|
return false, "title not found"
|
|
end
|
|
local page = "$$HEADER$$\n"
|
|
menu.title = r.title
|
|
|
|
-- process overview
|
|
if not r.overview then
|
|
return false, "overview not found"
|
|
end
|
|
page = page .. '<a name="overview" /><h3>Overview</h3>\n<p>' .. format_string( r.overview ) .. "</p>\n\n"
|
|
|
|
-- process structures if needed
|
|
if r.structures then
|
|
local structures = r.structures
|
|
menu.structs = {}
|
|
page = page .. '<a name="structures" /><h3>Data structures, constants and types</h3>\n'
|
|
for i = 1, #structures do
|
|
local s = structures[ i ]
|
|
menu.structs[ #menu.structs + 1 ] = s.name
|
|
if not s.text or not s.desc or not s.name then
|
|
return false, "structure without text, desc or name fields"
|
|
end
|
|
-- text/name. The link name is ALWAYS the one in ENGLISH.
|
|
page = page .. string.format( '<a name="%s" />', name2link( res.en.menu.structs[ i ] ) )
|
|
page = page .. "<pre><code>" .. format_string( s.text, true ) .. "</code></pre>\n"
|
|
-- description
|
|
page = page .. '<div class="docdiv">\n<p>' .. format_string( s.desc ) .. "</p>\n</div>\n\n"
|
|
end
|
|
end
|
|
|
|
-- process functions now
|
|
if not r.funcs then
|
|
return false, "funcs not found"
|
|
end
|
|
local funcs = r.funcs
|
|
local functions_name = "<div class='functions'>"
|
|
for _,f in pairs(funcs)do
|
|
functions_name = functions_name.."<a href='#"..namefromsig( f.sig ) .."'>"..namefromsig( f.sig ) .."</a> "
|
|
end
|
|
|
|
page = page .. '<a name="funcs" /><h3>Functions</h3>\n<div class="docdiv">\n'.. functions_name.."</div>\n"
|
|
|
|
|
|
menu.funcs = {}
|
|
|
|
for i = 1, #funcs do
|
|
local f = funcs[ i ]
|
|
if not f.sig or not f.desc then
|
|
return false, "function without sig or desc fields"
|
|
end
|
|
local funcname = namefromsig( f.sig )
|
|
|
|
if not funcname then
|
|
return false, string.format( "'%s' should contain the function name between '*' chars", f.sig )
|
|
end
|
|
|
|
--menu.funcs[ #menu.funcs + 1 ] = funcname
|
|
-- signature
|
|
|
|
page = page .. string.format( '<a name="%s" />', funcname )
|
|
page = page .. "<div class='function-block'><h2>" .. f.sig:gsub( '#', '' ) .. "</h2>\n"
|
|
-- description
|
|
page = page .. "\n<p>" .. dot( format_string( f.desc ) ) .. "</p>\n"
|
|
-- arguments
|
|
page = page .. "<p><b>Arguments</b>: "
|
|
if f.args then
|
|
local a = f.args
|
|
if type( a ) == "string" or ( type( a ) == "table" and #a == 1 ) then
|
|
local text = type( a ) == "string" and a or a[ 1 ]
|
|
page = page .. dot( format_string( text ) ) .. "</p>"
|
|
else
|
|
page = page .. "</p>\n<ul>\n"
|
|
for i = 1, #a do page = page .. " <li>" .. dot( format_string( a[ i ] ) ) .. "</li>\n" end
|
|
page = page .. "</ul>"
|
|
end
|
|
else
|
|
page = page .. "none.</p>"
|
|
end
|
|
page = page .. "\n"
|
|
-- return value
|
|
page = page .. "<p><b>Returns</b>: "
|
|
if f.ret then
|
|
local r = f.ret
|
|
if type( r ) == "string" or ( type( r ) == "table" and #r == 1 ) then
|
|
local text = type( r ) == "string" and r or r[ 1 ]
|
|
page = page .. dot( format_string( text ) ) .. "</p>"
|
|
else
|
|
page = page .. "</p>\n<ul>\n"
|
|
for i = 1, #r do page = page .. " <li>" .. dot( format_string( r[ i ] ) ) .. "</li>\n" end
|
|
page = page .. "</ul>"
|
|
end
|
|
else
|
|
page = page .. "nothing.</p>"
|
|
end
|
|
page = page .. "\n\n</div>"
|
|
end
|
|
page = page .. "</div>\n"
|
|
|
|
-- aux data (if any)
|
|
if r.auxdata then
|
|
local auxdata = r.auxdata
|
|
menu.auxdata = {}
|
|
for i = 1, #auxdata do
|
|
local a = auxdata[ i ]
|
|
menu.auxdata[ #menu.auxdata + 1 ] = a.title
|
|
if not a.title or not a.desc then
|
|
return false, "auxdata without title or desc"
|
|
end
|
|
-- the link name is ALWAYS the one in ENGLISH
|
|
page = page .. string.format( '<a name="%s" />', name2link( res.en.menu.auxdata[ i ] ) )
|
|
page = page .. "<h3>" .. a.title .. "</h3>"
|
|
page = page .. "\n<p>" .. format_string( a.desc ) .. "</p>\n\n"
|
|
end
|
|
end
|
|
|
|
-- footer
|
|
page = page .. "$$FOOTER$$\n"
|
|
-- Cleanup: remove "<p></p>" (which might appear due to formatting)
|
|
page = page:gsub( "<p>%s-</p>", "" )
|
|
res[ lang ].page = page
|
|
end
|
|
|
|
return res
|
|
end
|
|
|
|
-------------------------------------------------------------------------------
|
|
-- Menu generation
|
|
|
|
-- Helper function to get strings in all languages when needed
|
|
local function all_langs( getstr )
|
|
local langs = {}
|
|
for _, lang in pairs( languages ) do
|
|
langs[ #langs + 1 ] = getstr( lang )
|
|
end
|
|
return langs
|
|
end
|
|
|
|
-- Transform the data from the menu dictionary (in 'fulldata') for component 'component' and section 'sect' to a menu structure
|
|
local function gen_menu( fulldata, component, sect )
|
|
local relfname = sect .. "_" .. component .. ".html"
|
|
local res = fulldata[ component ]
|
|
local themenu = { all_langs( function( x ) return res[ x ].menu.name end ), relfname, {}, all_langs( function( x ) return res[ x ].menu.title end ) }
|
|
local sub = themenu[ submenu_idx ]
|
|
|
|
-- Overview
|
|
sub[ #sub + 1 ] = { all_langs( function( x ) return getstr( "Overview", x ) end ), sf( "%s#overview", relfname ) }
|
|
|
|
-- Data structures (if needed)
|
|
if res.en.menu.structs then
|
|
sub[ #sub + 1 ] = { all_langs( function( x ) return getstr( "Data structures", x ) end ), sf( "%s#structures", relfname ), {} }
|
|
local s_sub = sub[ #sub ][ submenu_idx ]
|
|
for i = 1, #res.en.menu.structs do
|
|
local v = res.en.menu.structs[ i ]
|
|
s_sub[ #s_sub + 1 ] = { all_langs( function( x ) return res[ x ].menu.structs[ i ] end ), sf( "%s#%s", relfname, name2link( v ) ) }
|
|
end
|
|
end
|
|
|
|
-- Functions
|
|
--[[
|
|
sub[ #sub + 1 ] = { all_langs( function( x ) return getstr( "Functions", x ) end ), sf( "%s#funcs", relfname ), {} }
|
|
local f_sub = sub[ #sub ][ submenu_idx ]
|
|
for _, v in pairs( res.en.menu.funcs ) do
|
|
f_sub[ #f_sub + 1 ] = { all_langs( function( x ) return v end ), sf( "%s#%s", relfname, name2link( v ) ) }
|
|
end
|
|
]]
|
|
sub[ #sub + 1 ] = { all_langs( function( x ) return getstr( "Functions", x ) end ), sf( "%s#funcs", relfname ) }
|
|
|
|
-- Aux data (if needed)
|
|
if res.en.menu.auxdata then
|
|
for i = 1, #res.en.menu.auxdata do
|
|
local v = res.en.menu.auxdata[ i ]
|
|
sub[ #sub + 1 ] = { all_langs( function( x ) return res[ x ].menu.auxdata[ i ] end ), sf( "%s#%s", relfname, name2link( v ) ) }
|
|
end
|
|
end
|
|
|
|
return themenu
|
|
end
|
|
|
|
-------------------------------------------------------------------------------
|
|
-- Generate documentation from eluadoc for all languages
|
|
|
|
function gen_html_doc()
|
|
local menu, genfiles = {}, {}
|
|
|
|
for k, v in pairs( components ) do
|
|
table.sort( v )
|
|
end
|
|
|
|
for _, section in pairs( doc_sections ) do
|
|
-- Generate documentation for each module in turn
|
|
local fulldata = {}
|
|
menu[ section ] = {}
|
|
local ms = menu[ section ]
|
|
-- First generate HTML documentation
|
|
for _, modname in pairs( components[ section ] ) do
|
|
local descfname = string.format( "eluadoc/%s_%s.lua", section, modname )
|
|
local res, err = build_file( descfname )
|
|
if res then
|
|
fulldata[ modname ] = res
|
|
-- Write doc for each language
|
|
for _, lang in pairs( languages ) do
|
|
local fname = string.format( "%s/%s_%s.html", lang, section, modname )
|
|
local f = io.open( fname, "wb" )
|
|
if not f then
|
|
print( string.format( "Unable to open %s for writing", fname ) )
|
|
return
|
|
else
|
|
f:write( res[ lang ].page )
|
|
f:close()
|
|
print( ( "Wrote %s" ):format( fname ) )
|
|
genfiles[ #genfiles + 1 ] = fname
|
|
end
|
|
end
|
|
else
|
|
print( string.format( "Error processing module '%s': %s", modname, err ) )
|
|
return
|
|
end
|
|
end
|
|
|
|
-- Then generate menu data
|
|
for _, modname in pairs( components[ section ] ) do
|
|
local submenu= gen_menu( fulldata, modname, section )
|
|
ms[ #ms + 1 ] = submenu
|
|
end
|
|
end
|
|
return menu, genfiles
|
|
end
|
|
|