# # # Nim Website Generator # (c) Copyright 2014 Andreas Rumpf # # See the file "copying.txt", included in this # distribution, for details about the copyright. # import os, strutils, times, parseopt, parsecfg, streams, strtabs, tables, re, htmlgen, macros, md5, osproc type TKeyValPair = tuple[key, id, val: string] TConfigData = object of RootObj tabs, links: seq[TKeyValPair] doc, srcdoc, srcdoc2, webdoc, pdf: seq[string] authors, projectName, projectTitle, logo, infile, outdir, ticker: string vars: StringTableRef nimArgs: string gitCommit: string quotations: Table[string, tuple[quote, author: string]] numProcessors: int # Set by parallelBuild:n, only works for values > 0. TRssItem = object year, month, day, title: string proc initConfigData(c: var TConfigData) = c.tabs = @[] c.links = @[] c.doc = @[] c.srcdoc = @[] c.srcdoc2 = @[] c.webdoc = @[] c.pdf = @[] c.infile = "" c.outdir = "" c.nimArgs = "--hint[Conf]:off --hint[Path]:off --hint[Processing]:off " c.authors = "" c.projectTitle = "" c.projectName = "" c.logo = "" c.ticker = "" c.vars = newStringTable(modeStyleInsensitive) c.gitCommit = "master" c.numProcessors = countProcessors() # Attempts to obtain the git current commit. when false: let (output, code) = execCmdEx("git log -n 1 --format=%H") if code == 0 and output.strip.len == 40: c.gitCommit = output.strip c.quotations = initTable[string, tuple[quote, author: string]]() include "website.tmpl" # ------------------------- configuration file ------------------------------- const version = "0.7" usage = "nimweb - Nim Website Generator Version " & version & """ (c) 2014 Andreas Rumpf Usage: nimweb [options] ini-file[.ini] [compile_options] Options: -o, --output:dir set the output directory (default: same as ini-file) --var:name=value set the value of a variable -h, --help shows this help -v, --version shows the version Compile_options: will be passed to the Nim compiler """ rYearMonthDayTitle = r"(\d{4})-(\d{2})-(\d{2})\s+(.*)" rssUrl = "http://nim-lang.org/news.xml" rssNewsUrl = "http://nim-lang.org/news.html" validAnchorCharacters = Letters + Digits macro id(e: expr): expr {.immediate.} = ## generates the rss xml ``id`` element. let e = callsite() result = xmlCheckedTag(e, "id") macro updated(e: expr): expr {.immediate.} = ## generates the rss xml ``updated`` element. let e = callsite() result = xmlCheckedTag(e, "updated") proc updatedDate(year, month, day: string): string = ## wrapper around the update macro with easy input. result = updated("$1-$2-$3T00:00:00Z" % [year, repeatStr(2 - len(month), "0") & month, repeatStr(2 - len(day), "0") & day]) macro entry(e: expr): expr {.immediate.} = ## generates the rss xml ``entry`` element. let e = callsite() result = xmlCheckedTag(e, "entry") macro content(e: expr): expr {.immediate.} = ## generates the rss xml ``content`` element. let e = callsite() result = xmlCheckedTag(e, "content", reqAttr = "type") proc parseCmdLine(c: var TConfigData) = var p = initOptParser() while true: next(p) var kind = p.kind var key = p.key var val = p.val case kind of cmdArgument: c.infile = addFileExt(key, "ini") c.nimArgs.add(cmdLineRest(p)) break of cmdLongOption, cmdShortOption: case normalize(key) of "help", "h": stdout.write(usage) quit(0) of "version", "v": stdout.write(version & "\n") quit(0) of "o", "output": c.outdir = val of "parallelbuild": try: let num = parseInt(val) if num != 0: c.numProcessors = num except ValueError: quit("invalid numeric value for --parallelBuild") of "var": var idx = val.find('=') if idx < 0: quit("invalid command line") c.vars[substr(val, 0, idx-1)] = substr(val, idx+1) else: quit(usage) of cmdEnd: break if c.infile.len == 0: quit(usage) proc walkDirRecursively(s: var seq[string], root, ext: string) = for k, f in walkDir(root): case k of pcFile, pcLinkToFile: if cmpIgnoreCase(ext, splitFile(f).ext) == 0: add(s, f) of pcDir: walkDirRecursively(s, f, ext) of pcLinkToDir: discard proc addFiles(s: var seq[string], dir, ext: string, patterns: seq[string]) = for p in items(patterns): if existsDir(dir / p): walkDirRecursively(s, dir / p, ext) else: add(s, dir / addFileExt(p, ext)) proc parseIniFile(c: var TConfigData) = var p: CfgParser section: string # current section var input = newFileStream(c.infile, fmRead) if input == nil: quit("cannot open: " & c.infile) open(p, input, c.infile) while true: var k = next(p) case k.kind of cfgEof: break of cfgSectionStart: section = normalize(k.section) case section of "project", "links", "tabs", "ticker", "documentation", "var": discard else: echo("[Warning] Skipping unknown section: " & section) of cfgKeyValuePair: var v = k.value % c.vars c.vars[k.key] = v case section of "project": case normalize(k.key) of "name": c.projectName = v of "title": c.projectTitle = v of "logo": c.logo = v of "authors": c.authors = v else: quit(errorStr(p, "unknown variable: " & k.key)) of "var": discard of "links": let valID = v.split(';') add(c.links, (k.key.replace('_', ' '), valID[1], valID[0])) of "tabs": add(c.tabs, (k.key, "", v)) of "ticker": c.ticker = v of "documentation": case normalize(k.key) of "doc": addFiles(c.doc, "doc", ".txt", split(v, {';'})) of "pdf": addFiles(c.pdf, "doc", ".txt", split(v, {';'})) of "srcdoc": addFiles(c.srcdoc, "lib", ".nim", split(v, {';'})) of "srcdoc2": addFiles(c.srcdoc2, "lib", ".nim", split(v, {';'})) of "webdoc": addFiles(c.webdoc, "lib", ".nim", split(v, {';'})) of "parallelbuild": try: let num = parseInt(v) if num != 0: c.numProcessors = num except ValueError: quit("invalid numeric value for --parallelBuild in config") else: quit(errorStr(p, "unknown variable: " & k.key)) of "quotations": let vSplit = v.split('-') doAssert vSplit.len == 2 c.quotations[k.key.normalize] = (vSplit[0], vSplit[1]) else: discard of cfgOption: quit(errorStr(p, "syntax error")) of cfgError: quit(errorStr(p, k.msg)) close(p) if c.projectName.len == 0: c.projectName = changeFileExt(extractFilename(c.infile), "") if c.outdir.len == 0: c.outdir = splitFile(c.infile).dir # Ugly hack to override git command output when building private repo. if c.vars.hasKey("githash"): let githash = c.vars["githash"].strip if githash.len > 0: c.gitCommit = githash # ------------------- main ---------------------------------------------------- proc exec(cmd: string) = echo(cmd) if os.execShellCmd(cmd) != 0: quit("external program failed") proc sexec(cmds: openarray[string]) = ## Serial queue wrapper around exec. for cmd in cmds: exec(cmd) proc mexec(cmds: openarray[string], processors: int) = ## Multiprocessor version of exec if processors < 2: sexec(cmds) return if execProcesses(cmds, {poStdErrToStdOut, poParentStreams, poEchoCmd}) != 0: echo "external program failed, retrying serial work queue for logs!" sexec(cmds) proc buildDocSamples(c: var TConfigData, destPath: string) = ## Special case documentation sample proc. ## ## The docgen sample needs to be generated twice with different commands, so ## it didn't make much sense to integrate into the existing generic ## documentation builders. const src = "doc"/"docgen_sample.nim" exec("nim doc $# -o:$# $#" % [c.nimArgs, destPath / "docgen_sample.html", src]) exec("nim doc2 $# -o:$# $#" % [c.nimArgs, destPath / "docgen_sample2.html", src]) proc buildDoc(c: var TConfigData, destPath: string) = # call nim for the documentation: var commands = newSeq[string](len(c.doc) + len(c.srcdoc) + len(c.srcdoc2)) i = 0 for d in items(c.doc): commands[i] = "nim rst2html $# --docSeeSrcUrl:$# -o:$# --index:on $#" % [c.nimArgs, c.gitCommit, destPath / changeFileExt(splitFile(d).name, "html"), d] i.inc for d in items(c.srcdoc): commands[i] = "nim doc $# --docSeeSrcUrl:$# -o:$# --index:on $#" % [c.nimArgs, c.gitCommit, destPath / changeFileExt(splitFile(d).name, "html"), d] i.inc for d in items(c.srcdoc2): commands[i] = "nim doc2 $# --docSeeSrcUrl:$# -o:$# --index:on $#" % [c.nimArgs, c.gitCommit, destPath / changeFileExt(splitFile(d).name, "html"), d] i.inc mexec(commands, c.numProcessors) exec("nim buildIndex -o:$1/theindex.html $1" % [destPath]) proc buildPdfDoc(c: var TConfigData, destPath: string) = if os.execShellCmd("pdflatex -version") != 0: echo "pdflatex not found; no PDF documentation generated" else: for d in items(c.pdf): exec("nim rst2tex $# $#" % [c.nimArgs, d]) # call LaTeX twice to get cross references right: exec("pdflatex " & changeFileExt(d, "tex")) exec("pdflatex " & changeFileExt(d, "tex")) # delete all the crappy temporary files: let pdf = splitFile(d).name & ".pdf" let dest = destPath / pdf removeFile(dest) moveFile(dest=dest, source=pdf) removeFile(changeFileExt(pdf, "aux")) if existsFile(changeFileExt(pdf, "toc")): removeFile(changeFileExt(pdf, "toc")) removeFile(changeFileExt(pdf, "log")) removeFile(changeFileExt(pdf, "out")) removeFile(changeFileExt(d, "tex")) proc buildAddDoc(c: var TConfigData, destPath: string) = # build additional documentation (without the index): var commands = newSeq[string](c.webdoc.len) for i, doc in pairs(c.webdoc): commands[i] = "nim doc $# --docSeeSrcUrl:$# -o:$# $#" % [c.nimArgs, c.gitCommit, destPath / changeFileExt(splitFile(doc).name, "html"), doc] mexec(commands, c.numProcessors) proc parseNewsTitles(inputFilename: string): seq[TRssItem] = # parses the file for titles and returns them as TRssItem blocks. let reYearMonthDayTitle = re(rYearMonthDayTitle) var input: File line = "" result = @[] if not open(input, inputFilename): quit("Could not read $1 for rss generation" % [inputFilename]) finally: input.close() while input.readLine(line): if line =~ reYearMonthDayTitle: result.add(TRssItem(year: matches[0], month: matches[1], day: matches[2], title: matches[3])) proc genUUID(text: string): string = # Returns a valid RSS uuid, which is basically md5 with dashes and a prefix. result = getMD5(text) result.insert("-", 20) result.insert("-", 16) result.insert("-", 12) result.insert("-", 8) result.insert("urn:uuid:") proc genNewsLink(title: string): string = # Mangles a title string into an expected news.html anchor. result = title result.insert("Z") for i in 1..len(result)-1: let letter = result[i].toLower() if letter in validAnchorCharacters: result[i] = letter else: result[i] = '-' result.insert(rssNewsUrl & "#") proc generateRss(outputFilename: string, news: seq[TRssItem]) = # Given a list of rss items generates an rss overwriting destination. var output: File if not open(output, outputFilename, mode = fmWrite): quit("Could not write to $1 for rss generation" % [outputFilename]) finally: output.close() output.write(""" """) output.write(title("Nim website news")) output.write(link(href = rssUrl, rel = "self")) output.write(link(href = rssNewsUrl)) output.write(id(rssNewsUrl)) let now = getGMTime(getTime()) output.write(updatedDate($now.year, $(int(now.month) + 1), $now.monthday)) for rss in news: let joinedTitle = "$1-$2-$3 $4" % [rss.year, rss.month, rss.day, rss.title] output.write(entry( title(joinedTitle), id(genUUID(joinedTitle)), link(`type` = "text/html", rel = "alternate", href = genNewsLink(joinedTitle)), updatedDate(rss.year, rss.month, rss.day), "Nim", content(joinedTitle, `type` = "text"), )) output.write("""""") proc buildNewsRss(c: var TConfigData, destPath: string) = # generates an xml feed from the web/news.txt file let srcFilename = "web" / "news.txt" destFilename = destPath / changeFileExt(splitFile(srcFilename).name, "xml") generateRss(destFilename, parseNewsTitles(srcFilename)) proc buildJS(destPath: string) = exec("nim js -d:release --out:$1 web/babelpkglist.nim" % [destPath / "babelpkglist.js"]) proc main(c: var TConfigData) = const cmd = "nim rst2html --compileonly $1 -o:web/$2.temp web/$2.txt" if c.ticker.len > 0: try: c.ticker = readFile("web" / c.ticker) except IOError: quit("[Error] cannot open: " & c.ticker) for i in 0..c.tabs.len-1: var file = c.tabs[i].val let rss = if file in ["news", "index"]: extractFilename(rssUrl) else: "" exec(cmd % [c.nimArgs, file]) var temp = "web" / changeFileExt(file, "temp") var content: string try: content = readFile(temp) except IOError: quit("[Error] cannot open: " & temp) var f: File var outfile = "web/upload/$#.html" % file if not existsDir("web/upload"): createDir("web/upload") if open(f, outfile, fmWrite): writeln(f, generateHTMLPage(c, file, content, rss)) close(f) else: quit("[Error] cannot write file: " & outfile) removeFile(temp) copyDir("web/assets", "web/upload/assets") buildJS("web/upload") buildNewsRss(c, "web/upload") buildAddDoc(c, "web/upload") buildDocSamples(c, "web/upload") buildDoc(c, "web/upload") buildDocSamples(c, "doc") buildDoc(c, "doc") buildPdfDoc(c, "doc") var c: TConfigData initConfigData(c) parseCmdLine(c) parseIniFile(c) when false: buildPdfDoc(c, "doc") else: main(c)