kochdocs.nim 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. ## Part of 'koch' responsible for the documentation generation.
  2. import os, strutils, osproc, sets, pathnorm
  3. const
  4. gaCode* = " --doc.googleAnalytics:UA-48159761-1"
  5. # --warning[LockLevel]:off pending #13218
  6. nimArgs = "--warning[LockLevel]:off --hint[Conf]:off --hint[Path]:off --hint[Processing]:off -d:boot --putenv:nimversion=$#" % system.NimVersion
  7. gitUrl = "https://github.com/nim-lang/Nim"
  8. docHtmlOutput = "doc/html"
  9. webUploadOutput = "web/upload"
  10. docHackDir = "tools/dochack"
  11. var nimExe*: string
  12. proc exe*(f: string): string =
  13. result = addFileExt(f, ExeExt)
  14. when defined(windows):
  15. result = result.replace('/','\\')
  16. proc findNimImpl*(): tuple[path: string, ok: bool] =
  17. if nimExe.len > 0: return (nimExe, true)
  18. let nim = "nim".exe
  19. result.path = "bin" / nim
  20. result.ok = true
  21. if existsFile(result.path): return
  22. for dir in split(getEnv("PATH"), PathSep):
  23. result.path = dir / nim
  24. if existsFile(result.path): return
  25. # assume there is a symlink to the exe or something:
  26. return (nim, false)
  27. proc findNim*(): string = findNimImpl().path
  28. proc exec*(cmd: string, errorcode: int = QuitFailure, additionalPath = "") =
  29. let prevPath = getEnv("PATH")
  30. if additionalPath.len > 0:
  31. var absolute = additionalPath
  32. if not absolute.isAbsolute:
  33. absolute = getCurrentDir() / absolute
  34. echo("Adding to $PATH: ", absolute)
  35. putEnv("PATH", (if prevPath.len > 0: prevPath & PathSep else: "") & absolute)
  36. echo(cmd)
  37. if execShellCmd(cmd) != 0: quit("FAILURE", errorcode)
  38. putEnv("PATH", prevPath)
  39. template inFold*(desc, body) =
  40. if existsEnv("TRAVIS"):
  41. echo "travis_fold:start:" & desc.replace(" ", "_")
  42. body
  43. if existsEnv("TRAVIS"):
  44. echo "travis_fold:end:" & desc.replace(" ", "_")
  45. proc execFold*(desc, cmd: string, errorcode: int = QuitFailure, additionalPath = "") =
  46. ## Execute shell command. Add log folding on Travis CI.
  47. # https://github.com/travis-ci/travis-ci/issues/2285#issuecomment-42724719
  48. inFold(desc):
  49. exec(cmd, errorcode, additionalPath)
  50. proc execCleanPath*(cmd: string,
  51. additionalPath = ""; errorcode: int = QuitFailure) =
  52. # simulate a poor man's virtual environment
  53. let prevPath = getEnv("PATH")
  54. when defined(windows):
  55. let cleanPath = r"$1\system32;$1;$1\System32\Wbem" % getEnv"SYSTEMROOT"
  56. else:
  57. const cleanPath = r"/usr/local/bin:/usr/bin:/bin:/usr/sbin:/sbin:/opt/X11/bin"
  58. putEnv("PATH", cleanPath & PathSep & additionalPath)
  59. echo(cmd)
  60. if execShellCmd(cmd) != 0: quit("FAILURE", errorcode)
  61. putEnv("PATH", prevPath)
  62. proc nimexec*(cmd: string) =
  63. # Consider using `nimCompile` instead
  64. exec findNim().quoteShell() & " " & cmd
  65. proc nimCompile*(input: string, outputDir = "bin", mode = "c", options = "") =
  66. let output = outputDir / input.splitFile.name.exe
  67. let cmd = findNim().quoteShell() & " " & mode & " -o:" & output & " " & options & " " & input
  68. exec cmd
  69. proc nimCompileFold*(desc, input: string, outputDir = "bin", mode = "c", options = "") =
  70. let output = outputDir / input.splitFile.name.exe
  71. let cmd = findNim().quoteShell() & " " & mode & " -o:" & output & " " & options & " " & input
  72. execFold(desc, cmd)
  73. const
  74. pdf = """
  75. doc/manual.rst
  76. doc/lib.rst
  77. doc/tut1.rst
  78. doc/tut2.rst
  79. doc/tut3.rst
  80. doc/nimc.rst
  81. doc/niminst.rst
  82. doc/gc.rst
  83. """.splitWhitespace()
  84. rst2html = """
  85. doc/intern.rst
  86. doc/apis.rst
  87. doc/lib.rst
  88. doc/manual.rst
  89. doc/manual_experimental.rst
  90. doc/destructors.rst
  91. doc/tut1.rst
  92. doc/tut2.rst
  93. doc/tut3.rst
  94. doc/nimc.rst
  95. doc/hcr.rst
  96. doc/drnim.rst
  97. doc/overview.rst
  98. doc/filters.rst
  99. doc/tools.rst
  100. doc/niminst.rst
  101. doc/nimgrep.rst
  102. doc/gc.rst
  103. doc/estp.rst
  104. doc/idetools.rst
  105. doc/docgen.rst
  106. doc/koch.rst
  107. doc/backends.rst
  108. doc/nimsuggest.rst
  109. doc/nep1.rst
  110. doc/nims.rst
  111. doc/contributing.rst
  112. doc/codeowners.rst
  113. doc/packaging.rst
  114. doc/manual/var_t_return.rst
  115. """.splitWhitespace()
  116. doc0 = """
  117. lib/system/threads.nim
  118. lib/system/channels.nim
  119. """.splitWhitespace() # ran by `nim doc0` instead of `nim doc`
  120. withoutIndex = """
  121. lib/wrappers/mysql.nim
  122. lib/wrappers/iup.nim
  123. lib/wrappers/sqlite3.nim
  124. lib/wrappers/postgres.nim
  125. lib/wrappers/tinyc.nim
  126. lib/wrappers/odbcsql.nim
  127. lib/wrappers/pcre.nim
  128. lib/wrappers/openssl.nim
  129. lib/posix/posix.nim
  130. lib/posix/linux.nim
  131. lib/posix/termios.nim
  132. lib/js/jscore.nim
  133. """.splitWhitespace()
  134. # some of these are include files so shouldn't be docgen'd
  135. ignoredModules = """
  136. lib/prelude.nim
  137. lib/pure/future.nim
  138. lib/pure/collections/hashcommon.nim
  139. lib/pure/collections/tableimpl.nim
  140. lib/pure/collections/setimpl.nim
  141. lib/pure/ioselects/ioselectors_kqueue.nim
  142. lib/pure/ioselects/ioselectors_select.nim
  143. lib/pure/ioselects/ioselectors_poll.nim
  144. lib/pure/ioselects/ioselectors_epoll.nim
  145. lib/posix/posix_macos_amd64.nim
  146. lib/posix/posix_other.nim
  147. lib/posix/posix_nintendoswitch.nim
  148. lib/posix/posix_nintendoswitch_consts.nim
  149. lib/posix/posix_linux_amd64.nim
  150. lib/posix/posix_linux_amd64_consts.nim
  151. lib/posix/posix_other_consts.nim
  152. lib/posix/posix_openbsd_amd64.nim
  153. """.splitWhitespace()
  154. when (NimMajor, NimMinor) < (1, 1) or not declared(isRelativeTo):
  155. proc isRelativeTo(path, base: string): bool =
  156. let path = path.normalizedPath
  157. let base = base.normalizedPath
  158. let ret = relativePath(path, base)
  159. result = path.len > 0 and not ret.startsWith ".."
  160. proc getDocList(): seq[string] =
  161. var docIgnore: HashSet[string]
  162. for a in doc0: docIgnore.incl a
  163. for a in withoutIndex: docIgnore.incl a
  164. for a in ignoredModules: docIgnore.incl a
  165. # don't ignore these even though in lib/system
  166. const goodSystem = """
  167. lib/system/io.nim
  168. lib/system/nimscript.nim
  169. lib/system/assertions.nim
  170. lib/system/iterators.nim
  171. lib/system/dollars.nim
  172. lib/system/widestrs.nim
  173. """.splitWhitespace()
  174. for a in walkDirRec("lib"):
  175. if a.splitFile.ext != ".nim" or
  176. a.isRelativeTo("lib/pure/includes") or
  177. a.isRelativeTo("lib/genode") or
  178. a.isRelativeTo("lib/deprecated") or
  179. (a.isRelativeTo("lib/system") and a.replace('\\', '/') notin goodSystem) or
  180. a.replace('\\', '/') in docIgnore:
  181. continue
  182. result.add a
  183. result.add normalizePath("nimsuggest/sexp.nim")
  184. let doc = getDocList()
  185. proc sexec(cmds: openArray[string]) =
  186. ## Serial queue wrapper around exec.
  187. for cmd in cmds:
  188. echo(cmd)
  189. let (outp, exitCode) = osproc.execCmdEx(cmd)
  190. if exitCode != 0: quit outp
  191. proc mexec(cmds: openArray[string]) =
  192. ## Multiprocessor version of exec
  193. let r = execProcesses(cmds, {poStdErrToStdOut, poParentStreams, poEchoCmd})
  194. if r != 0:
  195. echo "external program failed, retrying serial work queue for logs!"
  196. sexec(cmds)
  197. proc buildDocSamples(nimArgs, destPath: string) =
  198. ## Special case documentation sample proc.
  199. ##
  200. ## TODO: consider integrating into the existing generic documentation builders
  201. ## now that we have a single `doc` command.
  202. exec(findNim().quoteShell() & " doc $# -o:$# $#" %
  203. [nimArgs, destPath / "docgen_sample.html", "doc" / "docgen_sample.nim"])
  204. proc buildDoc(nimArgs, destPath: string) =
  205. # call nim for the documentation:
  206. var
  207. commands = newSeq[string](rst2html.len + len(doc0) + len(doc) + withoutIndex.len)
  208. i = 0
  209. let nim = findNim().quoteShell()
  210. for d in items(rst2html):
  211. commands[i] = nim & " rst2html $# --git.url:$# -o:$# --index:on $#" %
  212. [nimArgs, gitUrl,
  213. destPath / changeFileExt(splitFile(d).name, "html"), d]
  214. i.inc
  215. for d in items(doc0):
  216. commands[i] = nim & " doc0 $# --git.url:$# -o:$# --index:on $#" %
  217. [nimArgs, gitUrl,
  218. destPath / changeFileExt(splitFile(d).name, "html"), d]
  219. i.inc
  220. for d in items(doc):
  221. var nimArgs2 = nimArgs
  222. if d.isRelativeTo("compiler"):
  223. nimArgs2.add " --docroot"
  224. commands[i] = nim & " doc $# --git.url:$# --outdir:$# --index:on $#" %
  225. [nimArgs2, gitUrl, destPath, d]
  226. i.inc
  227. for d in items(withoutIndex):
  228. commands[i] = nim & " doc2 $# --git.url:$# -o:$# $#" %
  229. [nimArgs, gitUrl,
  230. destPath / changeFileExt(splitFile(d).name, "html"), d]
  231. i.inc
  232. mexec(commands)
  233. exec(nim & " buildIndex -o:$1/theindex.html $1" % [destPath])
  234. proc buildPdfDoc*(nimArgs, destPath: string) =
  235. createDir(destPath)
  236. if os.execShellCmd("pdflatex -version") != 0:
  237. echo "pdflatex not found; no PDF documentation generated"
  238. else:
  239. const pdflatexcmd = "pdflatex -interaction=nonstopmode "
  240. for d in items(pdf):
  241. exec(findNim().quoteShell() & " rst2tex $# $#" % [nimArgs, d])
  242. # call LaTeX twice to get cross references right:
  243. exec(pdflatexcmd & changeFileExt(d, "tex"))
  244. exec(pdflatexcmd & changeFileExt(d, "tex"))
  245. # delete all the crappy temporary files:
  246. let pdf = splitFile(d).name & ".pdf"
  247. let dest = destPath / pdf
  248. removeFile(dest)
  249. moveFile(dest=dest, source=pdf)
  250. removeFile(changeFileExt(pdf, "aux"))
  251. if existsFile(changeFileExt(pdf, "toc")):
  252. removeFile(changeFileExt(pdf, "toc"))
  253. removeFile(changeFileExt(pdf, "log"))
  254. removeFile(changeFileExt(pdf, "out"))
  255. removeFile(changeFileExt(d, "tex"))
  256. proc buildJS() =
  257. exec(findNim().quoteShell() & " js -d:release --out:$1 tools/nimblepkglist.nim" %
  258. [webUploadOutput / "nimblepkglist.js"])
  259. exec(findNim().quoteShell() & " js " & (docHackDir / "dochack.nim"))
  260. proc buildDocs*(args: string) =
  261. const
  262. docHackJs = "dochack.js"
  263. let
  264. a = nimArgs & " " & args
  265. docHackJsSource = docHackDir / docHackJs
  266. docHackJsDest = docHtmlOutput / docHackJs
  267. buildJS() # This call generates docHackJsSource
  268. let docup = webUploadOutput / NimVersion
  269. createDir(docup)
  270. buildDocSamples(a, docup)
  271. buildDoc(a, docup)
  272. # 'nimArgs' instead of 'a' is correct here because we don't want
  273. # that the offline docs contain the 'gaCode'!
  274. createDir(docHtmlOutput)
  275. buildDocSamples(nimArgs, docHtmlOutput)
  276. buildDoc(nimArgs, docHtmlOutput)
  277. copyFile(docHackJsSource, docHackJsDest)
  278. copyFile(docHackJsSource, docup / docHackJs)