123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556 |
- #
- #
- # Maintenance program for Nim
- # (c) Copyright 2017 Andreas Rumpf
- #
- # See the file "copying.txt", included in this
- # distribution, for details about the copyright.
- #
- # See doc/koch.txt for documentation.
- #
- when defined(gcc) and defined(windows):
- when defined(x86):
- {.link: "icons/koch.res".}
- else:
- {.link: "icons/koch_icon.o".}
- when defined(amd64) and defined(windows) and defined(vcc):
- {.link: "icons/koch-amd64-windows-vcc.res".}
- when defined(i386) and defined(windows) and defined(vcc):
- {.link: "icons/koch-i386-windows-vcc.res".}
- import
- os, strutils, parseopt, osproc, streams
- import tools / kochdocs
- const VersionAsString = system.NimVersion
- const
- HelpText = """
- +-----------------------------------------------------------------+
- | Maintenance program for Nim |
- | Version $1|
- | (c) 2017 Andreas Rumpf |
- +-----------------------------------------------------------------+
- Build time: $2, $3
- Usage:
- koch [options] command [options for command]
- Options:
- --help, -h shows this help and quits
- --latest bundle the installers with a bleeding edge Nimble
- --stable bundle the installers with a stable Nimble
- Possible Commands:
- boot [options] bootstraps with given command line options
- distrohelper [bindir] helper for distro packagers
- tools builds Nim related tools
- nimble builds the Nimble tool
- Boot options:
- -d:release produce a release version of the compiler
- -d:useLinenoise use the linenoise library for interactive mode
- (not needed on Windows)
- -d:leanCompiler produce a compiler without JS codegen or
- documentation generator in order to use less RAM
- for bootstrapping
- Commands for core developers:
- docs [options] generates the full documentation
- csource -d:release builds the C sources for installation
- pdf builds the PDF documentation
- zip builds the installation zip package
- xz builds the installation tar.xz package
- testinstall test tar.xz package; Unix only!
- tests [options] run the testsuite (run a subset of tests by
- specifying a category, e.g. `tests cat async`)
- temp options creates a temporary compiler for testing
- pushcsource push generated C sources to its repo
- Web options:
- --googleAnalytics:UA-... add the given google analytics code to the docs. To
- build the official docs, use UA-48159761-1
- """
- template withDir(dir, body) =
- let old = getCurrentDir()
- try:
- setCurrentDir(dir)
- body
- finally:
- setCurrentdir(old)
- let origDir = getCurrentDir()
- setCurrentDir(getAppDir())
- proc tryExec(cmd: string): bool =
- echo(cmd)
- result = execShellCmd(cmd) == 0
- proc safeRemove(filename: string) =
- if existsFile(filename): removeFile(filename)
- proc overwriteFile(source, dest: string) =
- safeRemove(dest)
- moveFile(source, dest)
- proc copyExe(source, dest: string) =
- safeRemove(dest)
- copyFile(dest=dest, source=source)
- inclFilePermissions(dest, {fpUserExec})
- const
- compileNimInst = "tools/niminst/niminst"
- proc csource(args: string) =
- nimexec(("cc $1 -r $3 --var:version=$2 --var:mingw=none csource " &
- "--main:compiler/nim.nim compiler/installer.ini $1") %
- [args, VersionAsString, compileNimInst])
- proc bundleNimbleSrc(latest: bool) =
- ## bunldeNimbleSrc() bundles a specific Nimble commit with the tarball. We
- ## always bundle the latest official release.
- if not dirExists("dist/nimble/.git"):
- exec("git clone https://github.com/nim-lang/nimble.git dist/nimble")
- if not latest:
- withDir("dist/nimble"):
- exec("git checkout -f stable")
- exec("git pull")
- proc bundleNimbleExe(latest: bool) =
- bundleNimbleSrc(latest)
- # installer.ini expects it under $nim/bin
- nimCompile("dist/nimble/src/nimble.nim", options = "-d:release --nilseqs:on")
- proc buildNimfind() =
- nimCompile("tools/nimfind.nim", options = "-d:release")
- proc buildNimble(latest: bool) =
- # old installations created nim/nimblepkg/*.nim files. We remove these
- # here so that it cannot cause problems (nimble bug #306):
- if dirExists("bin/nimblepkg"):
- removeDir("bin/nimblepkg")
- # if koch is used for a tar.xz, build the dist/nimble we shipped
- # with the tarball:
- var installDir = "dist/nimble"
- if not latest and dirExists(installDir) and not dirExists("dist/nimble/.git"):
- discard "don't do the git dance"
- else:
- if not dirExists("dist/nimble/.git"):
- if dirExists(installDir):
- var id = 0
- while dirExists("dist/nimble" & $id):
- inc id
- installDir = "dist/nimble" & $id
- exec("git clone https://github.com/nim-lang/nimble.git " & installDir)
- withDir(installDir):
- if latest:
- exec("git checkout -f master")
- else:
- exec("git checkout -f stable")
- exec("git pull")
- nimCompile(installDir / "src/nimble.nim", options = "--noNimblePath --nilseqs:on -d:release")
- proc bundleNimsuggest() =
- nimCompile("nimsuggest/nimsuggest.nim", options = "-d:release")
- proc buildVccTool() =
- nimCompile("tools/vccexe/vccexe.nim")
- proc bundleWinTools() =
- # TODO: consider building under `bin` instead of `.`
- nimCompile("tools/finish.nim", outputDir = "")
- buildVccTool()
- nimCompile("tools/nimgrab.nim", options = "-d:ssl")
- nimCompile("tools/nimgrep.nim")
- when false:
- # not yet a tool worth including
- nimCompile(r"tools\downloader.nim", options = r"--cc:vcc --app:gui -d:ssl --noNimblePath --path:..\ui")
- proc zip(latest: bool; args: string) =
- bundleNimbleExe(latest)
- bundleNimsuggest()
- bundleWinTools()
- nimexec("cc -r $2 --var:version=$1 --var:mingw=none --main:compiler/nim.nim scripts compiler/installer.ini" %
- [VersionAsString, compileNimInst])
- exec("$# --var:version=$# --var:mingw=none --main:compiler/nim.nim zip compiler/installer.ini" %
- ["tools/niminst/niminst".exe, VersionAsString])
- proc ensureCleanGit() =
- let (outp, status) = osproc.execCmdEx("git diff")
- if outp.len != 0:
- quit "Not a clean git repository; 'git diff' not empty!"
- if status != 0:
- quit "Not a clean git repository; 'git diff' returned non-zero!"
- proc xz(latest: bool; args: string) =
- ensureCleanGit()
- bundleNimbleSrc(latest)
- when false:
- bundleNimsuggest()
- nimexec("cc -r $2 --var:version=$1 --var:mingw=none --main:compiler/nim.nim scripts compiler/installer.ini" %
- [VersionAsString, compileNimInst])
- exec("$# --var:version=$# --var:mingw=none --main:compiler/nim.nim xz compiler/installer.ini" %
- ["tools" / "niminst" / "niminst".exe, VersionAsString])
- proc buildTool(toolname, args: string) =
- nimexec("cc $# $#" % [args, toolname])
- copyFile(dest="bin" / splitFile(toolname).name.exe, source=toolname.exe)
- proc buildTools(latest: bool) =
- bundleNimsuggest()
- nimCompile("tools/nimgrep.nim", options = "-d:release")
- when defined(windows): buildVccTool()
- nimCompile("nimpretty/nimpretty.nim", options = "-d:release")
- buildNimble(latest)
- buildNimfind()
- proc nsis(latest: bool; args: string) =
- bundleNimbleExe(latest)
- bundleNimsuggest()
- bundleWinTools()
- # make sure we have generated the niminst executables:
- buildTool("tools/niminst/niminst", args)
- #buildTool("tools/nimgrep", args)
- # produce 'nim_debug.exe':
- #exec "nim c compiler" / "nim.nim"
- #copyExe("compiler/nim".exe, "bin/nim_debug".exe)
- exec(("tools" / "niminst" / "niminst --var:version=$# --var:mingw=mingw$#" &
- " nsis compiler/installer.ini") % [VersionAsString, $(sizeof(pointer)*8)])
- proc geninstall(args="") =
- nimexec("cc -r $# --var:version=$# --var:mingw=none --main:compiler/nim.nim scripts compiler/installer.ini $#" %
- [compileNimInst, VersionAsString, args])
- proc install(args: string) =
- geninstall()
- exec("sh ./install.sh $#" % args)
- when false:
- proc web(args: string) =
- nimexec("js tools/dochack/dochack.nim")
- nimexec("cc -r tools/nimweb.nim $# web/website.ini --putenv:nimversion=$#" %
- [args, VersionAsString])
- proc website(args: string) =
- nimexec("cc -r tools/nimweb.nim $# --website web/website.ini --putenv:nimversion=$#" %
- [args, VersionAsString])
- proc pdf(args="") =
- exec("$# cc -r tools/nimweb.nim $# --pdf web/website.ini --putenv:nimversion=$#" %
- [findNim(), args, VersionAsString], additionalPATH=findNim().splitFile.dir)
- # -------------- boot ---------------------------------------------------------
- proc findStartNim: string =
- # we try several things before giving up:
- # * bin/nim
- # * $PATH/nim
- # If these fail, we try to build nim with the "build.(sh|bat)" script.
- var nim = "nim".exe
- result = "bin" / nim
- if existsFile(result): return
- for dir in split(getEnv("PATH"), PathSep):
- if existsFile(dir / nim): return dir / nim
- when defined(Posix):
- const buildScript = "build.sh"
- if existsFile(buildScript):
- if tryExec("./" & buildScript): return "bin" / nim
- else:
- const buildScript = "build.bat"
- if existsFile(buildScript):
- if tryExec(buildScript): return "bin" / nim
- echo("Found no nim compiler and every attempt to build one failed!")
- quit("FAILURE")
- proc thVersion(i: int): string =
- result = ("compiler" / "nim" & $i).exe
- proc boot(args: string) =
- var output = "compiler" / "nim".exe
- var finalDest = "bin" / "nim".exe
- # default to use the 'c' command:
- let defaultCommand = if getEnv("NIM_COMPILE_TO_CPP", "false") == "true": "cpp" else: "c"
- let bootOptions = if args.len == 0 or args.startsWith("-"): defaultCommand else: ""
- let smartNimcache = (if "release" in args: "nimcache/r_" else: "nimcache/d_") &
- hostOs & "_" & hostCpu
- copyExe(findStartNim(), 0.thVersion)
- for i in 0..2:
- echo "iteration: ", i+1
- exec i.thVersion & " $# $# --nimcache:$# compiler" / "nim.nim" % [bootOptions, args,
- smartNimcache]
- if sameFileContent(output, i.thVersion):
- copyExe(output, finalDest)
- echo "executables are equal: SUCCESS!"
- return
- copyExe(output, (i+1).thVersion)
- copyExe(output, finalDest)
- when not defined(windows): echo "[Warning] executables are still not equal"
- # -------------- clean --------------------------------------------------------
- const
- cleanExt = [
- ".ppu", ".o", ".obj", ".dcu", ".~pas", ".~inc", ".~dsk", ".~dpr",
- ".map", ".tds", ".err", ".bak", ".pyc", ".exe", ".rod", ".pdb", ".idb",
- ".idx", ".ilk"
- ]
- ignore = [
- ".bzrignore", "nim", "nim.exe", "koch", "koch.exe", ".gitignore"
- ]
- proc cleanAux(dir: string) =
- for kind, path in walkDir(dir):
- case kind
- of pcFile:
- var (_, name, ext) = splitFile(path)
- if ext == "" or cleanExt.contains(ext):
- if not ignore.contains(name):
- echo "removing: ", path
- removeFile(path)
- of pcDir:
- case splitPath(path).tail
- of "nimcache":
- echo "removing dir: ", path
- removeDir(path)
- of "dist", ".git", "icons": discard
- else: cleanAux(path)
- else: discard
- proc removePattern(pattern: string) =
- for f in walkFiles(pattern):
- echo "removing: ", f
- removeFile(f)
- proc clean(args: string) =
- removePattern("web/*.html")
- removePattern("doc/*.html")
- cleanAux(getCurrentDir())
- for kind, path in walkDir(getCurrentDir() / "build"):
- if kind == pcDir:
- echo "removing dir: ", path
- removeDir(path)
- # -------------- builds a release ---------------------------------------------
- proc winReleaseArch(arch: string) =
- doAssert arch in ["32", "64"]
- let cpu = if arch == "32": "i386" else: "amd64"
- template withMingw(path, body) =
- let prevPath = getEnv("PATH")
- putEnv("PATH", (if path.len > 0: path & PathSep else: "") & prevPath)
- try:
- body
- finally:
- putEnv("PATH", prevPath)
- withMingw r"..\mingw" & arch & r"\bin":
- # Rebuilding koch is necessary because it uses its pointer size to
- # determine which mingw link to put in the NSIS installer.
- nimexec "c --cpu:$# koch" % cpu
- exec "koch boot -d:release --cpu:$#" % cpu
- exec "koch --latest zip -d:release"
- overwriteFile r"build\nim-$#.zip" % VersionAsString,
- r"web\upload\download\nim-$#_x$#.zip" % [VersionAsString, arch]
- proc winRelease*() =
- # Now used from "tools/winrelease" and not directly supported by koch
- # anymore!
- # Build -docs file:
- when true:
- buildDocs(gaCode)
- withDir "web/upload/" & VersionAsString:
- exec "7z a -tzip docs-$#.zip *.html" % VersionAsString
- overwriteFile "web/upload/$1/docs-$1.zip" % VersionAsString,
- "web/upload/download/docs-$1.zip" % VersionAsString
- when true:
- csource("-d:release")
- when sizeof(pointer) == 4:
- winReleaseArch "32"
- when sizeof(pointer) == 8:
- winReleaseArch "64"
- # -------------- tests --------------------------------------------------------
- template `|`(a, b): string = (if a.len > 0: a else: b)
- proc tests(args: string) =
- nimexec "cc --opt:speed testament/tester"
- # Since tests take a long time (on my machine), and we want to defy Murhpys
- # law - lets make sure the compiler really is freshly compiled!
- nimexec "c --lib:lib -d:release --opt:speed compiler/nim.nim"
- let tester = quoteShell(getCurrentDir() / "testament/tester".exe)
- let success = tryExec tester & " " & (args|"all")
- if not existsEnv("TRAVIS") and not existsEnv("APPVEYOR"):
- exec tester & " html"
- if not success:
- quit("tests failed", QuitFailure)
- proc temp(args: string) =
- proc splitArgs(a: string): (string, string) =
- # every --options before the command (indicated by starting
- # with not a dash) is part of the bootArgs, the rest is part
- # of the programArgs:
- let args = os.parseCmdLine a
- result = ("", "")
- var i = 0
- while i < args.len and args[i][0] == '-':
- result[0].add " " & quoteShell(args[i])
- inc i
- while i < args.len:
- result[1].add " " & quoteShell(args[i])
- inc i
- let d = getAppDir()
- var output = d / "compiler" / "nim".exe
- var finalDest = d / "bin" / "nim_temp".exe
- # 125 is the magic number to tell git bisect to skip the current
- # commit.
- let (bootArgs, programArgs) = splitArgs(args)
- let nimexec = findNim()
- exec(nimexec & " c -d:debug --debugger:native " & bootArgs & " " & (d / "compiler" / "nim"), 125)
- copyExe(output, finalDest)
- setCurrentDir(origDir)
- if programArgs.len > 0: exec(finalDest & " " & programArgs)
- proc xtemp(cmd: string) =
- let d = getAppDir()
- copyExe(d / "bin" / "nim".exe, d / "bin" / "nim_backup".exe)
- try:
- withDir(d):
- temp""
- copyExe(d / "bin" / "nim_temp".exe, d / "bin" / "nim".exe)
- exec(cmd)
- finally:
- copyExe(d / "bin" / "nim_backup".exe, d / "bin" / "nim".exe)
- proc pushCsources() =
- if not dirExists("../csources/.git"):
- quit "[Error] no csources git repository found"
- csource("-d:release")
- let cwd = getCurrentDir()
- try:
- copyDir("build/c_code", "../csources/c_code")
- copyFile("build/build.sh", "../csources/build.sh")
- copyFile("build/build.bat", "../csources/build.bat")
- copyFile("build/build64.bat", "../csources/build64.bat")
- copyFile("build/makefile", "../csources/makefile")
- setCurrentDir("../csources")
- for kind, path in walkDir("c_code"):
- if kind == pcDir:
- exec("git add " & path / "*.c")
- exec("git commit -am \"updated csources to version " & NimVersion & "\"")
- exec("git push origin master")
- exec("git tag -am \"Version $1\" v$1" % NimVersion)
- exec("git push origin v$1" % NimVersion)
- finally:
- setCurrentDir(cwd)
- proc testUnixInstall(cmdLineRest: string) =
- csource("-d:release " & cmdLineRest)
- xz(false, cmdLineRest)
- let oldCurrentDir = getCurrentDir()
- try:
- let destDir = getTempDir()
- copyFile("build/nim-$1.tar.xz" % VersionAsString,
- destDir / "nim-$1.tar.xz" % VersionAsString)
- setCurrentDir(destDir)
- execCleanPath("tar -xJf nim-$1.tar.xz" % VersionAsString)
- setCurrentDir("nim-$1" % VersionAsString)
- execCleanPath("sh build.sh")
- # first test: try if './bin/nim --version' outputs something sane:
- let output = execProcess("./bin/nim --version").splitLines
- if output.len > 0 and output[0].contains(VersionAsString):
- echo "Version check: success"
- execCleanPath("./bin/nim c koch.nim")
- execCleanPath("./koch boot -d:release", destDir / "bin")
- # check the docs build:
- execCleanPath("./koch docs", destDir / "bin")
- # check nimble builds:
- execCleanPath("./koch --latest tools")
- # check the tests work:
- putEnv("NIM_EXE_NOT_IN_PATH", "NOT_IN_PATH")
- execCleanPath("./koch tests", destDir / "bin")
- #execCleanPath("./koch tests cat newconfig", destDir / "bin")
- else:
- echo "Version check: failure"
- finally:
- setCurrentDir oldCurrentDir
- proc valgrind(cmd: string) =
- # somewhat hacky: '=' sign means "pass to valgrind" else "pass to Nim"
- let args = parseCmdLine(cmd)
- var nimcmd = ""
- var valcmd = ""
- for i, a in args:
- if i == args.len-1:
- # last element is the filename:
- valcmd.add ' '
- valcmd.add changeFileExt(a, ExeExt)
- nimcmd.add ' '
- nimcmd.add a
- elif '=' in a:
- valcmd.add ' '
- valcmd.add a
- else:
- nimcmd.add ' '
- nimcmd.add a
- exec("nim c" & nimcmd)
- let supp = getAppDir() / "tools" / "nimgrind.supp"
- exec("valgrind --suppressions=" & supp & valcmd)
- proc showHelp() =
- quit(HelpText % [VersionAsString & spaces(44-len(VersionAsString)),
- CompileDate, CompileTime], QuitSuccess)
- when isMainModule:
- var op = initOptParser()
- var latest = false
- var stable = false
- while true:
- op.next()
- case op.kind
- of cmdLongOption, cmdShortOption:
- case normalize(op.key)
- of "latest": latest = true
- of "stable": stable = true
- else: showHelp()
- of cmdArgument:
- case normalize(op.key)
- of "boot": boot(op.cmdLineRest)
- of "clean": clean(op.cmdLineRest)
- of "doc", "docs": buildDocs(op.cmdLineRest)
- of "doc0", "docs0":
- # undocumented command for Araq-the-merciful:
- buildDocs(op.cmdLineRest & gaCode)
- of "pdf": buildPdfDoc(op.cmdLineRest, "doc/pdf")
- of "csource", "csources": csource(op.cmdLineRest)
- of "zip": zip(latest, op.cmdLineRest)
- of "xz": xz(latest, op.cmdLineRest)
- of "nsis": nsis(latest, op.cmdLineRest)
- of "geninstall": geninstall(op.cmdLineRest)
- of "distrohelper": geninstall()
- of "install": install(op.cmdLineRest)
- of "testinstall": testUnixInstall(op.cmdLineRest)
- of "test", "tests": tests(op.cmdLineRest)
- of "temp": temp(op.cmdLineRest)
- of "xtemp": xtemp(op.cmdLineRest)
- of "wintools": bundleWinTools()
- of "nimble":
- if stable: buildNimble(false)
- else: buildNimble(existsDir(".git") or latest)
- of "nimsuggest": bundleNimsuggest()
- of "tools":
- if stable: buildTools(false)
- else: buildTools(existsDir(".git") or latest)
- of "pushcsource", "pushcsources": pushCsources()
- of "valgrind": valgrind(op.cmdLineRest)
- else: showHelp()
- break
- of cmdEnd: break
|