mirror of
https://github.com/nim-lang/Nim.git
synced 2026-01-04 04:02:41 +00:00
* docgen: implement cross-document links Fully implements https://github.com/nim-lang/RFCs/issues/125 Follow-up of: https://github.com/nim-lang/Nim/pull/18642 (for internal links) and https://github.com/nim-lang/Nim/issues/20127. Overview -------- Explicit import-like directive is required, called `.. importdoc::`. (the syntax is % RST, Markdown will use it for a while). Then one can reference any symbols/headings/anchors, as if they were in the local file (but they will be prefixed with a module name or markup document in link text). It's possible to reference anything from anywhere (any direction in `.nim`/`.md`/`.rst` files). See `doc/docgen.md` for full description. Working is based on `.idx` files, hence one needs to generate all `.idx` beforehand. A dedicated option `--index:only` is introduced (and a separate stage for `--index:only` is added to `kochdocs.nim`). Performance note ---------------- Full run for `./koch docs` now takes 185% of the time before this PR. (After: 315 s, before: 170 s on my PC). All the time seems to be spent on `--index:only` run, which takes almost as much (85%) of normal doc run -- it seems that most time is spent on file parsing, turning off HTML generation phase has not helped much. (One could avoid it by specifying list of files that can be referenced and pre-processing only them. But it can become error-prone and I assume that these linke will be **everywhere** in the repository anyway, especially considering https://github.com/nim-lang/RFCs/issues/478. So every `.nim`/`.md` file is processed for `.idx` first). But that's all without significant part of repository converted to cross-module auto links. To estimate impact I checked the time for `doc`ing a few files (after all indexes have been generated), and everywhere difference was **negligible**. E.g. for `lib/std/private/osfiles.nim` that `importdoc`s large `os.idx` and hence should have been a case with relatively large performance impact, but: * After: 0.59 s. * Before: 0.59 s. So Nim compiler works so slow that doc part basically does not matter :-) Testing ------- 1) added `extlinks` test to `nimdoc/` 2) checked that `theindex.html` is still correct 2) fixed broken auto-links for modules that were derived from `os.nim` by adding appropriate ``importdoc`` Implementation note ------------------- Parsing and formating of `.idx` entries is moved into a dedicated `rstidx.nim` module from `rstgen.nim`. `.idx` file format changed: * fields are not escaped in most cases because we need original strings for referencing, not HTML ones (the exception is linkTitle for titles and headings). Escaping happens later -- on the stage of `rstgen` buildIndex, etc. * all lines have fixed number of columns 6 * added discriminator tag as a first column, it always allows distinguish Nim/markup entries, titles/headings, etc. `rstgen` does not rely any more (in most cases) on ad-hoc logic to determine what type each entry is. * there is now always a title entry added at the first line. * add a line number as 6th column * linkTitle (4th) column has a different format: before it was like `module: funcName()`, now it's `proc funcName()`. (This format is also propagated to `theindex.html` and search results, I kept it that way since I like it more though it's discussible.) This column is what used for Nim symbols resolution. * also changed details on column format for headings and titles: "keyword" is original, "linkTitle" is HTML one * fix paths on Windows + more clear code * Update compiler/docgen.nim Co-authored-by: Andreas Rumpf <rumpf_a@web.de> * Handle .md and .nim paths uniformly in findRefFile * handle titles better + more comments * don't allow markup overwrite index title for .nim files Co-authored-by: Andreas Rumpf <rumpf_a@web.de>
425 lines
15 KiB
Nim
425 lines
15 KiB
Nim
include system/inclrtl
|
|
import std/private/since
|
|
import std/oserrors
|
|
|
|
import oscommon
|
|
export fileExists
|
|
|
|
import ospaths2, ossymlinks
|
|
|
|
## .. importdoc:: osdirs.nim, os.nim
|
|
|
|
when defined(nimPreviewSlimSystem):
|
|
import std/[syncio, assertions, widestrs]
|
|
|
|
when weirdTarget:
|
|
discard
|
|
elif defined(windows):
|
|
import winlean
|
|
elif defined(posix):
|
|
import posix, times
|
|
|
|
proc toTime(ts: Timespec): times.Time {.inline.} =
|
|
result = initTime(ts.tv_sec.int64, ts.tv_nsec.int)
|
|
else:
|
|
{.error: "OS module not ported to your operating system!".}
|
|
|
|
|
|
when weirdTarget:
|
|
{.pragma: noWeirdTarget, error: "this proc is not available on the NimScript/js target".}
|
|
else:
|
|
{.pragma: noWeirdTarget.}
|
|
|
|
|
|
when defined(nimscript):
|
|
# for procs already defined in scriptconfig.nim
|
|
template noNimJs(body): untyped = discard
|
|
elif defined(js):
|
|
{.pragma: noNimJs, error: "this proc is not available on the js target".}
|
|
else:
|
|
{.pragma: noNimJs.}
|
|
|
|
|
|
type
|
|
FilePermission* = enum ## File access permission, modelled after UNIX.
|
|
##
|
|
## See also:
|
|
## * `getFilePermissions`_
|
|
## * `setFilePermissions`_
|
|
## * `FileInfo object`_
|
|
fpUserExec, ## execute access for the file owner
|
|
fpUserWrite, ## write access for the file owner
|
|
fpUserRead, ## read access for the file owner
|
|
fpGroupExec, ## execute access for the group
|
|
fpGroupWrite, ## write access for the group
|
|
fpGroupRead, ## read access for the group
|
|
fpOthersExec, ## execute access for others
|
|
fpOthersWrite, ## write access for others
|
|
fpOthersRead ## read access for others
|
|
|
|
proc getFilePermissions*(filename: string): set[FilePermission] {.
|
|
rtl, extern: "nos$1", tags: [ReadDirEffect], noWeirdTarget.} =
|
|
## Retrieves file permissions for `filename`.
|
|
##
|
|
## `OSError` is raised in case of an error.
|
|
## On Windows, only the ``readonly`` flag is checked, every other
|
|
## permission is available in any case.
|
|
##
|
|
## See also:
|
|
## * `setFilePermissions proc`_
|
|
## * `FilePermission enum`_
|
|
when defined(posix):
|
|
var a: Stat
|
|
if stat(filename, a) < 0'i32: raiseOSError(osLastError(), filename)
|
|
result = {}
|
|
if (a.st_mode and S_IRUSR.Mode) != 0.Mode: result.incl(fpUserRead)
|
|
if (a.st_mode and S_IWUSR.Mode) != 0.Mode: result.incl(fpUserWrite)
|
|
if (a.st_mode and S_IXUSR.Mode) != 0.Mode: result.incl(fpUserExec)
|
|
|
|
if (a.st_mode and S_IRGRP.Mode) != 0.Mode: result.incl(fpGroupRead)
|
|
if (a.st_mode and S_IWGRP.Mode) != 0.Mode: result.incl(fpGroupWrite)
|
|
if (a.st_mode and S_IXGRP.Mode) != 0.Mode: result.incl(fpGroupExec)
|
|
|
|
if (a.st_mode and S_IROTH.Mode) != 0.Mode: result.incl(fpOthersRead)
|
|
if (a.st_mode and S_IWOTH.Mode) != 0.Mode: result.incl(fpOthersWrite)
|
|
if (a.st_mode and S_IXOTH.Mode) != 0.Mode: result.incl(fpOthersExec)
|
|
else:
|
|
when useWinUnicode:
|
|
wrapUnary(res, getFileAttributesW, filename)
|
|
else:
|
|
var res = getFileAttributesA(filename)
|
|
if res == -1'i32: raiseOSError(osLastError(), filename)
|
|
if (res and FILE_ATTRIBUTE_READONLY) != 0'i32:
|
|
result = {fpUserExec, fpUserRead, fpGroupExec, fpGroupRead,
|
|
fpOthersExec, fpOthersRead}
|
|
else:
|
|
result = {fpUserExec..fpOthersRead}
|
|
|
|
proc setFilePermissions*(filename: string, permissions: set[FilePermission],
|
|
followSymlinks = true)
|
|
{.rtl, extern: "nos$1", tags: [ReadDirEffect, WriteDirEffect],
|
|
noWeirdTarget.} =
|
|
## Sets the file permissions for `filename`.
|
|
##
|
|
## If `followSymlinks` set to true (default) and ``filename`` points to a
|
|
## symlink, permissions are set to the file symlink points to.
|
|
## `followSymlinks` set to false is a noop on Windows and some POSIX
|
|
## systems (including Linux) on which `lchmod` is either unavailable or always
|
|
## fails, given that symlinks permissions there are not observed.
|
|
##
|
|
## `OSError` is raised in case of an error.
|
|
## On Windows, only the ``readonly`` flag is changed, depending on
|
|
## ``fpUserWrite`` permission.
|
|
##
|
|
## See also:
|
|
## * `getFilePermissions proc`_
|
|
## * `FilePermission enum`_
|
|
when defined(posix):
|
|
var p = 0.Mode
|
|
if fpUserRead in permissions: p = p or S_IRUSR.Mode
|
|
if fpUserWrite in permissions: p = p or S_IWUSR.Mode
|
|
if fpUserExec in permissions: p = p or S_IXUSR.Mode
|
|
|
|
if fpGroupRead in permissions: p = p or S_IRGRP.Mode
|
|
if fpGroupWrite in permissions: p = p or S_IWGRP.Mode
|
|
if fpGroupExec in permissions: p = p or S_IXGRP.Mode
|
|
|
|
if fpOthersRead in permissions: p = p or S_IROTH.Mode
|
|
if fpOthersWrite in permissions: p = p or S_IWOTH.Mode
|
|
if fpOthersExec in permissions: p = p or S_IXOTH.Mode
|
|
|
|
if not followSymlinks and filename.symlinkExists:
|
|
when declared(lchmod):
|
|
if lchmod(filename, cast[Mode](p)) != 0:
|
|
raiseOSError(osLastError(), $(filename, permissions))
|
|
else:
|
|
if chmod(filename, cast[Mode](p)) != 0:
|
|
raiseOSError(osLastError(), $(filename, permissions))
|
|
else:
|
|
when useWinUnicode:
|
|
wrapUnary(res, getFileAttributesW, filename)
|
|
else:
|
|
var res = getFileAttributesA(filename)
|
|
if res == -1'i32: raiseOSError(osLastError(), filename)
|
|
if fpUserWrite in permissions:
|
|
res = res and not FILE_ATTRIBUTE_READONLY
|
|
else:
|
|
res = res or FILE_ATTRIBUTE_READONLY
|
|
when useWinUnicode:
|
|
wrapBinary(res2, setFileAttributesW, filename, res)
|
|
else:
|
|
var res2 = setFileAttributesA(filename, res)
|
|
if res2 == - 1'i32: raiseOSError(osLastError(), $(filename, permissions))
|
|
|
|
|
|
const hasCCopyfile = defined(osx) and not defined(nimLegacyCopyFile)
|
|
# xxx instead of `nimLegacyCopyFile`, support something like: `when osxVersion >= (10, 5)`
|
|
|
|
when hasCCopyfile:
|
|
# `copyfile` API available since osx 10.5.
|
|
{.push nodecl, header: "<copyfile.h>".}
|
|
type
|
|
copyfile_state_t {.nodecl.} = pointer
|
|
copyfile_flags_t = cint
|
|
proc copyfile_state_alloc(): copyfile_state_t
|
|
proc copyfile_state_free(state: copyfile_state_t): cint
|
|
proc c_copyfile(src, dst: cstring, state: copyfile_state_t, flags: copyfile_flags_t): cint {.importc: "copyfile".}
|
|
# replace with `let` pending bootstrap >= 1.4.0
|
|
var
|
|
COPYFILE_DATA {.nodecl.}: copyfile_flags_t
|
|
COPYFILE_XATTR {.nodecl.}: copyfile_flags_t
|
|
{.pop.}
|
|
|
|
type
|
|
CopyFlag* = enum ## Copy options.
|
|
cfSymlinkAsIs, ## Copy symlinks as symlinks
|
|
cfSymlinkFollow, ## Copy the files symlinks point to
|
|
cfSymlinkIgnore ## Ignore symlinks
|
|
|
|
const copyFlagSymlink = {cfSymlinkAsIs, cfSymlinkFollow, cfSymlinkIgnore}
|
|
|
|
proc copyFile*(source, dest: string, options = {cfSymlinkFollow}) {.rtl,
|
|
extern: "nos$1", tags: [ReadDirEffect, ReadIOEffect, WriteIOEffect],
|
|
noWeirdTarget.} =
|
|
## Copies a file from `source` to `dest`, where `dest.parentDir` must exist.
|
|
##
|
|
## On non-Windows OSes, `options` specify the way file is copied; by default,
|
|
## if `source` is a symlink, copies the file symlink points to. `options` is
|
|
## ignored on Windows: symlinks are skipped.
|
|
##
|
|
## If this fails, `OSError` is raised.
|
|
##
|
|
## On the Windows platform this proc will
|
|
## copy the source file's attributes into dest.
|
|
##
|
|
## On other platforms you need
|
|
## to use `getFilePermissions`_ and
|
|
## `setFilePermissions`_
|
|
## procs
|
|
## to copy them by hand (or use the convenience `copyFileWithPermissions
|
|
## proc`_),
|
|
## otherwise `dest` will inherit the default permissions of a newly
|
|
## created file for the user.
|
|
##
|
|
## If `dest` already exists, the file attributes
|
|
## will be preserved and the content overwritten.
|
|
##
|
|
## On OSX, `copyfile` C api will be used (available since OSX 10.5) unless
|
|
## `-d:nimLegacyCopyFile` is used.
|
|
##
|
|
## See also:
|
|
## * `CopyFlag enum`_
|
|
## * `copyDir proc`_
|
|
## * `copyFileWithPermissions proc`_
|
|
## * `tryRemoveFile proc`_
|
|
## * `removeFile proc`_
|
|
## * `moveFile proc`_
|
|
|
|
doAssert card(copyFlagSymlink * options) == 1, "There should be exactly " &
|
|
"one cfSymlink* in options"
|
|
let isSymlink = source.symlinkExists
|
|
if isSymlink and (cfSymlinkIgnore in options or defined(windows)):
|
|
return
|
|
when defined(windows):
|
|
when useWinUnicode:
|
|
let s = newWideCString(source)
|
|
let d = newWideCString(dest)
|
|
if copyFileW(s, d, 0'i32) == 0'i32:
|
|
raiseOSError(osLastError(), $(source, dest))
|
|
else:
|
|
if copyFileA(source, dest, 0'i32) == 0'i32:
|
|
raiseOSError(osLastError(), $(source, dest))
|
|
else:
|
|
if isSymlink and cfSymlinkAsIs in options:
|
|
createSymlink(expandSymlink(source), dest)
|
|
else:
|
|
when hasCCopyfile:
|
|
let state = copyfile_state_alloc()
|
|
# xxx `COPYFILE_STAT` could be used for one-shot
|
|
# `copyFileWithPermissions`.
|
|
let status = c_copyfile(source.cstring, dest.cstring, state,
|
|
COPYFILE_DATA)
|
|
if status != 0:
|
|
let err = osLastError()
|
|
discard copyfile_state_free(state)
|
|
raiseOSError(err, $(source, dest))
|
|
let status2 = copyfile_state_free(state)
|
|
if status2 != 0: raiseOSError(osLastError(), $(source, dest))
|
|
else:
|
|
# generic version of copyFile which works for any platform:
|
|
const bufSize = 8000 # better for memory manager
|
|
var d, s: File
|
|
if not open(s, source):raiseOSError(osLastError(), source)
|
|
if not open(d, dest, fmWrite):
|
|
close(s)
|
|
raiseOSError(osLastError(), dest)
|
|
var buf = alloc(bufSize)
|
|
while true:
|
|
var bytesread = readBuffer(s, buf, bufSize)
|
|
if bytesread > 0:
|
|
var byteswritten = writeBuffer(d, buf, bytesread)
|
|
if bytesread != byteswritten:
|
|
dealloc(buf)
|
|
close(s)
|
|
close(d)
|
|
raiseOSError(osLastError(), dest)
|
|
if bytesread != bufSize: break
|
|
dealloc(buf)
|
|
close(s)
|
|
flushFile(d)
|
|
close(d)
|
|
|
|
proc copyFileToDir*(source, dir: string, options = {cfSymlinkFollow})
|
|
{.noWeirdTarget, since: (1,3,7).} =
|
|
## Copies a file `source` into directory `dir`, which must exist.
|
|
##
|
|
## On non-Windows OSes, `options` specify the way file is copied; by default,
|
|
## if `source` is a symlink, copies the file symlink points to. `options` is
|
|
## ignored on Windows: symlinks are skipped.
|
|
##
|
|
## See also:
|
|
## * `CopyFlag enum`_
|
|
## * `copyFile proc`_
|
|
if dir.len == 0: # treating "" as "." is error prone
|
|
raise newException(ValueError, "dest is empty")
|
|
copyFile(source, dir / source.lastPathPart, options)
|
|
|
|
|
|
proc copyFileWithPermissions*(source, dest: string,
|
|
ignorePermissionErrors = true,
|
|
options = {cfSymlinkFollow}) {.noWeirdTarget.} =
|
|
## Copies a file from `source` to `dest` preserving file permissions.
|
|
##
|
|
## On non-Windows OSes, `options` specify the way file is copied; by default,
|
|
## if `source` is a symlink, copies the file symlink points to. `options` is
|
|
## ignored on Windows: symlinks are skipped.
|
|
##
|
|
## This is a wrapper proc around `copyFile`_,
|
|
## `getFilePermissions`_ and `setFilePermissions`_
|
|
## procs on non-Windows platforms.
|
|
##
|
|
## On Windows this proc is just a wrapper for `copyFile proc`_ since
|
|
## that proc already copies attributes.
|
|
##
|
|
## On non-Windows systems permissions are copied after the file itself has
|
|
## been copied, which won't happen atomically and could lead to a race
|
|
## condition. If `ignorePermissionErrors` is true (default), errors while
|
|
## reading/setting file attributes will be ignored, otherwise will raise
|
|
## `OSError`.
|
|
##
|
|
## See also:
|
|
## * `CopyFlag enum`_
|
|
## * `copyFile proc`_
|
|
## * `copyDir proc`_
|
|
## * `tryRemoveFile proc`_
|
|
## * `removeFile proc`_
|
|
## * `moveFile proc`_
|
|
## * `copyDirWithPermissions proc`_
|
|
copyFile(source, dest, options)
|
|
when not defined(windows):
|
|
try:
|
|
setFilePermissions(dest, getFilePermissions(source), followSymlinks =
|
|
(cfSymlinkFollow in options))
|
|
except:
|
|
if not ignorePermissionErrors:
|
|
raise
|
|
|
|
when not declared(ENOENT) and not defined(windows):
|
|
when defined(nimscript):
|
|
when not defined(haiku):
|
|
const ENOENT = cint(2) # 2 on most systems including Solaris
|
|
else:
|
|
const ENOENT = cint(-2147459069)
|
|
else:
|
|
var ENOENT {.importc, header: "<errno.h>".}: cint
|
|
|
|
when defined(windows) and not weirdTarget:
|
|
when useWinUnicode:
|
|
template deleteFile(file: untyped): untyped = deleteFileW(file)
|
|
template setFileAttributes(file, attrs: untyped): untyped =
|
|
setFileAttributesW(file, attrs)
|
|
else:
|
|
template deleteFile(file: untyped): untyped = deleteFileA(file)
|
|
template setFileAttributes(file, attrs: untyped): untyped =
|
|
setFileAttributesA(file, attrs)
|
|
|
|
proc tryRemoveFile*(file: string): bool {.rtl, extern: "nos$1", tags: [WriteDirEffect], noWeirdTarget.} =
|
|
## Removes the `file`.
|
|
##
|
|
## If this fails, returns `false`. This does not fail
|
|
## if the file never existed in the first place.
|
|
##
|
|
## On Windows, ignores the read-only attribute.
|
|
##
|
|
## See also:
|
|
## * `copyFile proc`_
|
|
## * `copyFileWithPermissions proc`_
|
|
## * `removeFile proc`_
|
|
## * `moveFile proc`_
|
|
result = true
|
|
when defined(windows):
|
|
when useWinUnicode:
|
|
let f = newWideCString(file)
|
|
else:
|
|
let f = file
|
|
if deleteFile(f) == 0:
|
|
result = false
|
|
let err = getLastError()
|
|
if err == ERROR_FILE_NOT_FOUND or err == ERROR_PATH_NOT_FOUND:
|
|
result = true
|
|
elif err == ERROR_ACCESS_DENIED and
|
|
setFileAttributes(f, FILE_ATTRIBUTE_NORMAL) != 0 and
|
|
deleteFile(f) != 0:
|
|
result = true
|
|
else:
|
|
if unlink(file) != 0'i32 and errno != ENOENT:
|
|
result = false
|
|
|
|
proc removeFile*(file: string) {.rtl, extern: "nos$1", tags: [WriteDirEffect], noWeirdTarget.} =
|
|
## Removes the `file`.
|
|
##
|
|
## If this fails, `OSError` is raised. This does not fail
|
|
## if the file never existed in the first place.
|
|
##
|
|
## On Windows, ignores the read-only attribute.
|
|
##
|
|
## See also:
|
|
## * `removeDir proc`_
|
|
## * `copyFile proc`_
|
|
## * `copyFileWithPermissions proc`_
|
|
## * `tryRemoveFile proc`_
|
|
## * `moveFile proc`_
|
|
if not tryRemoveFile(file):
|
|
raiseOSError(osLastError(), file)
|
|
|
|
proc moveFile*(source, dest: string) {.rtl, extern: "nos$1",
|
|
tags: [ReadDirEffect, ReadIOEffect, WriteIOEffect], noWeirdTarget.} =
|
|
## Moves a file from `source` to `dest`.
|
|
##
|
|
## Symlinks are not followed: if `source` is a symlink, it is itself moved,
|
|
## not its target.
|
|
##
|
|
## If this fails, `OSError` is raised.
|
|
## If `dest` already exists, it will be overwritten.
|
|
##
|
|
## Can be used to `rename files`:idx:.
|
|
##
|
|
## See also:
|
|
## * `moveDir proc`_
|
|
## * `copyFile proc`_
|
|
## * `copyFileWithPermissions proc`_
|
|
## * `removeFile proc`_
|
|
## * `tryRemoveFile proc`_
|
|
|
|
if not tryMoveFSObject(source, dest, isDir = false):
|
|
when defined(windows):
|
|
doAssert false
|
|
else:
|
|
# Fallback to copy & del
|
|
copyFile(source, dest, {cfSymlinkAsIs})
|
|
try:
|
|
removeFile(source)
|
|
except:
|
|
discard tryRemoveFile(dest)
|
|
raise
|