local fs = require 'fs' local emmyPattern = '^%-%-%- (.+)' local emmyPattern2 = '^%-%- (.+)' local modpattern = '^%-+ @module (.+)' local pieces = {} local descriptions = {} local files = fs.readdir 'nature' for _, fname in ipairs(files) do local isScript = fname:match'%.lua$' if not isScript then goto continue end local f = io.open(string.format('nature/%s', fname)) local header = f:read '*l' local mod = header:match(modpattern) if not mod then goto continue end print(fname, mod) pieces[mod] = {} descriptions[mod] = {} local docPiece = {} local lines = {} local lineno = 0 local doingDescription = true for line in f:lines() do lineno = lineno + 1 lines[lineno] = line if line == header then goto continue2 end if line:match(emmyPattern) or line:match(emmyPattern2) then if doingDescription then table.insert(descriptions[mod], line:match(emmyPattern) or line:match(emmyPattern2)) end else doingDescription = false if line:match '^function' then local pattern = (string.format('^function %s%%.', mod) .. '(%w+)') local funcName = line:match(pattern) if not funcName then goto continue2 end local dps = { description = {}, params = {} } local offset = 1 while true do local prev = lines[lineno - offset] local docline = prev:match '^%-+ (.+)' if docline then local emmy = docline:match '@(%w+)' local cut = 0 if emmy then cut = emmy:len() + 3 end local emmythings = string.split(docline:sub(cut), ' ') if emmy then if emmy == 'param' then print('bruh', emmythings[1], emmythings[2]) table.insert(dps.params, 1, { name = emmythings[1], type = emmythings[2], -- the +1 accounts for space. description = table.concat(emmythings, ' '):sub(emmythings[1]:len() + 1 + emmythings[2]:len() + 1) }) print(table.concat(emmythings, '/')) end else table.insert(dps.description, 1, docline) end offset = offset + 1 else break end end pieces[mod][funcName] = dps end docPiece = {} goto continue2 end table.insert(docPiece, line) ::continue2:: end ::continue:: end local header = [[--- title: %s %s description: %s layout: doc menu: docs: parent: "%s" --- ]] for iface, dps in pairs(pieces) do local mod = iface:match '(%w+)%.' or 'nature' local docParent = 'Nature' path = string.format('docs/%s/%s.md', mod, iface) if mod ~= 'nature' then docParent = "API" path = string.format('docs/api/%s/%s.md', mod, iface) end fs.mkdir(fs.dir(path), true) local exists = pcall(fs.stat, path) local newOrNotNature = exists and mod ~= 'nature' local f = io.open(path, newOrNotNature and 'r+' or 'w+') local tocPos if not newOrNotNature then f:write(string.format(header, 'Module', iface, (descriptions[iface] and #descriptions[iface] > 0) and descriptions[iface][1] or 'No description.', docParent)) if descriptions[iface] and #descriptions[iface] > 0 then table.remove(descriptions[iface], 1) f:write(string.format('\n## Introduction\n%s\n\n', table.concat(descriptions[iface], '\n'))) f:write('## Functions\n') f:write([[||| |----|----| ]]) tocPos = f:seek() end end print(f) print('mod and path:', mod, path) local tocSearch = false for line in f:lines() do if line:match '^## Functions' then tocSearch = true end if tocSearch and line == '' then tocSearch = false tocPos = f:seek() - 1 end end for func, docs in pairs(dps) do local sig = string.format('%s.%s(', iface, func) local params = '' for idx, param in ipairs(docs.params) do sig = sig .. param.name:gsub('%?$', '') params = params .. param.name:gsub('%?$', '') if idx ~= #docs.params then sig = sig .. ', ' params = params .. ', ' end end sig = sig .. ')' if tocPos then f:seek('set', tocPos) local contents = f:read '*a' f:seek('set', tocPos) local tocLine = string.format('|%s|%s|\n', func, string.format('%s(%s)', func, params), docs.description[1]) f:write(tocLine .. contents) f:seek 'end' end f:write(string.format('
\n
\n', func)) f:write(string.format([[

%s

]], sig, func)) f:write(table.concat(docs.description, '\n') .. '\n') f:write '#### Parameters\n' if #docs.params == 0 then f:write 'This function has no parameters. \n' end for _, param in ipairs(docs.params) do f:write(string.format('`%s` **`%s`** \n', param.name:gsub('%?$', ''), param.type)) f:write(string.format('%s\n\n', param.description)) end --[[ local params = table.filter(docs, function(t) return t:match '^%-%-%- @param' end) for i, str in ipairs(params) do if i ~= 1 then f:write ', ' end f:write(str:match '^%-%-%- @param ([%w]+) ') end f:write(')\n') for _, str in ipairs(docs) do if not str:match '^%-%-%- @' then f:write(str:match '^%-%-%- (.+)' .. '\n') end end ]]-- f:write('
') f:write('\n\n') end end