import $file.shared
import $file.upload
import java.io.File
import java.nio.file.attribute.PosixFilePermission
import ammonite.ops._
import coursier.maven.MavenRepository
import mill._
import mill.scalalib._
import publish._
import mill.modules.Jvm.createAssembly
import upickle.Js
trait MillPublishModule extends PublishModule{
def scalaVersion = "2.12.4"
def artifactName = "mill-" + super.artifactName()
def publishVersion = build.publishVersion()._2
def pomSettings = PomSettings(
description = artifactName(),
organization = "com.lihaoyi",
url = "https://github.com/lihaoyi/mill",
licenses = Seq(
License("MIT license", "http://www.opensource.org/licenses/mit-license.php")
),
// TODO 0.1.4
// licenses = Seq(License.MIT),
scm = SCM(
"git://github.com/lihaoyi/mill.git",
"scm:git://github.com/lihaoyi/mill.git"
),
// TODO 0.1.4:
// versionControl = VersionControl.github("lihaoyi", "mill")
developers = Seq(
Developer("lihaoyi", "Li Haoyi","https://github.com/lihaoyi")
)
)
}
object moduledefs extends MillPublishModule{
def ivyDeps = Agg(
ivy"org.scala-lang:scala-compiler:${scalaVersion()}",
ivy"com.lihaoyi::sourcecode:0.1.4"
)
}
trait MillModule extends MillPublishModule{ outer =>
def compileIvyDeps = Agg(ivy"com.lihaoyi::acyclic:0.1.7")
def scalacOptions = Seq("-P:acyclic:force")
def scalacPluginIvyDeps = Agg(ivy"com.lihaoyi::acyclic:0.1.7")
def repositories = super.repositories ++ Seq(
MavenRepository("https://oss.sonatype.org/content/repositories/releases")
)
def testArgs = T{ Seq.empty[String] }
val test = new Tests(implicitly)
class Tests(ctx0: mill.define.Ctx) extends mill.Module()(ctx0) with super.Tests{
def forkArgs = T{ testArgs() }
def moduleDeps =
if (this == main.test) Seq(main)
else Seq(outer, main.test)
def ivyDeps = Agg(ivy"com.lihaoyi::utest:0.6.4")
def testFrameworks = Seq("mill.UTestFramework")
def scalacPluginClasspath = super.scalacPluginClasspath() ++ Seq(moduledefs.jar())
}
}
object clientserver extends MillModule{
def ivyDeps = Agg(
ivy"com.lihaoyi:::ammonite:1.0.5-14-daf7e0d",
ivy"com.lihaoyi::upickle:0.6.0-RC6",
ivy"org.scala-sbt.ipcsocket:ipcsocket:1.0.0"
)
val test = new Tests(implicitly)
}
object core extends MillModule {
def moduleDeps = Seq(moduledefs)
def compileIvyDeps = Agg(
ivy"org.scala-lang:scala-reflect:${scalaVersion()}"
)
def ivyDeps = Agg(
ivy"com.lihaoyi::sourcecode:0.1.4",
ivy"com.lihaoyi:::ammonite:1.0.5-14-daf7e0d",
ivy"com.lihaoyi::upickle:0.6.0-RC6",
ivy"jline:jline:2.14.5"
)
def generatedSources = T {
Seq(PathRef(shared.generateCoreSources(T.ctx().dest)))
}
}
object main extends MillModule {
def moduleDeps = Seq(core, clientserver)
def compileIvyDeps = Agg(
ivy"org.scala-lang:scala-reflect:${scalaVersion()}"
)
def generatedSources = T {
Seq(PathRef(shared.generateCoreSources(T.ctx().dest)))
}
val test = new Tests(implicitly)
class Tests(ctx0: mill.define.Ctx) extends super.Tests(ctx0){
def generatedSources = T {
Seq(PathRef(shared.generateCoreTestSources(T.ctx().dest)))
}
}
}
object scalaworker extends MillModule{
def moduleDeps = Seq(main, scalalib)
def ivyDeps = Agg(
ivy"org.scala-sbt::zinc:1.0.5"
)
def testArgs = Seq(
"-DMILL_SCALA_WORKER=" + runClasspath().map(_.path).mkString(",")
)
}
object scalalib extends MillModule {
def moduleDeps = Seq(main)
def ivyDeps = Agg(
ivy"org.scala-sbt:test-interface:1.0"
)
def genTask(m: ScalaModule) = T.task{
Seq(m.jar(), m.sourceJar()) ++
m.runClasspath()
}
def testArgs = T{
val genIdeaArgs =
genTask(moduledefs)() ++
genTask(core)() ++
genTask(main)() ++
genTask(scalalib)() ++
genTask(scalajslib)()
scalaworker.testArgs() ++
Seq("-DMILL_BUILD_LIBRARIES=" + genIdeaArgs.map(_.path).mkString(","))
}
}
object scalajslib extends MillModule {
def moduleDeps = Seq(scalalib)
def testArgs = T{
val mapping = Map(
"MILL_SCALAJS_BRIDGE_0_6" -> jsbridges("0.6").compile().classes.path,
"MILL_SCALAJS_BRIDGE_1_0" -> jsbridges("1.0").compile().classes.path
)
scalaworker.testArgs() ++ (for((k, v) <- mapping.toSeq) yield s"-D$k=$v")
}
object jsbridges extends Cross[JsBridgeModule]("0.6", "1.0")
class JsBridgeModule(scalajsBinary: String) extends MillModule{
def moduleDeps = Seq(scalajslib)
def ivyDeps = scalajsBinary match {
case "0.6" =>
Agg(
ivy"org.scala-js::scalajs-tools:0.6.22",
ivy"org.scala-js::scalajs-sbt-test-adapter:0.6.22",
ivy"org.scala-js::scalajs-js-envs:0.6.22"
)
case "1.0" =>
Agg(
ivy"org.scala-js::scalajs-tools:1.0.0-M2",
ivy"org.scala-js::scalajs-sbt-test-adapter:1.0.0-M2",
ivy"org.scala-js::scalajs-env-nodejs:1.0.0-M2"
)
}
}
}
def testRepos = T{
Seq(
"MILL_ACYCLIC_REPO" ->
shared.downloadTestRepo("lihaoyi/acyclic", "bc41cd09a287e2c270271e27ccdb3066173a8598", T.ctx().dest/"acyclic"),
"MILL_JAWN_REPO" ->
shared.downloadTestRepo("non/jawn", "fd8dc2b41ce70269889320aeabf8614fe1e8fbcb", T.ctx().dest/"jawn"),
"MILL_BETTERFILES_REPO" ->
shared.downloadTestRepo("pathikrit/better-files", "ba74ae9ef784dcf37f1b22c3990037a4fcc6b5f8", T.ctx().dest/"better-files"),
"MILL_AMMONITE_REPO" ->
shared.downloadTestRepo("lihaoyi/ammonite", "96ea548d5e3b72ab6ad4d9765e205bf6cc1c82ac", T.ctx().dest/"ammonite"),
"MILL_UPICKLE_REPO" ->
shared.downloadTestRepo("lihaoyi/upickle", "7f33085c890db7550a226c349832eabc3cd18769", T.ctx().dest/"upickle")
)
}
object integration extends MillModule{
def moduleDeps = Seq(moduledefs, scalalib, scalajslib)
def testArgs = T{
scalajslib.testArgs() ++
scalaworker.testArgs() ++
(for((k, v) <- testRepos()) yield s"-D$k=$v")
}
def forkArgs = testArgs()
}
def launcherScript(isWin: Boolean,
jvmArgs: Seq[String],
classPath: Agg[String]) = {
val jvmArgsStr = jvmArgs.mkString(" ")
val classPathStr = if (isWin) classPath.mkString(";") else classPath.mkString(":")
if (isWin)
s"""::#!
|@echo off
|if "%1" == "-i" set _I_=true
|if "%1" == "--interactive" set _I_=true
|if defined _I_ (
| if "%2" == "" (
| echo mill repl is currently only available on a sh environment
| exit /B 1
| ) else (
| java $jvmArgsStr %JAVA_OPTS% -cp "$classPathStr" mill.Main %*
| )
|) else (
| java $jvmArgsStr %JAVA_OPTS% -cp "$classPathStr" mill.clientserver.Client %*
|)
|EXIT /B %errorlevel%
""".stripMargin.split('\n').mkString("\r\n")
else
s"""#!/usr/bin/env sh
|
|case "$$1" in
| -i | --interactive )
| exec java $jvmArgsStr $$JAVA_OPTS -cp "$classPathStr" mill.Main "$$@"
| ;;
| *)
| exec java $jvmArgsStr $$JAVA_OPTS -cp "$classPathStr" mill.clientserver.Client "$$@"
| ;;
|esac
""".stripMargin
}
object dev extends MillModule{
def moduleDeps = Seq(scalalib, scalajslib)
def forkArgs = T{
scalalib.testArgs() ++ scalajslib.testArgs() ++ scalaworker.testArgs()
}
def launcher = T{
val isWin = scala.util.Properties.isWin
val outputPath = T.ctx().dest / (if (isWin) "run.bat" else "run")
write(outputPath, prependShellScript())
if (!isWin) {
val perms = java.nio.file.Files.getPosixFilePermissions(outputPath.toNIO)
perms.add(PosixFilePermission.GROUP_EXECUTE)
perms.add(PosixFilePermission.OWNER_EXECUTE)
perms.add(PosixFilePermission.OTHERS_EXECUTE)
java.nio.file.Files.setPosixFilePermissions(outputPath.toNIO, perms)
}
PathRef(outputPath)
}
def assembly = T{
val filename = if (scala.util.Properties.isWin) "mill.bat" else "mill"
mv(super.assembly().path, T.ctx().dest / filename)
PathRef(T.ctx().dest / filename)
}
def prependShellScript = launcherScript(scala.util.Properties.isWin, forkArgs(), runClasspath().map(_.path.toString))
def run(args: String*) = T.command{
args match{
case Nil => mill.eval.Result.Failure("Need to pass in cwd as first argument to dev.run")
case wd0 +: rest =>
val wd = Path(wd0, pwd)
mkdir(wd)
mill.modules.Jvm.baseInteractiveSubprocess(
Seq(launcher().path.toString) ++ rest,
forkEnv(),
workingDir = wd
)
mill.eval.Result.Success(())
}
}
}
private def releaseHelper(dest: Path,
cp: Agg[Path],
ver: String,
isWin: Boolean)
(implicit ctx: mill.util.Ctx.Dest): PathRef = {
val (filename, arg) =
if (isWin) ("mill.bat", "%~dp0%~nx0")
else ("mill", "$0")
mv(
createAssembly(
cp,
prependShellScript = launcherScript(
isWin,
Seq("-DMILL_VERSION=" + ver),
Agg(arg)
)
).path,
dest / filename
)
PathRef(dest / filename)
}
def release = T{
releaseHelper(
T.ctx().dest,
dev.runClasspath().map(_.path),
publishVersion()._2,
false)
}
def releaseBatch = T{
releaseHelper(
T.ctx().dest,
dev.runClasspath().map(_.path),
publishVersion()._2,
true)
}
def releaseAll = T{
val dest = T.ctx().dest
val cp = dev.runClasspath().map(_.path)
val ver = publishVersion()._2
for (isWin <- Seq(false, true))
yield (isWin, releaseHelper(dest, cp, ver, isWin))
}
val isMasterCommit = {
sys.env.get("TRAVIS_PULL_REQUEST") == Some("false") &&
(sys.env.get("TRAVIS_BRANCH") == Some("master") || sys.env("TRAVIS_TAG") != "")
}
def gitHead = T.input{
sys.env.get("TRAVIS_COMMIT").getOrElse(
%%('git, "rev-parse", "HEAD")(pwd).out.string.trim()
)
}
def publishVersion = T.input{
val tag =
try Option(
%%('git, 'describe, "--exact-match", "--tags", gitHead())(pwd).out.string.trim()
)
catch{case e => None}
tag match{
case Some(t) => (t, t)
case None =>
val latestTaggedVersion = %%('git, 'describe, "--abbrev=0", "--tags")(pwd).out.trim
val commitsSinceLastTag =
%%('git, "rev-list", gitHead(), "--count")(pwd).out.trim.toInt -
%%('git, "rev-list", latestTaggedVersion, "--count")(pwd).out.trim.toInt
(latestTaggedVersion, s"$latestTaggedVersion-$commitsSinceLastTag-${gitHead().take(6)}")
}
}
def uploadToGithub(authKey: String) = T.command{
val (releaseTag, label) = publishVersion()
if (releaseTag == label){
scalaj.http.Http("https://api.github.com/repos/lihaoyi/mill/releases")
.postData(
upickle.json.write(
Js.Obj(
"tag_name" -> Js.Str(releaseTag),
"name" -> Js.Str(releaseTag)
)
)
)
.header("Authorization", "token " + authKey)
.asString
}
for ((isWin, pr) <- releaseAll())
upload.apply(pr.path, releaseTag,
if (isWin) s"mill-$label.bat" // so browser downloads it as mill-<version>.bat (?)
else label, authKey)
}