diff --git a/.gitignore b/.gitignore
index 7e740ce2..a680191b 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,15 +1,15 @@
.gradle
annotations
.idea/*
-!.idea/modules.xml
-!.idea/libraries
-!.idea/misc.xml
-buildScript
+*.iml
+nonBuildScript
kobaltBuild
local.properties
classes
libs
.kobalt/
-./build/
out
.DS_Store
+lib/kotlin-*
+build
+.history
diff --git a/.idea/libraries/kobalt_jar.xml b/.idea/libraries/kobalt_jar.xml
deleted file mode 100644
index 9e02804a..00000000
--- a/.idea/libraries/kobalt_jar.xml
+++ /dev/null
@@ -1,11 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/.idea/misc.xml b/.idea/misc.xml
deleted file mode 100644
index df7e8eb0..00000000
--- a/.idea/misc.xml
+++ /dev/null
@@ -1,9 +0,0 @@
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/.idea/modules.xml b/.idea/modules.xml
deleted file mode 100644
index a41ffd7c..00000000
--- a/.idea/modules.xml
+++ /dev/null
@@ -1,11 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/.travis.yml b/.travis.yml
deleted file mode 100644
index 0c1d3b1d..00000000
--- a/.travis.yml
+++ /dev/null
@@ -1,13 +0,0 @@
-cache:
- directories:
- - $HOME/.m2
- - $HOME/.kobalt
-
-language: java
-
-jdk:
- - oraclejdk8
-
-install: true
-
-script: ./build-travis.sh
diff --git a/README.md b/README.md
index e8db3ef6..d5d7cbe0 100644
--- a/README.md
+++ b/README.md
@@ -1,11 +1,14 @@
-# Kobalt [](https://travis-ci.org/cbeust/kobalt)
+# Kobalt
+
+[
](https://teamcity.jetbrains.com/project.html?projectId=OpenSourceProjects_Kobalt&tab=projectOverview)
+
Kobalt is a universal build system.
To build it:
```
-./kobaltw assemble
+$ ./kobaltw assemble
```
Please see [the web site](http://beust.com/kobalt/) for the full documentation.
diff --git a/build-travis.sh b/build-travis.sh
index 467849b9..c7cb1152 100755
--- a/build-travis.sh
+++ b/build-travis.sh
@@ -1,4 +1,8 @@
-java -Xmx2048m -jar $(dirname $0)/kobalt/wrapper/kobalt-wrapper.jar $* assemble
+ulimit -s 1082768
+
+#java -Xmx2048m -jar $(dirname $0)/kobalt/wrapper/kobalt-wrapper.jar $* clean
+java -Xmx2048m -jar $(dirname $0)/kobalt/wrapper/kobalt-wrapper.jar $* clean assemble test --parallel
+
+
-java -Xmx2048m -jar $(dirname $0)/kobalt/wrapper/kobalt-wrapper.jar $* test
diff --git a/build.gradle b/build.gradle
new file mode 100644
index 00000000..3f0053cc
--- /dev/null
+++ b/build.gradle
@@ -0,0 +1,58 @@
+allprojects {
+ group = 'com.beust'
+ version = '1.1.0'
+}
+
+subprojects {
+ apply plugin: 'java'
+ apply plugin: 'maven-publish'
+
+ ext {
+ bndlib = '3.5.0'
+ findbugs = '3.0.2'
+ groovy = '2.4.12'
+ gson = '2.8.2'
+ guice = '4.2.2'
+ inject = '1'
+ jaxb = '2.3.0'
+ jcommander = '1.72'
+ kotlin = '1.2.71'
+ maven = '3.5.2'
+ mavenResolver = '1.1.0'
+ okhttp = '3.9.1'
+ okio = '1.13.0'
+ retrofit = '2.3.0'
+ slf4j = '1.7.3'
+ spark = '2.6.0'
+ testng = '6.12'
+
+ junit = '4.12'
+ junitJupiter = '5.1.0'
+ junitPlatform = '1.1.0'
+ }
+
+ repositories {
+ mavenCentral()
+ mavenLocal()
+ jcenter()
+ maven {
+ url = 'https://dl.bintray.com/cbeust/maven'
+ }
+
+ maven {
+ url = 'https://repo.maven.apache.org/maven2'
+ }
+ }
+
+ sourceCompatibility = '1.7'
+
+ task sourcesJar(type: Jar) {
+ from sourceSets.main.allJava
+ archiveClassifier = 'sources'
+ }
+
+ task javadocJar(type: Jar) {
+ from javadoc
+ archiveClassifier = 'javadoc'
+ }
+}
diff --git a/dist/kobaltw b/dist/kobaltw
index a8df4bb9..333738df 100755
--- a/dist/kobaltw
+++ b/dist/kobaltw
@@ -1,2 +1,11 @@
#!/usr/bin/env sh
-java -jar "`dirname "$0"`/../kobalt/wrapper/kobalt-wrapper.jar" $*
\ No newline at end of file
+
+case "$(uname)" in
+ CYGWIN*) DIRNAME=$(cygpath -d "$(dirname "$(readlink -f "$0")")");;
+ Darwin*) DIRNAME=$(dirname "$(readlink "$0")");;
+ *) DIRNAME=$(dirname "$(readlink -f "$0")");;
+esac
+if [ "$DIRNAME" = "." ]; then
+ DIRNAME="$(dirname "$0")"
+fi
+java -jar "${DIRNAME}/../kobalt/wrapper/kobalt-wrapper.jar" $*
\ No newline at end of file
diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar
new file mode 100644
index 00000000..5c2d1cf0
Binary files /dev/null and b/gradle/wrapper/gradle-wrapper.jar differ
diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties
new file mode 100644
index 00000000..838e6bc8
--- /dev/null
+++ b/gradle/wrapper/gradle-wrapper.properties
@@ -0,0 +1,5 @@
+distributionBase=GRADLE_USER_HOME
+distributionPath=wrapper/dists
+distributionUrl=https\://services.gradle.org/distributions/gradle-5.3-bin.zip
+zipStoreBase=GRADLE_USER_HOME
+zipStorePath=wrapper/dists
diff --git a/gradlew b/gradlew
new file mode 100755
index 00000000..b0d6d0ab
--- /dev/null
+++ b/gradlew
@@ -0,0 +1,188 @@
+#!/usr/bin/env sh
+
+#
+# Copyright 2015 the original author or authors.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+##############################################################################
+##
+## Gradle start up script for UN*X
+##
+##############################################################################
+
+# Attempt to set APP_HOME
+# Resolve links: $0 may be a link
+PRG="$0"
+# Need this for relative symlinks.
+while [ -h "$PRG" ] ; do
+ ls=`ls -ld "$PRG"`
+ link=`expr "$ls" : '.*-> \(.*\)$'`
+ if expr "$link" : '/.*' > /dev/null; then
+ PRG="$link"
+ else
+ PRG=`dirname "$PRG"`"/$link"
+ fi
+done
+SAVED="`pwd`"
+cd "`dirname \"$PRG\"`/" >/dev/null
+APP_HOME="`pwd -P`"
+cd "$SAVED" >/dev/null
+
+APP_NAME="Gradle"
+APP_BASE_NAME=`basename "$0"`
+
+# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
+DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
+
+# Use the maximum available, or set MAX_FD != -1 to use that value.
+MAX_FD="maximum"
+
+warn () {
+ echo "$*"
+}
+
+die () {
+ echo
+ echo "$*"
+ echo
+ exit 1
+}
+
+# OS specific support (must be 'true' or 'false').
+cygwin=false
+msys=false
+darwin=false
+nonstop=false
+case "`uname`" in
+ CYGWIN* )
+ cygwin=true
+ ;;
+ Darwin* )
+ darwin=true
+ ;;
+ MINGW* )
+ msys=true
+ ;;
+ NONSTOP* )
+ nonstop=true
+ ;;
+esac
+
+CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
+
+# Determine the Java command to use to start the JVM.
+if [ -n "$JAVA_HOME" ] ; then
+ if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
+ # IBM's JDK on AIX uses strange locations for the executables
+ JAVACMD="$JAVA_HOME/jre/sh/java"
+ else
+ JAVACMD="$JAVA_HOME/bin/java"
+ fi
+ if [ ! -x "$JAVACMD" ] ; then
+ die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
+
+Please set the JAVA_HOME variable in your environment to match the
+location of your Java installation."
+ fi
+else
+ JAVACMD="java"
+ which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
+
+Please set the JAVA_HOME variable in your environment to match the
+location of your Java installation."
+fi
+
+# Increase the maximum file descriptors if we can.
+if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
+ MAX_FD_LIMIT=`ulimit -H -n`
+ if [ $? -eq 0 ] ; then
+ if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
+ MAX_FD="$MAX_FD_LIMIT"
+ fi
+ ulimit -n $MAX_FD
+ if [ $? -ne 0 ] ; then
+ warn "Could not set maximum file descriptor limit: $MAX_FD"
+ fi
+ else
+ warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
+ fi
+fi
+
+# For Darwin, add options to specify how the application appears in the dock
+if $darwin; then
+ GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
+fi
+
+# For Cygwin, switch paths to Windows format before running java
+if $cygwin ; then
+ APP_HOME=`cygpath --path --mixed "$APP_HOME"`
+ CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
+ JAVACMD=`cygpath --unix "$JAVACMD"`
+
+ # We build the pattern for arguments to be converted via cygpath
+ ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
+ SEP=""
+ for dir in $ROOTDIRSRAW ; do
+ ROOTDIRS="$ROOTDIRS$SEP$dir"
+ SEP="|"
+ done
+ OURCYGPATTERN="(^($ROOTDIRS))"
+ # Add a user-defined pattern to the cygpath arguments
+ if [ "$GRADLE_CYGPATTERN" != "" ] ; then
+ OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
+ fi
+ # Now convert the arguments - kludge to limit ourselves to /bin/sh
+ i=0
+ for arg in "$@" ; do
+ CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
+ CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
+
+ if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
+ eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
+ else
+ eval `echo args$i`="\"$arg\""
+ fi
+ i=$((i+1))
+ done
+ case $i in
+ (0) set -- ;;
+ (1) set -- "$args0" ;;
+ (2) set -- "$args0" "$args1" ;;
+ (3) set -- "$args0" "$args1" "$args2" ;;
+ (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
+ (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
+ (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
+ (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
+ (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
+ (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
+ esac
+fi
+
+# Escape application args
+save () {
+ for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
+ echo " "
+}
+APP_ARGS=$(save "$@")
+
+# Collect all arguments for the java command, following the shell quoting and substitution rules
+eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
+
+# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong
+if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then
+ cd "$(dirname "$0")"
+fi
+
+exec "$JAVACMD" "$@"
diff --git a/gradlew.bat b/gradlew.bat
new file mode 100644
index 00000000..9991c503
--- /dev/null
+++ b/gradlew.bat
@@ -0,0 +1,100 @@
+@rem
+@rem Copyright 2015 the original author or authors.
+@rem
+@rem Licensed under the Apache License, Version 2.0 (the "License");
+@rem you may not use this file except in compliance with the License.
+@rem You may obtain a copy of the License at
+@rem
+@rem http://www.apache.org/licenses/LICENSE-2.0
+@rem
+@rem Unless required by applicable law or agreed to in writing, software
+@rem distributed under the License is distributed on an "AS IS" BASIS,
+@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+@rem See the License for the specific language governing permissions and
+@rem limitations under the License.
+@rem
+
+@if "%DEBUG%" == "" @echo off
+@rem ##########################################################################
+@rem
+@rem Gradle startup script for Windows
+@rem
+@rem ##########################################################################
+
+@rem Set local scope for the variables with windows NT shell
+if "%OS%"=="Windows_NT" setlocal
+
+set DIRNAME=%~dp0
+if "%DIRNAME%" == "" set DIRNAME=.
+set APP_BASE_NAME=%~n0
+set APP_HOME=%DIRNAME%
+
+@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
+set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
+
+@rem Find java.exe
+if defined JAVA_HOME goto findJavaFromJavaHome
+
+set JAVA_EXE=java.exe
+%JAVA_EXE% -version >NUL 2>&1
+if "%ERRORLEVEL%" == "0" goto init
+
+echo.
+echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
+echo.
+echo Please set the JAVA_HOME variable in your environment to match the
+echo location of your Java installation.
+
+goto fail
+
+:findJavaFromJavaHome
+set JAVA_HOME=%JAVA_HOME:"=%
+set JAVA_EXE=%JAVA_HOME%/bin/java.exe
+
+if exist "%JAVA_EXE%" goto init
+
+echo.
+echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
+echo.
+echo Please set the JAVA_HOME variable in your environment to match the
+echo location of your Java installation.
+
+goto fail
+
+:init
+@rem Get command-line arguments, handling Windows variants
+
+if not "%OS%" == "Windows_NT" goto win9xME_args
+
+:win9xME_args
+@rem Slurp the command line arguments.
+set CMD_LINE_ARGS=
+set _SKIP=2
+
+:win9xME_args_slurp
+if "x%~1" == "x" goto execute
+
+set CMD_LINE_ARGS=%*
+
+:execute
+@rem Setup the command line
+
+set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
+
+@rem Execute Gradle
+"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
+
+:end
+@rem End local scope for the variables with windows NT shell
+if "%ERRORLEVEL%"=="0" goto mainEnd
+
+:fail
+rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
+rem the _cmd.exe /c_ return code!
+if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
+exit /b 1
+
+:mainEnd
+if "%OS%"=="Windows_NT" endlocal
+
+:omega
diff --git a/kobalt.iml b/kobalt.iml
deleted file mode 100644
index f4ff4022..00000000
--- a/kobalt.iml
+++ /dev/null
@@ -1,408 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/kobalt/Build.kt.iml b/kobalt/Build.kt.iml
deleted file mode 100644
index 7bc9dd64..00000000
--- a/kobalt/Build.kt.iml
+++ /dev/null
@@ -1,12 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/kobalt/src/Build.kt b/kobalt/src/Build.kt
index 7226dcfc..0d09844a 100644
--- a/kobalt/src/Build.kt
+++ b/kobalt/src/Build.kt
@@ -1,32 +1,57 @@
-import com.beust.kobalt.plugin.application.application
-import com.beust.kobalt.plugin.java.javaCompiler
-import com.beust.kobalt.plugin.packaging.assemble
-import com.beust.kobalt.project
-import com.beust.kobalt.TaskResult
+import com.beust.kobalt.*
import com.beust.kobalt.api.Project
import com.beust.kobalt.api.annotation.Task
-import com.beust.kobalt.homeDir
+import com.beust.kobalt.plugin.application.application
+import com.beust.kobalt.plugin.java.javaCompiler
import com.beust.kobalt.plugin.kotlin.kotlinCompiler
+import com.beust.kobalt.plugin.packaging.assemble
+import com.beust.kobalt.plugin.publish.autoGitTag
import com.beust.kobalt.plugin.publish.bintray
import com.beust.kobalt.plugin.publish.github
-import com.beust.kobalt.test
-import org.apache.maven.model.*
+import org.apache.maven.model.Developer
+import org.apache.maven.model.License
+import org.apache.maven.model.Model
+import org.apache.maven.model.Scm
import java.io.File
import java.nio.file.Files
import java.nio.file.Paths
import java.nio.file.StandardCopyOption
+import java.util.zip.ZipEntry
+import java.util.zip.ZipOutputStream
+
+val bs = buildScript {
+ repos("https://dl.bintray.com/cbeust/maven")
+}
object Versions {
- val okhttp = "3.2.0"
- val okio = "1.6.0"
- val retrofit = "2.1.0"
- val gson = "2.6.2"
- val aether = "1.1.0"
- val sonatypeAether = "1.13.1"
- val maven = "3.3.9"
+ val kotlin = "1.2.71"
+ val okhttp = "3.9.1"
+ val okio = "1.13.0"
+ val retrofit = "2.3.0"
+ val gson = "2.8.2"
+ val guice = "4.2.2"
+ val maven = "3.5.2"
+ val mavenResolver = "1.1.0"
+ val slf4j = "1.7.3"
+ val aether = "1.0.2.v20150114"
+ val testng = "6.12"
+ val jcommander = "1.72"
+
+ // JUnit 5
+ val junit = "4.12"
+ val junitPlatform = "1.1.0"
+ val junitJupiter = "5.1.0"
}
+fun mavenResolver(vararg m: String)
+ = m.map { "org.apache.maven.resolver:maven-resolver-$it:${Versions.mavenResolver}" }
+ .toTypedArray()
+
+fun aether(vararg m: String)
+ = m.map { "org.eclipse.aether:aether-$it:${Versions.aether}" }
+ .toTypedArray()
+
val wrapper = project {
name = "kobalt-wrapper"
group = "com.beust"
@@ -39,6 +64,7 @@ val wrapper = project {
}
assemble {
+ jar { }
jar {
name = projectName + ".jar"
manifest {
@@ -50,6 +76,13 @@ val wrapper = project {
application {
mainClass = "com.beust.kobalt.wrapper.Main"
}
+
+ bintray {
+ publish = true
+ sign = true
+ }
+
+ pom = createPom(name, "Wrapper for Kobalt")
}
val kobaltPluginApi = project {
@@ -59,52 +92,44 @@ val kobaltPluginApi = project {
version = readVersion()
directory = "modules/kobalt-plugin-api"
description = "A build system in Kotlin"
- url = "http://beust.com/kobalt"
+ url = "https://beust.com/kobalt"
- pom = Model().apply {
- name = project.name
- description = "A build system in Kotlin"
- url = "http://beust.com/kobalt"
- licenses = listOf(License().apply {
- name = "Apache 2.0"
- url = "http://www.apache .org/licenses/LICENSE-2.0"
- })
- scm = Scm().apply {
- url = "http://github.com/cbeust/kobalt"
- connection = "https://github.com/cbeust/kobalt.git"
- developerConnection = "git@github.com:cbeust/kobalt.git"
- }
- developers = listOf(Developer().apply {
- name = "Cedric Beust"
- email = "cedric@beust.com"
- })
- }
+ pom = createPom(name, "A build system in Kotlin")
dependencies {
compile(
- "com.google.inject:guice:4.0",
- "com.google.inject.extensions:guice-assistedinject:4.0",
+ "org.jetbrains.kotlin:kotlin-stdlib:${Versions.kotlin}",
+ "com.google.inject:guice:${Versions.guice}",
+ "com.google.inject.extensions:guice-assistedinject:4.1.0",
"javax.inject:javax.inject:1",
- "com.google.guava:guava:19.0",
+ "com.google.guava:guava:27.0.1-jre",
"org.apache.maven:maven-model:${Versions.maven}",
- "io.reactivex:rxjava:1.1.5",
+ "io.reactivex:rxjava:1.3.3",
"com.squareup.okio:okio:${Versions.okio}",
"com.google.code.gson:gson:${Versions.gson}",
"com.squareup.okhttp3:okhttp:${Versions.okhttp}",
"com.squareup.retrofit2:retrofit:${Versions.retrofit}",
"com.squareup.retrofit2:converter-gson:${Versions.retrofit}",
- "com.beust:jcommander:1.48",
+ "com.beust:jcommander:${Versions.jcommander}",
+ "org.eclipse.jgit:org.eclipse.jgit:4.9.0.201710071750-r",
+ "org.slf4j:slf4j-simple:${Versions.slf4j}",
+ *mavenResolver("api", "spi", "util", "impl", "connector-basic", "transport-http", "transport-file"),
+ "org.apache.maven:maven-aether-provider:3.3.9",
+ "org.testng.testng-remote:testng-remote:1.3.2",
+ "org.testng:testng:${Versions.testng}",
+ "org.junit.platform:junit-platform-surefire-provider:${Versions.junitPlatform}",
+ "org.junit.platform:junit-platform-runner:${Versions.junitPlatform}",
+ "org.junit.platform:junit-platform-engine:${Versions.junitPlatform}",
+ "org.junit.platform:junit-platform-console:${Versions.junitPlatform}",
+ "org.junit.jupiter:junit-jupiter-engine:${Versions.junitJupiter}",
+ "org.junit.vintage:junit-vintage-engine:${Versions.junitJupiter}",
+ "org.apache.commons:commons-compress:1.15",
+ "commons-io:commons-io:2.6",
- "org.slf4j:slf4j-nop:1.6.0",
- "org.eclipse.aether:aether-spi:${Versions.aether}",
- "org.eclipse.aether:aether-impl:${Versions.aether}",
- "org.eclipse.aether:aether-connector-basic:${Versions.aether}",
- "org.eclipse.aether:aether-transport-file:${Versions.aether}",
- "org.eclipse.aether:aether-transport-http:${Versions.aether}",
- "org.sonatype.aether:aether-api:${Versions.sonatypeAether}",
- "org.sonatype.aether:aether-connector-wagon:1.13.1",
- "org.apache.maven:maven-aether-provider:${Versions.maven}"
+ // Java 9
+ "javax.xml.bind:jaxb-api:2.3.0"
)
+ exclude(*aether("impl", "spi", "util", "api"))
}
@@ -117,12 +142,8 @@ val kobaltPluginApi = project {
}
}
-// install {
-// libDir = "lib-test"
-// }
-
kotlinCompiler {
- args("-nowarn")
+ args("nowarn")
}
bintray {
@@ -138,26 +159,35 @@ val kobaltApp = project(kobaltPluginApi, wrapper) {
dependencies {
// Used by the plugins
- compile("org.jetbrains.kotlin:kotlin-compiler-embeddable:1.0.3")
+ compile("org.jetbrains.kotlin:kotlin-compiler-embeddable:${Versions.kotlin}")
// Used by the main app
- compile("com.github.spullara.mustache.java:compiler:0.9.1",
+ compile(
+ "org.jetbrains.kotlin:kotlin-stdlib:${Versions.kotlin}",
+ "com.github.spullara.mustache.java:compiler:0.9.5",
"javax.inject:javax.inject:1",
- "com.google.inject:guice:4.0",
- "com.google.inject.extensions:guice-assistedinject:4.0",
- "com.beust:jcommander:1.48",
+ "com.google.inject:guice:${Versions.guice}",
+ "com.google.inject.extensions:guice-assistedinject:${Versions.guice}",
+ "com.beust:jcommander:${Versions.jcommander}",
"org.apache.maven:maven-model:${Versions.maven}",
- "com.google.code.findbugs:jsr305:3.0.1",
+ "com.google.code.findbugs:jsr305:3.0.2",
"com.google.code.gson:gson:${Versions.gson}",
"com.squareup.retrofit2:retrofit:${Versions.retrofit}",
"com.squareup.retrofit2:converter-gson:${Versions.retrofit}",
- "com.squareup.okhttp3:okhttp-ws:${Versions.okhttp}",
- "org.codehaus.plexus:plexus-utils:3.0.22",
- "biz.aQute.bnd:bndlib:2.4.0",
+// "com.squareup.okhttp3:okhttp-ws:3.4.2",
+ "biz.aQute.bnd:biz.aQute.bndlib:3.5.0",
+ *mavenResolver("spi"),
- "com.squareup.okhttp3:logging-interceptor:3.2.0",
+ "com.squareup.okhttp3:logging-interceptor:3.9.0",
- "com.sparkjava:spark-core:2.5"
+ "com.sparkjava:spark-core:2.6.0",
+ "org.codehaus.groovy:groovy:2.4.12",
+
+ // Java 9
+ "javax.xml.bind:jaxb-api:2.3.0",
+ "com.sun.xml.bind:jaxb-impl:2.3.0",
+ "com.sun.xml.bind:jaxb-core:2.3.0",
+ "com.sun.activation:javax.activation:1.2.0"
// "org.eclipse.jetty:jetty-server:${Versions.jetty}",
// "org.eclipse.jetty:jetty-servlet:${Versions.jetty}",
@@ -171,8 +201,11 @@ val kobaltApp = project(kobaltPluginApi, wrapper) {
}
dependenciesTest {
- compile("org.testng:testng:6.9.11",
- "org.assertj:assertj-core:3.4.1")
+ compile("org.jetbrains.kotlin:kotlin-test:${Versions.kotlin}",
+ "org.testng:testng:${Versions.testng}",
+ "org.assertj:assertj-core:3.8.0",
+ *mavenResolver("util")
+ )
}
assemble {
@@ -184,17 +217,27 @@ val kobaltApp = project(kobaltPluginApi, wrapper) {
}
zip {
val dir = "kobalt-$version"
- include(from("dist"), to("$dir/bin"), "kobaltw")
- include(from("dist"), to("$dir/bin"), "kobaltw.bat")
- include(from("$buildDirectory/libs"), to("$dir/kobalt/wrapper"),
- "$projectName-$version.jar")
- include(from("modules/wrapper/$buildDirectory/libs"), to("$dir/kobalt/wrapper"),
- "$projectName-wrapper.jar")
+ val files = listOf(
+ "dist", "$dir/bin", "kobaltw",
+ "dist", "$dir/bin", "kobaltw.bat",
+ "$buildDirectory/libs", "$dir/kobalt/wrapper", "$projectName-$version.jar",
+ "modules/wrapper/$buildDirectory/libs", "$dir/kobalt/wrapper", "$projectName-wrapper.jar")
+
+ (0 .. files.size - 1 step 3).forEach { i ->
+ include(from(files[i]), to(files[i + 1]), files[i + 2])
+ }
+
+ // Package the sources
+ val currentDir = Paths.get(".").toAbsolutePath().normalize().toString()
+ zipFolders("$currentDir/$buildDirectory/libs/all-sources/$projectName-$version-sources.jar",
+ "$currentDir/$directory/src/main/kotlin",
+ "$currentDir/${kobaltPluginApi.directory}/src/main/kotlin")
+ include(from("$buildDirectory/libs/all-sources"), to("$dir/kobalt/wrapper"), "$projectName-$version-sources.jar")
}
}
kotlinCompiler {
- args("-nowarn")
+ args("nowarn")
}
bintray {
@@ -208,6 +251,32 @@ val kobaltApp = project(kobaltPluginApi, wrapper) {
test {
args("-log", "2", "src/test/resources/testng.xml")
}
+
+ autoGitTag {
+ enabled = true
+ }
+}
+
+fun zipFolders(zipFilePath: String, vararg foldersPath: String) {
+ val zip = Paths.get(zipFilePath)
+ Files.deleteIfExists(zip)
+ Files.createDirectories(zip.parent)
+ val zipPath = Files.createFile(zip)
+ ZipOutputStream(Files.newOutputStream(zipPath)).use {
+ foldersPath.map {Paths.get(it)}.forEach { folderPath ->
+ Files.walk(folderPath)
+ .filter { path -> !Files.isDirectory(path) }
+ .forEach { path ->
+ val zipEntry = ZipEntry(folderPath.relativize(path).toString())
+ try {
+ it.putNextEntry(zipEntry)
+ Files.copy(path, it)
+ it.closeEntry()
+ } catch (e: Exception) {
+ }
+ }
+ }
+ }
}
fun readVersion() : String {
@@ -220,16 +289,38 @@ fun readVersion() : String {
}
}
-@Task(name = "copyVersionForWrapper", reverseDependsOn = arrayOf("assemble"))
+@Task(name = "copyVersionForWrapper", reverseDependsOn = arrayOf("assemble"), runAfter = arrayOf("clean"))
fun taskCopyVersionForWrapper(project: Project) : TaskResult {
if (project.name == "kobalt-wrapper") {
val toString = "modules/wrapper/kobaltBuild/classes"
File(toString).mkdirs()
val from = Paths.get("src/main/resources/kobalt.properties")
val to = Paths.get("$toString/kobalt.properties")
- Files.copy(from,
- to,
- StandardCopyOption.REPLACE_EXISTING)
+ // Only copy if necessary so we don't break incremental compilation
+ if (! to.toFile().exists() || (from.toFile().readLines() != to.toFile().readLines())) {
+ Files.copy(from,
+ to,
+ StandardCopyOption.REPLACE_EXISTING)
+ }
}
return TaskResult()
}
+
+fun createPom(projectName: String, projectDescription: String) = Model().apply {
+ name = projectName
+ description = projectDescription
+ url = "https://beust.com/kobalt"
+ licenses = listOf(License().apply {
+ name = "Apache-2.0"
+ url = "https://www.apache.org/licenses/LICENSE-2.0"
+ })
+ scm = Scm().apply {
+ url = "https://github.com/cbeust/kobalt"
+ connection = "https://github.com/cbeust/kobalt.git"
+ developerConnection = "git@github.com:cbeust/kobalt.git"
+ }
+ developers = listOf(Developer().apply {
+ name = "Cedric Beust"
+ email = "cedric@beust.com"
+ })
+}
diff --git a/kobalt/wrapper/kobalt-wrapper.jar b/kobalt/wrapper/kobalt-wrapper.jar
index 6ca6ca57..848fb463 100644
Binary files a/kobalt/wrapper/kobalt-wrapper.jar and b/kobalt/wrapper/kobalt-wrapper.jar differ
diff --git a/kobalt/wrapper/kobalt-wrapper.properties b/kobalt/wrapper/kobalt-wrapper.properties
index a936b53d..0ca8045f 100644
--- a/kobalt/wrapper/kobalt-wrapper.properties
+++ b/kobalt/wrapper/kobalt-wrapper.properties
@@ -1 +1 @@
-kobalt.version=0.881
\ No newline at end of file
+kobalt.version=1.0.122
\ No newline at end of file
diff --git a/kobaltw b/kobaltw
index b27b3d89..c5186d5a 100755
--- a/kobaltw
+++ b/kobaltw
@@ -1 +1,2 @@
-java -jar $(dirname $0)/kobalt/wrapper/kobalt-wrapper.jar $*
+#!/usr/bin/env sh
+java -jar "`dirname "$0"`/kobalt/wrapper/kobalt-wrapper.jar" $*
diff --git a/kobaltw-test b/kobaltw-test
new file mode 100755
index 00000000..2693c3aa
--- /dev/null
+++ b/kobaltw-test
@@ -0,0 +1,8 @@
+#!/usr/bin/env sh
+JAR=$(ls -1 -t kobaltBuild/libs/*.jar | grep -Ev "(sources|javadoc)" | head -1)
+TEMPDIR=$(mktemp -d)
+cp -pf "$JAR" "$TEMPDIR"
+TEMPJAR=$TEMPDIR/$(basename "$JAR")
+export KOBALT_JAR=$TEMPJAR
+java -jar "$TEMPJAR" "$@"
+rm -rf "$TEMPDIR"
\ No newline at end of file
diff --git a/kobaltw.bat b/kobaltw.bat
index d5780716..2887a567 100644
--- a/kobaltw.bat
+++ b/kobaltw.bat
@@ -1,4 +1,4 @@
-@echo off
-set DIRNAME=%~dp0
-if "%DIRNAME%" == "" set DIRNAME=.
-java -jar "%DIRNAME%/kobalt/wrapper/kobalt-wrapper.jar" %*
+@echo off
+set DIRNAME=%~dp0
+if "%DIRNAME%" == "" set DIRNAME=.
+java -jar "%DIRNAME%/kobalt/wrapper/kobalt-wrapper.jar" %*
diff --git a/lib/kotlin-reflect.jar b/lib/kotlin-reflect.jar
deleted file mode 100644
index 62340e2d..00000000
Binary files a/lib/kotlin-reflect.jar and /dev/null differ
diff --git a/lib/kotlin-runtime-sources.jar b/lib/kotlin-runtime-sources.jar
deleted file mode 100644
index e661b300..00000000
Binary files a/lib/kotlin-runtime-sources.jar and /dev/null differ
diff --git a/lib/kotlin-runtime.jar b/lib/kotlin-runtime.jar
deleted file mode 100644
index 4d7a9270..00000000
Binary files a/lib/kotlin-runtime.jar and /dev/null differ
diff --git a/modules/kobalt-plugin-api/build.gradle b/modules/kobalt-plugin-api/build.gradle
new file mode 100644
index 00000000..56085220
--- /dev/null
+++ b/modules/kobalt-plugin-api/build.gradle
@@ -0,0 +1,85 @@
+plugins {
+ id 'org.jetbrains.kotlin.jvm' version '1.2.71'
+ id 'com.github.johnrengelman.shadow' version '5.0.0'
+}
+
+dependencies {
+ implementation "biz.aQute.bnd:biz.aQute.bndlib:$bndlib"
+ implementation "com.google.code.findbugs:jsr305:$findbugs"
+ implementation "com.sparkjava:spark-core:$spark"
+ implementation "com.squareup.okhttp3:logging-interceptor:$okhttp"
+ implementation 'commons-io:commons-io:2.6'
+ implementation 'io.reactivex:rxjava:1.3.3'
+ implementation "javax.inject:javax.inject:$inject"
+ implementation "javax.xml.bind:jaxb-api:$jaxb"
+ implementation 'org.apache.commons:commons-compress:1.15'
+ implementation 'org.apache.maven:maven-aether-provider:3.3.9'
+ implementation "org.apache.maven.resolver:maven-resolver-api:$mavenResolver"
+ implementation "org.apache.maven.resolver:maven-resolver-connector-basic:$mavenResolver"
+ implementation "org.apache.maven.resolver:maven-resolver-impl:$mavenResolver"
+ implementation "org.apache.maven.resolver:maven-resolver-spi:$mavenResolver"
+ implementation "org.apache.maven.resolver:maven-resolver-transport-file:$mavenResolver"
+ implementation "org.apache.maven.resolver:maven-resolver-transport-http:$mavenResolver"
+ implementation "org.apache.maven.resolver:maven-resolver-util:$mavenResolver"
+ implementation "org.codehaus.groovy:groovy:$groovy"
+ implementation 'org.eclipse.jgit:org.eclipse.jgit:4.9.0.201710071750-r'
+ implementation "org.junit.jupiter:junit-jupiter-engine:$junitJupiter"
+ implementation "org.junit.platform:junit-platform-console:$junitPlatform"
+ implementation "org.junit.platform:junit-platform-engine:$junitPlatform"
+ implementation "org.junit.platform:junit-platform-runner:$junitPlatform"
+ implementation "org.junit.platform:junit-platform-surefire-provider:$junitPlatform"
+ implementation "org.junit.vintage:junit-vintage-engine:$junitJupiter"
+ implementation "org.slf4j:slf4j-simple:$slf4j"
+ implementation "org.testng:testng:$testng"
+ implementation 'org.testng.testng-remote:testng-remote:1.3.2'
+ implementation "com.beust:jcommander:$jcommander"
+ implementation "com.google.code.gson:gson:$gson"
+ implementation "com.google.inject:guice:$guice"
+ implementation "com.google.inject.extensions:guice-assistedinject:$guice"
+ implementation "com.squareup.okio:okio:$okio"
+ implementation "com.squareup.retrofit2:converter-gson:$retrofit"
+ implementation "com.squareup.retrofit2:retrofit:$retrofit"
+ implementation "org.apache.maven:maven-model:$maven"
+ implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin"
+}
+
+shadowJar {
+ classifier = null
+}
+
+test {
+ useTestNG()
+}
+
+publishing {
+ publications {
+ shadow(MavenPublication) { publication ->
+ project.shadow.component(publication)
+ artifact sourcesJar
+ artifact javadocJar
+
+ pom {
+ name = project.name
+ description = 'A build system in Kotlin'
+ url = 'https://beust.com/kobalt'
+ licenses {
+ license {
+ name = 'Apache-2.0'
+ url = 'https://www.apache.org/licenses/LICENSE-2.0'
+ }
+ }
+ developers {
+ developer {
+ name = 'Cedric Beust'
+ email = 'cedric@beust.com'
+ }
+ }
+ scm {
+ connection = 'scm:https://github.com/cbeust/kobalt.git'
+ developerConnection = 'scm:git@github.com:cbeust/kobalt.git'
+ url = 'https://github.com/cbeust/kobalt'
+ }
+ }
+ }
+ }
+}
diff --git a/modules/kobalt-plugin-api/kobalt-plugin-api.iml b/modules/kobalt-plugin-api/kobalt-plugin-api.iml
deleted file mode 100644
index c9fd390e..00000000
--- a/modules/kobalt-plugin-api/kobalt-plugin-api.iml
+++ /dev/null
@@ -1,428 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
\ No newline at end of file
diff --git a/modules/kobalt-plugin-api/pom.xml b/modules/kobalt-plugin-api/pom.xml
new file mode 100644
index 00000000..f9026387
--- /dev/null
+++ b/modules/kobalt-plugin-api/pom.xml
@@ -0,0 +1,279 @@
+
+ 4.0.0
+
+ com.beust
+ kobalt-pom
+ 1.1.0
+ ../..
+
+
+ kobalt-plugin-api
+ jar
+ 1.1.0
+
+
+
+ org.jetbrains.kotlin
+ kotlin-stdlib
+ ${kotlin.version}
+
+
+ org.apache.maven
+ maven-aether-provider
+ 3.3.9
+
+
+ org.eclipse.aether
+ impl
+
+
+ org.eclipse.aether
+ spi
+
+
+ org.eclipse.aether
+ util
+
+
+ org.eclipse.aether
+ api
+
+
+
+
+ org.apache.maven.resolver
+ maven-resolver-api
+ ${mavenresolver.version}
+
+
+ org.apache.maven.resolver
+ maven-resolver-spi
+ ${mavenresolver.version}
+
+
+ org.apache.maven.resolver
+ maven-resolver-util
+ ${mavenresolver.version}
+
+
+ org.apache.maven.resolver
+ maven-resolver-impl
+ ${mavenresolver.version}
+
+
+ org.apache.maven.resolver
+ maven-resolver-connector-basic
+ ${mavenresolver.version}
+
+
+ org.apache.maven.resolver
+ maven-resolver-transport-http
+ ${mavenresolver.version}
+
+
+ org.apache.maven.resolver
+ maven-resolver-transport-file
+ ${mavenresolver.version}
+
+
+ io.reactivex
+ rxjava
+ 1.3.3
+
+
+ com.squareup.okio
+ okio
+ ${okio.version}
+
+
+ javax.inject
+ javax.inject
+ 1
+ compile
+
+
+ com.google.inject
+ guice
+ 4.2.2
+
+
+ com.google.inject.extensions
+ guice-assistedinject
+ 4.2.2
+
+
+ com.beust
+ jcommander
+ 1.72
+
+
+ org.apache.maven
+ maven-model
+ 3.5.2
+
+
+ com.google.code.findbugs
+ jsr305
+ 3.0.2
+
+
+ com.google.code.gson
+ gson
+ 2.8.2
+
+
+ com.squareup.retrofit2
+ retrofit
+ 2.3.0
+
+
+ com.squareup.retrofit2
+ converter-gson
+ 2.3.0
+
+
+ biz.aQute.bnd
+ biz.aQute.bndlib
+ 3.5.0
+
+
+ com.squareup.okhttp3
+ logging-interceptor
+ ${okhttp3.version}
+
+
+ com.sparkjava
+ spark-core
+ 2.6.0
+
+
+ org.codehaus.groovy
+ groovy
+ 2.4.12
+
+
+ org.apache.commons
+ commons-compress
+ 1.15
+
+
+ commons-io
+ commons-io
+ 2.6
+
+
+ org.junit.platform
+ junit-platform-surefire-provider
+ ${junit.version}
+
+
+ org.junit.platform
+ junit-platform-runner
+ ${junit.version}
+
+
+ org.junit.platform
+ junit-platform-engine
+ ${junit.version}
+
+
+ org.junit.platform
+ junit-platform-console
+ ${junit.version}
+
+
+ org.junit.jupiter
+ junit-jupiter-engine
+ ${junitJupiter.version}
+
+
+ org.junit.vintage
+ junit-vintage-engine
+ ${junitJupiter.version}
+
+
+ org.testng.testng-remote
+ testng-remote
+ 1.3.2
+
+
+ org.testng
+ testng
+ ${testng.version}
+
+
+ org.eclipse.jgit
+ org.eclipse.jgit
+ 4.9.0.201710071750-r
+
+
+ org.slf4j
+ slf4j-simple
+ ${slf4j.version}
+
+
+
+ javax.xml.bind
+ jaxb-api
+ 2.3.0
+
+
+
+
+
+
+ org.jetbrains.kotlin
+ kotlin-maven-plugin
+ ${kotlin.version}
+
+
+ compile
+ compile
+
+
+ ${project.basedir}/src/main/kotlin
+
+
+
+
+ test-compile
+ test-compile
+
+
+ ${project.basedir}/src/test/kotlin
+
+
+
+
+
+
+ org.apache.maven.plugins
+ maven-compiler-plugin
+ 3.5.1
+
+
+
+ default-compile
+ none
+
+
+
+ default-testCompile
+ none
+
+
+ java-compile
+ compile
+ compile
+
+
+ java-test-compile
+ test-compile
+ testCompile
+
+
+
+
+
+
\ No newline at end of file
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/ArchiveGenerator.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/ArchiveGenerator.kt
new file mode 100644
index 00000000..8158c642
--- /dev/null
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/ArchiveGenerator.kt
@@ -0,0 +1,21 @@
+package com.beust.kobalt
+
+import com.beust.kobalt.api.KobaltContext
+import com.beust.kobalt.api.Project
+import com.beust.kobalt.archive.Zip
+import com.beust.kobalt.misc.KFiles
+import java.io.File
+
+interface ArchiveGenerator {
+ fun findIncludedFiles(project: Project, context: KobaltContext, zip: Zip) : List
+ val suffix: String
+ fun generateArchive(project: Project, context: KobaltContext, zip: Zip, files: List) : File
+
+ fun fullArchiveName(project: Project, context: KobaltContext, archiveName: String?) : File {
+ val fullArchiveName = context.variant.archiveName(project, archiveName, suffix)
+ val archiveDir = File(KFiles.libsDir(project))
+ val result = File(archiveDir.path, fullArchiveName)
+ return result
+ }
+
+}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Args.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Args.kt
index f43a85b6..372f1ba1 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Args.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Args.kt
@@ -7,7 +7,7 @@ class Args {
var targets: List = arrayListOf()
@Parameter(names = arrayOf("-bf", "--buildFile"), description = "The build file")
- var buildFile: String? = null
+ var buildFile: String? = "kobalt/src/Build.kt"
@Parameter(names = arrayOf("--checkVersions"), description = "Check if there are any newer versions of the " +
"dependencies")
@@ -22,6 +22,10 @@ class Args {
@Parameter(names = arrayOf("--download"), description = "Force a download from the downloadUrl in the wrapper")
var download: Boolean = false
+ @Parameter(names = arrayOf("--downloadSources"),
+ description = "Force a download of sources and javadocs when resolving dependencies")
+ var downloadSources: Boolean = false
+
@Parameter(names = arrayOf("--dryRun"), description = "Display all the tasks that will get run without " +
"actually running them")
var dryRun: Boolean = false
@@ -43,9 +47,13 @@ class Args {
var listTemplates: Boolean = false
@Parameter(names = arrayOf("--log"), description = "Define the log level " +
- "(${Constants.LOG_DEFAULT_LEVEL}-${Constants.LOG_MAX_LEVEL})")
+ "(${Constants.LOG_QUIET_LEVEL}-${Constants.LOG_MAX_LEVEL})")
var log: Int = Constants.LOG_DEFAULT_LEVEL
+ @Parameter(names = arrayOf("--logTags"),
+ description = "Comma-separated list of tags to enable logging for")
+ var logTags: String = ""
+
@Parameter(names = arrayOf("--forceIncremental"),
description = "Force the build to be incremental even if the build file was modified")
var forceIncremental: Boolean = false
@@ -53,8 +61,8 @@ class Args {
@Parameter(names = arrayOf("--noIncremental"), description = "Turn off incremental builds")
var noIncremental: Boolean = false
- @Parameter(names = arrayOf("--parallel"), description = "Build all the projects in parallel whenever possible")
- var parallel: Boolean = false
+ @Parameter(names = arrayOf("--offline"), description = "Don't try to download dependencies even if there is no cached version")
+ var offline: Boolean = false
@Parameter(names = arrayOf("--plugins"), description = "Comma-separated list of plug-in Maven id's")
var pluginIds: String? = null
@@ -72,12 +80,22 @@ class Args {
var profiling: Boolean = false
@Parameter(names = arrayOf("--resolve"),
- description = "Resolve the given comma-separated dependencies and display their dependency tree")
- var dependencies: String? = null
+ description = "Resolve the given dependency and display its tree")
+ var dependency: String? = null
@Parameter(names = arrayOf("--projectInfo"), description = "Display information about the current projects")
var projectInfo: Boolean = false
+ @Parameter(names = arrayOf("--noIncrementalKotlin"), description = "Disable incremental Kotlin compilation")
+ var noIncrementalKotlin: Boolean = false
+
+ companion object {
+ const val SEQUENTIAL = "--sequential"
+ }
+
+ @Parameter(names = arrayOf(Args.SEQUENTIAL), description = "Build all the projects in sequence")
+ var sequential: Boolean = false
+
@Parameter(names = arrayOf("--server"), description = "Run in server mode")
var serverMode: Boolean = false
@@ -86,5 +104,8 @@ class Args {
@Parameter(names = arrayOf("--update"), description = "Update to the latest version of Kobalt")
var update: Boolean = false
+
+ @Parameter(names = arrayOf("--version"), description = "Display the current version of Kobalt")
+ var version: Boolean = false
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/AsciiArt.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/AsciiArt.kt
index fdd2b372..e138fabc 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/AsciiArt.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/AsciiArt.kt
@@ -1,6 +1,5 @@
package com.beust.kobalt
-import com.beust.kobalt.misc.log
import java.util.*
/**
@@ -18,16 +17,16 @@ class AsciiArt {
companion object {
private val BANNERS = arrayOf(
" __ __ __ __ __ \n" +
- " / //_/ ____ / /_ ____ _ / / / /_\n" +
- " / ,< / __ \\ / __ \\ / __ `/ / / / __/\n" +
- " / /| | / /_/ / / /_/ // /_/ / / / / /_ \n" +
- " /_/ |_| \\____/ /_.___/ \\__,_/ /_/ \\__/ ",
+ " / //_/ ____ / /_ ____ _ / / / /_\n" +
+ " / ,< / __ \\ / __ \\ / __ `/ / / / __/\n" +
+ " / /| | / /_/ / / /_/ // /_/ / / / / /_ \n" +
+ " /_/ |_| \\____/ /_.___/ \\__,_/ /_/ \\__/ ",
" _ __ _ _ _ \n" +
- " | |/ / ___ | |__ __ _ | | | |_ \n" +
- " | ' / / _ \\ | '_ \\ / _` | | | | __|\n" +
- " | . \\ | (_) | | |_) | | (_| | | | | |_ \n" +
- " |_|\\_\\ \\___/ |_.__/ \\__,_| |_| \\__| "
+ " | |/ / ___ | |__ __ _ | | | |_ \n" +
+ " | ' / / _ \\ | '_ \\ / _` | | | | __|\n" +
+ " | . \\ | (_) | | |_) | | (_| | | | | |_ \n" +
+ " |_|\\_\\ \\___/ |_.__/ \\__,_| |_| \\__| "
)
val banner : String get() = BANNERS[Random().nextInt(BANNERS.size)]
@@ -75,20 +74,20 @@ class AsciiArt {
return result
}
- private fun fill(n: Int) = StringBuffer().apply { repeat(n, { append(" ")})}.toString()
-
- val defaultLog : (s: String) -> Unit = { log(1, " $it") }
-
- fun logBox(strings: List, bl: String = bottomLeft, br: String = bottomRight,
- print: (String) -> Unit = defaultLog) {
- box(strings, bl, br).forEach {
- print(it)
+ fun logBox(strings: List, bl: String = bottomLeft, br: String = bottomRight, indent: Int = 0): String {
+ return buildString {
+ val boxLines = box(strings, bl, br)
+ boxLines.withIndex().forEach { iv ->
+ append(fill(indent)).append(iv.value)
+ if (iv.index < boxLines.size - 1) append("\n")
+ }
}
}
- fun logBox(s: String, bl: String = bottomLeft, br: String = bottomRight, print: (String) -> Unit = defaultLog) {
- logBox(listOf(s), bl, br, print)
- }
+ fun logBox(s: String, bl: String = bottomLeft, br: String = bottomRight, indent: Int = 0)
+ = logBox(listOf(s), bl, br, indent)
+
+ fun fill(n: Int) = buildString { repeat(n, { append(" ")})}.toString()
fun center(s: String, width: Int) : String {
val diff = width - s.length
@@ -106,7 +105,7 @@ class AsciiArt {
const val CYAN = "\u001B[36m"
const val WHITE = "\u001B[37m"
- private fun wrap(s: CharSequence, color: String) = color + s + RESET
+ fun wrap(s: CharSequence, color: String) = color + s + RESET
private fun blue(s: CharSequence) = wrap(s, BLUE)
private fun red(s: CharSequence) = wrap(s, RED)
private fun yellow(s: CharSequence) = wrap(s, YELLOW)
@@ -117,3 +116,49 @@ class AsciiArt {
}
}
+class AsciiTable {
+ class Builder {
+ private val headers = arrayListOf()
+ fun header(name: String) = headers.add(name)
+ fun headers(vararg names: String) = headers.addAll(names)
+
+ private val widths = arrayListOf()
+ fun columnWidth(w: Int) : Builder {
+ widths.add(w)
+ return this
+ }
+
+ private val rows = arrayListOf>()
+ fun addRow(row: List) = rows.add(row)
+
+ private fun col(width: Int, s: String) : String {
+ val format = " %1\$-${width.toString()}s"
+ val result = String.format(format, s)
+ return result
+ }
+
+ val vb = AsciiArt.verticalBar
+
+ fun build() : String {
+ val formattedHeaders =
+ headers.mapIndexed { index, s ->
+ val s2 = col(widths[index], s)
+ s2
+ }.joinToString(vb)
+ val result = StringBuffer().apply {
+ append(AsciiArt.logBox(formattedHeaders, AsciiArt.bottomLeft2, AsciiArt.bottomRight2))
+ append("\n")
+ }
+ var lineLength = 0
+ rows.forEachIndexed { _, row ->
+ val formattedRow = row.mapIndexed { i, s -> col(widths[i], s) }.joinToString(vb)
+ val line = "$vb $formattedRow $vb"
+ result.append(line).append("\n")
+ lineLength = line.length
+ }
+ result.append(AsciiArt.lowerBox(lineLength - 4))
+ return result.toString()
+ }
+
+ }
+}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/BuildScript.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/BuildScript.kt
index 7f646774..4c35b9ed 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/BuildScript.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/BuildScript.kt
@@ -7,10 +7,42 @@ import com.beust.kobalt.internal.KobaltSettings
import com.beust.kobalt.internal.PluginInfo
import com.beust.kobalt.maven.DependencyManager
import com.beust.kobalt.maven.dependency.FileDependency
+import com.beust.kobalt.misc.KobaltLogger
import org.eclipse.aether.repository.Proxy
import java.io.File
import java.net.InetSocketAddress
+var BUILD_SCRIPT_CONFIG : BuildScriptConfig? = null
+
+class BuildScriptConfig {
+ /** The list of repos used to locate plug-ins. */
+ @Directive
+ fun repos(vararg r: String) = newRepos(*r)
+
+ /** The list of plug-ins to use for this build file. */
+ @Directive
+ fun plugins(vararg pl: String) = newPlugins(*pl)
+
+ /** The build file classpath. */
+ @Directive
+ fun buildFileClasspath(vararg bfc: String) = newBuildFileClasspath(*bfc)
+
+ /** Options passed to Kobalt */
+ @Directive
+ fun kobaltOptions(vararg options: String) = Kobalt.addKobaltOptions(options)
+
+ /** Where to find additional build files */
+ @Directive
+ fun buildSourceDirs(vararg dirs: String) = Kobalt.addBuildSourceDirs(dirs)
+
+ // The following settings modify the compiler used to compile the build file, which regular users should
+ // probably never need to do. Projects should use kotlinCompiler { compilerVersion } to configure the
+ // Kotin compiler for their source files.
+ var kobaltCompilerVersion : String? = null
+ var kobaltCompilerRepo: String? = null
+ var kobaltCompilerFlags: String? = null
+}
+
@Directive
fun homeDir(vararg dirs: String) : String = SystemProperties.homeDir +
File.separator + dirs.toMutableList().joinToString(File.separator)
@@ -18,13 +50,18 @@ fun homeDir(vararg dirs: String) : String = SystemProperties.homeDir +
@Directive
fun file(file: String) : String = FileDependency.PREFIX_FILE + file
-@Directive
fun plugins(vararg dependency : IClasspathDependency) {
dependency.forEach { Plugins.addDynamicPlugin(it) }
}
-@Directive
fun plugins(vararg dependencies : String) {
+ KobaltLogger.logger.warn("Build.kt",
+ "Invoking plugins() directly is deprecated, use the buildScript{} directive")
+ newPlugins(*dependencies)
+}
+
+@Directive
+fun newPlugins(vararg dependencies : String) {
val factory = Kobalt.INJECTOR.getInstance(DependencyManager::class.java)
dependencies.forEach {
Plugins.addDynamicPlugin(factory.create(it))
@@ -37,7 +74,18 @@ data class ProxyConfig(val host: String = "", val port: Int = 0, val type: Strin
fun toAetherProxy() = Proxy(type, host, port) // TODO make support for proxy auth
}
-data class HostConfig(var url: String = "", var username: String? = null, var password: String? = null) {
+data class HostConfig(var url: String = "", var name: String = HostConfig.createRepoName(url),
+ var username: String? = null, var password: String? = null) {
+
+ companion object {
+ /**
+ * For repos specified in the build file (repos()) that don't have an associated unique name,
+ * create such a name from the URL. This is a requirement from Maven Resolver, and failing to do
+ * this leads to very weird resolution errors.
+ */
+ private fun createRepoName(url: String) = url.replace("/", "_").replace("\\", "_").replace(":", "_")
+ }
+
fun hasAuth() : Boolean {
return (! username.isNullOrBlank()) && (! password.isNullOrBlank())
}
@@ -51,13 +99,24 @@ data class HostConfig(var url: String = "", var username: String? = null, var pa
}
}
-@Directive
fun repos(vararg repos : String) {
+ KobaltLogger.logger.warn("Build.kt",
+ "Invoking repos() directly is deprecated, use the buildScript{} directive")
+ newRepos(*repos)
+}
+
+fun newRepos(vararg repos: String) {
repos.forEach { Kobalt.addRepo(HostConfig(it)) }
}
-@Directive
fun buildFileClasspath(vararg deps: String) {
+ KobaltLogger.logger.warn("Build.kt",
+ "Invoking buildFileClasspath() directly is deprecated, use the buildScript{} directive")
+ newBuildFileClasspath(*deps)
+}
+
+fun newBuildFileClasspath(vararg deps: String) {
+ //FIXME newBuildFileClasspath called twice
deps.forEach { Kobalt.addBuildFileClasspath(it) }
}
@@ -67,7 +126,7 @@ fun authRepos(vararg repos : HostConfig) {
}
@Directive
-fun authRepo(init: HostConfig.() -> Unit) = HostConfig().apply { init() }
+fun authRepo(init: HostConfig.() -> Unit) = HostConfig(name = "").apply { init() }
@Directive
fun glob(g: String) : IFileSpec.GlobSpec = IFileSpec.GlobSpec(g)
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Constants.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Constants.kt
index 530560d9..8eb73c84 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Constants.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Constants.kt
@@ -3,17 +3,19 @@ package com.beust.kobalt
import com.beust.kobalt.misc.KFiles
object Constants {
+ const val LOG_QUIET_LEVEL = 0
const val LOG_DEFAULT_LEVEL = 1
const val LOG_MAX_LEVEL = 3
val BUILD_FILE_NAME = "Build.kt"
val BUILD_FILE_DIRECTORY = "kobalt/src"
val BUILD_FILE_PATH = KFiles.joinDir(BUILD_FILE_DIRECTORY, BUILD_FILE_NAME)
+ val KOTLIN_COMPILER_VERSION = "1.2.70"
- internal val DEFAULT_REPOS = listOf(
+ internal val DEFAULT_REPOS = listOf(
// "https://maven-central.storage.googleapis.com/",
- "http://repo1.maven.org/maven2/",
- "https://jcenter.bintray.com/",
- "http://repository.jetbrains.com/all/"
+ HostConfig("https://repo1.maven.org/maven2/", "Maven"),
+ HostConfig("https://jcenter.bintray.com/", "JCenter")
+// "https://repository.jetbrains.com/all/", // <-- contains snapshots
// snapshots
// "https://oss.sonatype.org/content/repositories/snapshots/"
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Directives.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Directives.kt
index a655325a..93d9434c 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Directives.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Directives.kt
@@ -4,9 +4,11 @@ import com.beust.kobalt.api.Kobalt
import com.beust.kobalt.api.Project
import com.beust.kobalt.api.annotation.Directive
import com.beust.kobalt.internal.JvmCompilerPlugin
+import kotlin.properties.ReadWriteProperty
+import kotlin.reflect.KProperty
@Directive
-public fun project(vararg projects: Project, init: Project.() -> Unit): Project {
+fun project(vararg projects: Project, init: Project.() -> Unit): Project {
return Project("").apply {
init()
(Kobalt.findPlugin(JvmCompilerPlugin.PLUGIN_NAME) as JvmCompilerPlugin)
@@ -14,3 +16,24 @@ public fun project(vararg projects: Project, init: Project.() -> Unit): Project
}
}
+@Directive
+fun buildScript(init: BuildScriptConfig.() -> Unit): BuildScriptConfig {
+ val buildScriptConfig = BuildScriptConfig().apply { init() }
+ BUILD_SCRIPT_CONFIG = buildScriptConfig
+ return buildScriptConfig
+}
+
+@Directive
+fun profile(): ReadWriteProperty {
+ val result = object: ReadWriteProperty {
+ var value: Boolean = false
+ override operator fun getValue(thisRef: Nothing?, property: KProperty<*>): Boolean {
+ return value
+ }
+
+ override operator fun setValue(thisRef: Nothing?, property: KProperty<*>, value: Boolean) {
+ this.value = value
+ }
+ }
+ return result
+}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/FileSpec.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/FileSpec.kt
index 043e2bac..1eb409f4 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/FileSpec.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/FileSpec.kt
@@ -1,6 +1,6 @@
package com.beust.kobalt
-import com.beust.kobalt.misc.log
+import com.beust.kobalt.misc.kobaltLog
import java.io.File
import java.nio.file.*
import java.nio.file.attribute.BasicFileAttributes
@@ -28,15 +28,15 @@ sealed class IFileSpec {
private fun isIncluded(includeMatchers: Glob, excludes: List, rel: Path) : Boolean {
excludes.forEach {
if (it.matches(rel)) {
- log(3, "Excluding ${rel.toFile()}")
+ kobaltLog(3, " Excluding ${rel.toFile()}")
return false
}
}
if (includeMatchers.matches(rel)) {
- log(3, "Including ${rel.toFile().path}")
+ kobaltLog(3, " Including ${rel.toFile().path}")
return true
}
- log(2, "Excluding ${rel.toFile()} (not matching any include pattern")
+ kobaltLog(2, " Excluding ${rel.toFile()} (not matching any include pattern")
return false
}
@@ -56,7 +56,7 @@ sealed class IFileSpec {
val path = p.normalize()
val rel = orgRootDir.relativize(path)
if (isIncluded(includes, excludes, path)) {
- log(3, " including file " + rel.toFile() + " from rootDir $rootDir")
+ kobaltLog(3, " including file " + rel.toFile() + " from rootDir $rootDir")
result.add(rel.toFile())
}
return FileVisitResult.CONTINUE
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/IncludeFromTo.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/IncludeFromTo.kt
new file mode 100644
index 00000000..ea189851
--- /dev/null
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/IncludeFromTo.kt
@@ -0,0 +1,43 @@
+package com.beust.kobalt
+
+import com.beust.kobalt.api.annotation.Directive
+import java.io.File
+
+/**
+ * Base classes for directives that support install(from,to) (e.g. install{} or jar{}).
+ */
+open class IncludeFromTo {
+ /**
+ * Prefix path to be removed from the zip file. For example, if you add "build/lib/a.jar" to the zip
+ * file and the excludePrefix is "build/lib", then "a.jar" will be added at the root of the zip file.
+ */
+ val includedFiles = arrayListOf()
+
+ @Directive
+ fun from(s: String) = From(s)
+
+ @Directive
+ fun to(s: String) = To(s)
+
+ @Directive
+ fun copy(from: From, to: To) {
+ val dir = File(from.path).absoluteFile.parentFile
+ includedFiles.add(IncludedFile(from(dir.absolutePath), to, listOf(IFileSpec.FileSpec(from.path))))
+ }
+
+ @Directive
+ fun include(vararg files: String) {
+ includedFiles.add(IncludedFile(files.map { IFileSpec.FileSpec(it) }))
+ }
+
+ @Directive
+ fun include(from: From, to: To, vararg specs: String) {
+ includedFiles.add(IncludedFile(from, to, specs.map { IFileSpec.FileSpec(it) }))
+ }
+
+ @Directive
+ fun include(from: From, to: To, vararg specs: IFileSpec.GlobSpec) {
+ includedFiles.add(IncludedFile(from, to, listOf(*specs)))
+ }
+}
+
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/IncludedFile.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/IncludedFile.kt
new file mode 100644
index 00000000..46dea15e
--- /dev/null
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/IncludedFile.kt
@@ -0,0 +1,44 @@
+package com.beust.kobalt
+
+import com.beust.kobalt.misc.KFiles
+import com.beust.kobalt.misc.toString
+import java.io.File
+import java.nio.file.Paths
+
+class IncludedFile(val fromOriginal: From, val toOriginal: To, val specs: List,
+ val expandJarFiles: Boolean = false) {
+ constructor(specs: List, expandJarFiles: Boolean = false) : this(From(""), To(""), specs, expandJarFiles)
+ fun from(s: String) = File(if (fromOriginal.isCurrentDir()) s else KFiles.joinDir(from, s))
+ val from: String get() = fromOriginal.path.replace("\\", "/")
+ fun to(s: String) = File(if (toOriginal.isCurrentDir()) s else KFiles.joinDir(to, s))
+ val to: String get() = toOriginal.path.replace("\\", "/")
+ override fun toString() = toString("IncludedFile",
+ "files - ", specs.map { it.toString() },
+ "from", from,
+ "to", to)
+
+ fun allFromFiles(directory: String? = null): List {
+ val result = arrayListOf()
+ specs.forEach { spec ->
+// val fullDir = if (directory == null) from else KFiles.joinDir(directory, from)
+ spec.toFiles(directory, from).forEach { source ->
+ result.add(if (source.isAbsolute) source else File(source.path))
+ }
+ }
+ return result.map { Paths.get(it.path).normalize().toFile()}
+ }
+}
+
+open class Direction(open val p: String) {
+ override fun toString() = path
+ fun isCurrentDir() = path == "./"
+
+ val path: String get() =
+ if (p.isEmpty()) "./"
+ else if (p.startsWith("/") || p.endsWith("/")) p
+ else p + "/"
+}
+
+class From(override val p: String) : Direction(p)
+
+class To(override val p: String) : Direction(p)
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/JarGenerator.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/JarGenerator.kt
index 54f531bd..19bb52c6 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/JarGenerator.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/JarGenerator.kt
@@ -3,21 +3,22 @@ package com.beust.kobalt
import com.beust.kobalt.api.KobaltContext
import com.beust.kobalt.api.Project
import com.beust.kobalt.archive.Archives
-import com.beust.kobalt.archive.Jar
+import com.beust.kobalt.archive.MetaArchive
+import com.beust.kobalt.archive.Zip
import com.beust.kobalt.maven.DependencyManager
import com.beust.kobalt.maven.aether.Scope
-import com.beust.kobalt.misc.*
+import com.beust.kobalt.misc.KFiles
+import com.beust.kobalt.misc.kobaltLog
import com.google.inject.Inject
import java.io.File
import java.io.FileInputStream
-import java.io.OutputStream
import java.nio.file.Paths
-import java.util.jar.JarOutputStream
import java.util.jar.Manifest
-class JarGenerator @Inject constructor(val dependencyManager: DependencyManager) {
+class JarGenerator @Inject constructor(val dependencyManager: DependencyManager) : ArchiveGenerator {
companion object {
- fun findIncludedFiles(directory: String, files: List, excludes: List)
+ fun findIncludedFiles(directory: String, files: List, excludes: List,
+ throwOnError: Boolean = true)
: List {
val result = arrayListOf()
files.forEach { includedFile ->
@@ -27,7 +28,7 @@ class JarGenerator @Inject constructor(val dependencyManager: DependencyManager)
if (File(directory, fromPath).exists()) {
spec.toFiles(directory, fromPath).forEach { file ->
val fullFile = File(KFiles.joinDir(directory, fromPath, file.path))
- if (! fullFile.exists()) {
+ if (! fullFile.exists() && throwOnError) {
throw AssertionError("File should exist: $fullFile")
}
@@ -35,16 +36,16 @@ class JarGenerator @Inject constructor(val dependencyManager: DependencyManager)
val normalized = Paths.get(file.path).normalize().toFile().path
includedSpecs.add(IFileSpec.FileSpec(normalized))
} else {
- log(2, "Not adding ${file.path} to jar file because it's excluded")
+ kobaltLog(2, "Not adding ${file.path} to jar file because it's excluded")
}
}
} else {
- log(2, "Directory $fromPath doesn't exist, not including it in the jar")
+ kobaltLog(2, " Directory $fromPath doesn't exist, not including it in the jar")
}
}
if (includedSpecs.size > 0) {
- log(3, "Including specs $includedSpecs")
+ kobaltLog(3, "Including specs $includedSpecs")
result.add(IncludedFile(From(includedFile.from), To(includedFile.to), includedSpecs))
}
}
@@ -52,7 +53,9 @@ class JarGenerator @Inject constructor(val dependencyManager: DependencyManager)
}
}
- fun findIncludedFiles(project: Project, context: KobaltContext, jar: Jar) : List {
+ override val suffix = ".jar"
+
+ override fun findIncludedFiles(project: Project, context: KobaltContext, zip: Zip) : List {
//
// Add all the applicable files for the current project
//
@@ -60,7 +63,7 @@ class JarGenerator @Inject constructor(val dependencyManager: DependencyManager)
val result = arrayListOf()
val classesDir = KFiles.makeDir(buildDir.path, "classes")
- if (jar.includedFiles.isEmpty()) {
+ if (zip.includedFiles.isEmpty()) {
// If no includes were specified, assume the user wants a simple jar file made of the
// classes of the project, so we specify a From("build/classes/"), To("") and
// a list of files containing everything under it
@@ -70,7 +73,7 @@ class JarGenerator @Inject constructor(val dependencyManager: DependencyManager)
// Class files
val files = KFiles.findRecursively(classesDir).map { File(relClassesDir.toFile(), it) }
val filesNotExcluded : List = files.filter {
- ! KFiles.Companion.isExcluded(KFiles.joinDir(project.directory, it.path), jar.excludes)
+ ! KFiles.Companion.isExcluded(KFiles.joinDir(project.directory, it.path), zip.excludes)
}
val fileSpecs = arrayListOf()
filesNotExcluded.forEach {
@@ -86,16 +89,14 @@ class JarGenerator @Inject constructor(val dependencyManager: DependencyManager)
//
// The user specified an include, just use it verbatim
//
- val includedFiles = findIncludedFiles(project.directory, jar.includedFiles, jar.excludes)
+ val includedFiles = findIncludedFiles(project.directory, zip.includedFiles, zip.excludes, false)
result.addAll(includedFiles)
}
//
// If fatJar is true, add all the transitive dependencies as well: compile, runtime and dependent projects
//
- if (jar.fatJar) {
- log(2, "Finding included files for fat jar")
-
+ if (zip.fatJar) {
val seen = hashSetOf()
@Suppress("UNCHECKED_CAST")
val allDependencies = project.compileDependencies + project.compileRuntimeDependencies +
@@ -104,13 +105,13 @@ class JarGenerator @Inject constructor(val dependencyManager: DependencyManager)
context.variant.productFlavor.compileDependencies +
context.variant.productFlavor.compileRuntimeDependencies
val transitiveDependencies = dependencyManager.calculateDependencies(project, context,
- listOf(Scope.COMPILE), allDependencies)
+ scopes = listOf(Scope.COMPILE), passedDependencies = allDependencies)
transitiveDependencies.map {
it.jarFile.get()
}.forEach { file : File ->
if (! seen.contains(file.path)) {
seen.add(file.path)
- if (! KFiles.Companion.isExcluded(file, jar.excludes)) {
+ if (! KFiles.Companion.isExcluded(file, zip.excludes)) {
result.add(IncludedFile(specs = arrayListOf(IFileSpec.FileSpec(file.absolutePath)),
expandJarFiles = true))
}
@@ -121,19 +122,18 @@ class JarGenerator @Inject constructor(val dependencyManager: DependencyManager)
return result
}
- fun generateJar(project: Project, context: KobaltContext, jar: Jar) : File {
- val includedFiles = findIncludedFiles(project, context, jar)
-
+ override fun generateArchive(project: Project, context: KobaltContext, zip: Zip,
+ includedFiles: List) : File {
//
// Generate the manifest
// If manifest attributes were specified in the build file, use those to generateAndSave the manifest. Otherwise,
// try to find a META-INF/MANIFEST.MF and use that one if we find any. Otherwise, use the default manifest.
//
val manifest =
- if (jar.attributes.size > 1) {
- log(2, "Creating MANIFEST.MF from " + jar.attributes.size + " attributes")
+ if (zip.attributes.size > 1) {
+ context.logger.log(project.name, 2, "Creating MANIFEST.MF from " + zip.attributes.size + " attributes")
Manifest().apply {
- jar.attributes.forEach { attribute ->
+ zip.attributes.forEach { attribute ->
mainAttributes.putValue(attribute.first, attribute.second)
}
}
@@ -142,23 +142,21 @@ class JarGenerator @Inject constructor(val dependencyManager: DependencyManager)
val allFiles = includedFiles.flatMap { file ->
file.allFromFiles(project.directory).map { file.from(it.path) }
}
- val manifestFiles = allFiles.filter { it.path.contains("META-INF/MANIFEST.MF") }
+ val manifestFiles = allFiles.filter { it.path.contains(MetaArchive.MANIFEST_MF) }
return if (manifestFiles.any()) manifestFiles[0] else null
}
val manifestFile = findManifestFile(project, includedFiles)
if (manifestFile != null) {
- log(2, "Including MANIFEST.MF file $manifestFile")
+ context.logger.log(project.name, 2, "Including MANIFEST.MF file $manifestFile")
Manifest(FileInputStream(manifestFile))
} else {
- Manifest()
+ null
}
}
- val jarFactory = { os: OutputStream -> JarOutputStream(os, manifest) }
-
- return Archives.generateArchive(project, context, jar.name, ".jar", includedFiles,
- true /* expandJarFiles */, jarFactory)
+ return Archives.generateArchive(project, context, zip.name, ".jar", includedFiles,
+ true /* expandJarFiles */, manifest)
}
}
\ No newline at end of file
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/JavaInfo.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/JavaInfo.kt
index 53c4e010..3c957454 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/JavaInfo.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/JavaInfo.kt
@@ -2,18 +2,18 @@ package com.beust.kobalt
import java.io.File
-abstract public class JavaInfo {
- public var javaExecutable: File? = null
+abstract class JavaInfo {
+ val javaExecutable: File?
get() = findExecutable("java")
- public var javacExecutable: File? = null
+ val javacExecutable: File?
get() = findExecutable("javac")
- public var javadocExecutable: File? = null
+ val javadocExecutable: File?
get() = findExecutable("javadoc")
- abstract public var javaHome: File?
- abstract public var runtimeJar: File?
- abstract public var toolsJar: File?
+ abstract var javaHome: File?
+ abstract var runtimeJar: File?
+ abstract var toolsJar: File?
- abstract public fun findExecutable(command: String) : File
+ abstract fun findExecutable(command: String) : File
companion object {
fun create(javaBase: File?): Jvm {
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Jvm.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Jvm.kt
index 7dda6015..14c55efd 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Jvm.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Jvm.kt
@@ -1,18 +1,18 @@
package com.beust.kobalt
-import com.beust.kobalt.misc.log
+import com.beust.kobalt.misc.kobaltLog
import com.beust.kobalt.misc.warn
import java.io.File
import java.io.IOException
-public open class Jvm constructor(
+open class Jvm constructor(
val os: OperatingSystem,
var javaBase: File? = null) : JavaInfo() {
private var _javaHome: File? = null
- override public var javaHome: File? = null
+ override var javaHome: File? = null
get() = _javaHome!!
- override public var runtimeJar: File? = null
+ override var runtimeJar: File? = null
private fun findRuntimeJar() : File? {
var runtimeJar = File(javaBase, "lib/rt.jar")
if (runtimeJar.exists()) {
@@ -21,7 +21,7 @@ public open class Jvm constructor(
runtimeJar = File(javaBase, "jre/lib/rt.jar")
return if (runtimeJar.exists()) runtimeJar else null
}
- override public var toolsJar: File? = null
+ override var toolsJar: File? = null
private var userSupplied: Boolean? = false
private var javaVersion: String? = null
@@ -67,7 +67,7 @@ public open class Jvm constructor(
return toolsJar
}
if (javaHome!!.name.equals("jre", true)) {
- javaHome = javaHome!!.parentFile
+ _javaHome = javaHome!!.parentFile
toolsJar = File(javaHome, "lib/tools.jar")
if (toolsJar.exists()) {
return toolsJar
@@ -78,7 +78,7 @@ public open class Jvm constructor(
val version = SystemProperties.Companion.javaVersion
if (javaHome!!.name.toRegex().matches("jre\\d+")
|| javaHome!!.name == "jre$version") {
- javaHome = File(javaHome!!.parentFile, "jdk$version")
+ _javaHome = File(javaHome!!.parentFile, "jdk$version")
toolsJar = File(javaHome, "lib/tools.jar")
if (toolsJar.exists()) {
return toolsJar
@@ -89,15 +89,15 @@ public open class Jvm constructor(
return null
}
-// open public fun isIbmJvm(): Boolean {
+// open fun isIbmJvm(): Boolean {
// return false
// }
- override public fun findExecutable(command: String): File {
+ override fun findExecutable(command: String): File {
if (javaHome != null) {
val jdkHome = if (javaHome!!.endsWith("jre")) javaHome!!.parentFile else javaHome
val exec = File(jdkHome, "bin/" + command)
- var executable = File(os.getExecutableName(exec.absolutePath))
+ val executable = File(os.getExecutableName(exec.absolutePath))
if (executable.isFile) {
return executable
}
@@ -110,7 +110,7 @@ public open class Jvm constructor(
val pathExecutable = os.findInPath(command)
if (pathExecutable != null) {
- log(2, "Unable to find the $command executable using home: " +
+ kobaltLog(2, "Unable to find the $command executable using home: " +
"$javaHome but found it on the PATH: $pathExecutable.")
return pathExecutable
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Plugins.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Plugins.kt
index e794ae8d..0102dd8b 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Plugins.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Plugins.kt
@@ -4,14 +4,14 @@ import com.beust.kobalt.api.*
import com.beust.kobalt.api.annotation.IncrementalTask
import com.beust.kobalt.api.annotation.Task
import com.beust.kobalt.internal.IncrementalManager
+import com.beust.kobalt.internal.KobaltSettings
import com.beust.kobalt.internal.PluginInfo
import com.beust.kobalt.internal.TaskManager
import com.beust.kobalt.maven.DependencyManager
-import com.beust.kobalt.maven.LocalRepo
import com.beust.kobalt.misc.JarUtils
import com.beust.kobalt.misc.KFiles
import com.beust.kobalt.misc.KobaltExecutors
-import com.beust.kobalt.misc.log
+import com.beust.kobalt.misc.kobaltLog
import com.google.inject.Provider
import java.io.File
import java.lang.reflect.Method
@@ -26,9 +26,8 @@ import javax.inject.Singleton
class Plugins @Inject constructor (val taskManagerProvider : Provider,
val files: KFiles,
val depManager: DependencyManager,
- val localRepo: LocalRepo,
+ val settings: KobaltSettings,
val executors: KobaltExecutors,
- val pluginInfo: PluginInfo,
val incrementalManagerFactory: IncrementalManager.IFactory,
val taskManager: TaskManager) {
@@ -65,7 +64,7 @@ class Plugins @Inject constructor (val taskManagerProvider : Provider
plugin.apply(project, context)
}
@@ -171,6 +170,9 @@ class Plugins @Inject constructor (val taskManagerProvider : Provider()
+// @Inject
+// lateinit var pluginInfo: PluginInfo
+
fun installPlugins(dependencies: List, scriptClassLoader: ClassLoader) {
val executor = executors.newExecutor("Plugins", 5)
dependencies.forEach {
@@ -191,6 +193,8 @@ class Plugins @Inject constructor (val taskManagerProvider : Provider) = ids.forEach { displayDependenciesFor(it) }
+ fun run(id: String) = displayDependenciesFor(id)
+
+ private fun latestMavenArtifact(group: String, artifactId: String, extension: String = "jar"): DependencyNode {
+ val artifact = DefaultArtifact(group, artifactId, extension, "(0,]")
+ val resolved = aether.resolveRange(artifact)
+ if (resolved != null) {
+ val newArtifact = DefaultArtifact(artifact.groupId, artifact.artifactId, artifact.extension,
+ resolved.highestVersion.toString())
+ val artifactResult = aether.resolve(KobaltMavenResolver.artifactToId(newArtifact), null)
+ return artifactResult.root
+ } else {
+ throw KobaltException("Couldn't find latest artifact for $group:$artifactId")
+ }
+ }
+
+ class PairResult(val dependency: IClasspathDependency, val repoUrl: String)
+
+ fun latestArtifact(group: String, artifactId: String, extension: String = "jar"): PairResult
+ = latestMavenArtifact(group, artifactId, extension).let {
+ PairResult(AetherDependency(it.artifact), "(TBD repo)")
+ }
private fun displayDependenciesFor(id: String) {
val mavenId = MavenId.create(id)
- val resolved =
- if (mavenId.hasVersion) aether.resolve(id)
- else aether.latestArtifact(mavenId.groupId, mavenId.artifactId)
+ val resolved : PairResult =
+ if (mavenId.hasVersion) {
+ val node = aether.resolve(id, filter = Filters.EXCLUDE_OPTIONAL_FILTER)
+ PairResult(AetherDependency(node.root.artifact), node.artifactResults[0].repository.toString())
+ } else {
+ latestArtifact(mavenId.groupId, mavenId.artifactId)
+ }
displayDependencies(resolved.dependency, resolved.repoUrl)
}
@@ -42,10 +66,10 @@ class ResolveDependency @Inject constructor(
val seen = hashSetOf(dep.id)
root.addChildren(findChildren(root, seen))
- AsciiArt.logBox(listOf(dep.id, url, dep.jarFile.get()).map { " $it" }, print = {s -> println(s) })
+ kobaltLog(1, AsciiArt.logBox(listOf(dep.id, url, dep.jarFile.get()).map { " $it" }))
display(root.children)
- println("")
+ kobaltLog(1, "")
}
private fun display(nodes: List>) {
@@ -57,10 +81,12 @@ class ResolveDependency @Inject constructor(
else leftMiddle
val indent = level * increment
for(i in 0..indent - 2) {
- if (i % increment == 0) print(vertical)
- else print(" ")
+ if (!KobaltLogger.isQuiet) {
+ if (i == 0 || ((i + 1) % increment == 0)) print(vertical)
+ else print(" ")
+ }
}
- println(left + " " + dep.id)
+ kobaltLog(1, left + " " + dep.id + (if (dep.optional) " (optional)" else ""))
display(node.children)
}
}
@@ -73,13 +99,18 @@ class ResolveDependency @Inject constructor(
if (! seen.contains(it.id)) {
val dep = Dep(it, root.value.level + 1)
val node = Node(dep)
- log(2, "Found dependency ${dep.dep.id} level: ${dep.level}")
+ kobaltLog(2, "Found dependency ${dep.dep.id} level: ${dep.level}")
result.add(node)
seen.add(it.id)
- node.addChildren(findChildren(node, seen))
+ try {
+ node.addChildren(findChildren(node, seen))
+ } catch(ex: Exception) {
+ if (! it.optional) warn("Couldn't resolve " + node)
+ // else don't warn about missing optional dependencies
+ }
}
}
- log(2, "Children for ${root.value.dep.id}: ${result.size}")
+ kobaltLog(2, "Children for ${root.value.dep.id}: ${result.size}")
return result
}
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/SystemProperties.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/SystemProperties.kt
index b7877264..d5507497 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/SystemProperties.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/SystemProperties.kt
@@ -1,9 +1,17 @@
package com.beust.kobalt
-public class SystemProperties {
+class SystemProperties {
companion object {
- val javaBase = System.getProperty("java.home") ?:
- (System.getenv("JAVA_HOME") ?: throw IllegalArgumentException("JAVA_HOME not defined"))
+ val javaBase : String
+ get() {
+ val jh = System.getenv("JAVA_HOME")
+ ?: System.getProperty("java.home")
+ ?: throw IllegalArgumentException("JAVA_HOME not defined")
+ val result =
+ if (jh.toLowerCase().endsWith("jre")) jh.substring(0, jh.length - 4)
+ else jh
+ return result
+ }
val javaVersion = System.getProperty("java.version")
val homeDir = System.getProperty("user.home")
val tmpDir = System.getProperty("java.io.tmpdir")
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/TaskResult.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/TaskResult.kt
index 4d6a45f3..241bc045 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/TaskResult.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/TaskResult.kt
@@ -1,3 +1,8 @@
package com.beust.kobalt
-open public class TaskResult(val success: Boolean = true, val errorMessage: String? = null)
+class TestResult(val success: Boolean, val shortMessage: String? = null, val longMessage: String? = null)
+
+open class TaskResult(val success: Boolean = true,
+ val testResult: TestResult? = null,
+ val errorMessage: String? = null
+)
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/TestConfig.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/TestConfig.kt
new file mode 100644
index 00000000..f84b094d
--- /dev/null
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/TestConfig.kt
@@ -0,0 +1,40 @@
+package com.beust.kobalt
+
+import com.beust.kobalt.api.Project
+import com.beust.kobalt.api.annotation.Directive
+
+class TestConfig(val project: Project, val isDefault : Boolean = false) {
+ val testArgs = arrayListOf()
+ val jvmArgs = arrayListOf()
+ val testIncludes = arrayListOf("**/*Test.class")
+ val testExcludes = arrayListOf()
+
+ @Directive
+ var name: String = ""
+
+ @Directive
+ fun args(vararg arg: String) {
+ testArgs.addAll(arg)
+ }
+
+ @Directive
+ fun jvmArgs(vararg arg: String) {
+ jvmArgs.addAll(arg)
+ }
+
+ @Directive
+ fun include(vararg arg: String) {
+ testIncludes.apply {
+ clear()
+ addAll(arg)
+ }
+ }
+
+ @Directive
+ fun exclude(vararg arg: String) {
+ testExcludes.apply {
+ clear()
+ addAll(arg)
+ }
+ }
+}
\ No newline at end of file
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/TestDirective.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/TestDirective.kt
index d7f6cb7e..ad026380 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/TestDirective.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/TestDirective.kt
@@ -3,43 +3,13 @@ package com.beust.kobalt
import com.beust.kobalt.api.Project
import com.beust.kobalt.api.annotation.Directive
-class TestConfig(val project: Project, val isDefault : Boolean = false) {
- val testArgs = arrayListOf()
- val jvmArgs = arrayListOf()
- val testIncludes = arrayListOf("**/*Test.class")
- val testExcludes = arrayListOf()
-
- var name: String = ""
-
- fun args(vararg arg: String) {
- testArgs.addAll(arg)
- }
-
- fun jvmArgs(vararg arg: String) {
- jvmArgs.addAll(arg)
- }
-
- fun include(vararg arg: String) {
- testIncludes.apply {
- clear()
- addAll(arg)
- }
- }
-
- fun exclude(vararg arg: String) {
- testExcludes.apply {
- clear()
- addAll(arg)
- }
- }
-}
-
@Directive
-fun Project.test(init: TestConfig.() -> Unit) = let { project ->
+fun Project.test(init: TestConfig.() -> Unit): TestConfig = let { project ->
with(testConfigs) {
val tf = TestConfig(project).apply { init() }
if (! map { it.name }.contains(tf.name)) {
add(tf)
+ tf
} else {
throw KobaltException("Test configuration \"${tf.name}\" already exists, give it a different "
+ "name with test { name = ... }")
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Variant.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Variant.kt
index 2bafde37..13120fa0 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Variant.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/Variant.kt
@@ -2,9 +2,9 @@ package com.beust.kobalt
import com.beust.kobalt.api.*
import com.beust.kobalt.internal.ActorUtils
+import com.beust.kobalt.internal.ParallelLogger
import com.beust.kobalt.internal.SourceSet
import com.beust.kobalt.misc.KFiles
-import com.beust.kobalt.misc.log
import java.io.File
import java.util.*
@@ -29,7 +29,7 @@ class Variant(val initialProductFlavor: ProductFlavorConfig? = null,
/**
* for {internal, release}, return [internalRelease, internal, release]
*/
- fun allDirectories(project: Project): List {
+ fun allDirectories(): List {
return arrayListOf().apply {
add(toCamelcaseDir())
add(productFlavor.name)
@@ -70,28 +70,27 @@ class Variant(val initialProductFlavor: ProductFlavorConfig? = null,
val result = arrayListOf()
val sourceDirectories = sourceSet.correctSourceSet(project)
.filter { File(project.directory, it).exists() }
- .map { File(it) }
+ .map(::File)
if (isDefault) {
result.addAll(sourceDirectories)
} else {
// // The ordering of files is: 1) build type 2) product flavor 3) default
+ val kobaltLog = Kobalt.INJECTOR.getInstance(ParallelLogger::class.java)
buildType.let {
val dir = File(KFiles.joinDir("src", it.name, suffix))
- log(3, "Adding source for build type ${it.name}: ${dir.path}")
+ kobaltLog.log(project.name, 3, "Adding source for build type ${it.name}: ${dir.path}")
result.add(dir)
}
productFlavor.let {
val dir = File(KFiles.joinDir("src", it.name, suffix))
- log(3, "Adding source for product flavor ${it.name}: ${dir.path}")
+ kobaltLog.log(project.name, 3, "Adding source for product flavor ${it.name}: ${dir.path}")
result.add(dir)
}
- result.addAll(allDirectories(project).map {
- File(KFiles.joinDir("src", it, suffix))
- }.filter {
- it.exists()
- })
+ result.addAll(allDirectories()
+ .map { File(KFiles.joinDir("src", it, suffix)) }
+ .filter(File::exists))
// Now that all the variant source directories have been added, add the project's default ones
result.addAll(sourceDirectories)
@@ -114,8 +113,8 @@ class Variant(val initialProductFlavor: ProductFlavorConfig? = null,
if (isDefault) {
archiveName ?: project.name + "-" + project.version + suffix
} else {
- val base = if (archiveName != null) archiveName.substring(0, archiveName.length - suffix.length)
- else project.name + "-" + project.version
+ val base = archiveName?.substring(0, archiveName.length - suffix.length)
+ ?: project.name + "-" + project.version
val flavor = if (productFlavor.name.isEmpty()) "" else "-" + productFlavor.name
val type = if (buildType.name.isEmpty()) "" else "-" + buildType.name
val result: String = base + flavor + type + suffix
@@ -125,18 +124,16 @@ class Variant(val initialProductFlavor: ProductFlavorConfig? = null,
return result
}
- val shortArchiveName = if (isDefault) "" else "-" + productFlavor.name + "-" + buildType.name
-
var generatedSourceDirectory: File? = null
private fun findBuildTypeBuildConfig(project: Project, variant: Variant?) : BuildConfig? {
val buildTypeName = variant?.buildType?.name
- return project.buildTypes[buildTypeName]?.buildConfig ?: null
+ return project.buildTypes[buildTypeName]?.buildConfig
}
private fun findProductFlavorBuildConfig(project: Project, variant: Variant?) : BuildConfig? {
val buildTypeName = variant?.productFlavor?.name
- return project.productFlavors[buildTypeName]?.buildConfig ?: null
+ return project.productFlavors[buildTypeName]?.buildConfig
}
/**
@@ -165,7 +162,8 @@ class Variant(val initialProductFlavor: ProductFlavorConfig? = null,
?: throw KobaltException(
"packageName needs to be defined on the project in order to generateAndSave BuildConfig")
- val contributor = ActorUtils.selectAffinityActor(context.pluginInfo.buildConfigContributors, project)
+ val contributor = ActorUtils.selectAffinityActor(project, context,
+ context.pluginInfo.buildConfigContributors)
if (contributor != null) {
val code = contributor.generateBuildConfig(project, context, pkg, this, buildConfigs)
val result = KFiles.makeDir(KFiles.generatedSourceDir(project, this, "buildConfig"))
@@ -175,7 +173,7 @@ class Variant(val initialProductFlavor: ProductFlavorConfig? = null,
val outputGeneratedSourceDirectory = File(result, pkg.replace('.', File.separatorChar))
val outputDir = File(outputGeneratedSourceDirectory, "BuildConfig." + contributor.buildConfigSuffix)
KFiles.saveFile(outputDir, code)
- log(2, "Generated ${outputDir.path}")
+ context.logger.log(project.name, 2, "Generated ${outputDir.path}")
return result
} else {
throw KobaltException("Couldn't find a contributor to generateAndSave BuildConfig")
@@ -223,7 +221,7 @@ class Variant(val initialProductFlavor: ProductFlavorConfig? = null,
}
fun toCamelcaseDir() : String {
- fun lci(s : String) = if (s.length == 0 || s.length == 1) s else s[0].toLowerCase() + s.substring(1)
+ fun lci(s : String) = if (s.isEmpty() || s.length == 1) s else s[0].toLowerCase() + s.substring(1)
return lci(productFlavor.name) + buildType.name.capitalize()
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/CompilerActionInfo.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/CompilerActionInfo.kt
index f0be2eb5..e323e474 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/CompilerActionInfo.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/CompilerActionInfo.kt
@@ -11,4 +11,6 @@ data class CompilerActionInfo(val directory: String?,
val suffixesBeingCompiled: List,
val outputDir: File,
val compilerArgs: List,
- val friendPaths: List)
+ val friendPaths: List,
+ val forceRecompile: Boolean,
+ val compilerSeparateProcess: Boolean = false)
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/DependencyHolder.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/DependencyHolder.kt
index 2103bdc1..e1195ca3 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/DependencyHolder.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/DependencyHolder.kt
@@ -11,7 +11,9 @@ interface IDependencyHolder {
var project: Project
val compileDependencies : ArrayList
+ val optionalDependencies : ArrayList
val compileProvidedDependencies : ArrayList
+ val compileOnlyDependencies : ArrayList
val compileRuntimeDependencies : ArrayList
val excludedDependencies : ArrayList
val nativeDependencies : ArrayList
@@ -26,7 +28,9 @@ interface IDependencyHolder {
open class DependencyHolder : IDependencyHolder {
override lateinit var project: Project
override val compileDependencies : ArrayList = arrayListOf()
+ override val optionalDependencies : ArrayList = arrayListOf()
override val compileProvidedDependencies : ArrayList = arrayListOf()
+ override val compileOnlyDependencies : ArrayList = arrayListOf()
override val compileRuntimeDependencies : ArrayList = arrayListOf()
override val excludedDependencies : ArrayList = arrayListOf()
override val nativeDependencies : ArrayList = arrayListOf()
@@ -34,8 +38,8 @@ open class DependencyHolder : IDependencyHolder {
override var dependencies : Dependencies? = null
override fun dependencies(init: Dependencies.() -> Unit) : Dependencies {
- dependencies = Dependencies(project, compileDependencies, compileProvidedDependencies,
- compileRuntimeDependencies, excludedDependencies, nativeDependencies)
+ dependencies = Dependencies(project, compileDependencies, optionalDependencies, compileProvidedDependencies,
+ compileOnlyDependencies, compileRuntimeDependencies, excludedDependencies, nativeDependencies)
dependencies!!.init()
return dependencies!!
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IBuildConfigContributor.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IBuildConfigContributor.kt
index 35553f74..ef9d3b4d 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IBuildConfigContributor.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IBuildConfigContributor.kt
@@ -5,7 +5,7 @@ import com.beust.kobalt.Variant
/**
* Plug-ins that can generate a BuildConfig file.
*/
-interface IBuildConfigContributor : ISimpleAffinity {
+interface IBuildConfigContributor : IProjectAffinity {
fun generateBuildConfig(project: Project, context: KobaltContext, packageName: String, variant: Variant,
buildConfigs: List) : String
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IBuildListener.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IBuildListener.kt
index 1e53cc49..2b0fdadb 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IBuildListener.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IBuildListener.kt
@@ -4,8 +4,12 @@ package com.beust.kobalt.api
* Plug-ins that listen to build events.
*/
interface IBuildListener : IListener {
+
+ class TaskEndInfo(val success: Boolean, val shortMessage: String? = null,
+ val longMessage: String? = null)
+
fun taskStart(project: Project, context: KobaltContext, taskName: String) {}
- fun taskEnd(project: Project, context: KobaltContext, taskName: String, success: Boolean) {}
+ fun taskEnd(project: Project, context: KobaltContext, taskName: String, info: TaskEndInfo) {}
fun projectStart(project: Project, context: KobaltContext) {}
fun projectEnd(project: Project, context: KobaltContext, status: ProjectBuildStatus) {}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IClasspathDependency.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IClasspathDependency.kt
index 80d9e998..527e6f13 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IClasspathDependency.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IClasspathDependency.kt
@@ -22,15 +22,20 @@ interface IClasspathDependency {
/** @return true if this dependency represents a Maven coordinate */
val isMaven: Boolean
+ /** @return true if this dependency is optional */
+ val optional: Boolean
+
/** Absolute path to the jar file on the local file system */
val jarFile: Future
/** Convert to a Maven model tag */
- fun toMavenDependencies() : Dependency
+ fun toMavenDependencies(scope: String? = null) : Dependency
/** The list of dependencies for this element (not the transitive closure) */
fun directDependencies(): List
/** Used to only keep the most recent version for an artifact if no version was specified */
val shortId: String
+
+ val excluded: ArrayList
}
\ No newline at end of file
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ICompilerContributor.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ICompilerContributor.kt
index b4788370..b6ce8fd2 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ICompilerContributor.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ICompilerContributor.kt
@@ -1,7 +1,6 @@
package com.beust.kobalt.api
import com.beust.kobalt.TaskResult
-import com.beust.kobalt.misc.warn
interface ICompilerDescription : Comparable {
/**
@@ -55,13 +54,15 @@ class CompilerDescription(override val name: String, override val sourceDirecto
override val canCompileDirectories: Boolean = false) : ICompilerDescription {
override fun compile(project: Project, context: KobaltContext, info: CompilerActionInfo): TaskResult {
val result =
- if (info.sourceFiles.size > 0) {
+ if (info.sourceFiles.isNotEmpty()) {
compiler.compile(project, context, info)
} else {
- warn("Couldn't find any source files to compile")
+ context.logger.log(project.name, 2, "$name couldn't find any source files to compile")
TaskResult()
}
return result
}
+
+ override fun toString() = name + " compiler"
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ICompilerFlagContributor.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ICompilerFlagContributor.kt
index 5962e9fb..2d8febe4 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ICompilerFlagContributor.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ICompilerFlagContributor.kt
@@ -3,13 +3,27 @@ package com.beust.kobalt.api
/**
* Plugins that add compiler flags.
*/
-interface ICompilerFlagContributor : IContributor {
- fun flagsFor(project: Project, context: KobaltContext, currentFlags: List,
- suffixesBeingCompiled: List): List
- val flagPriority: Int
- get() = DEFAULT_FLAG_PRIORITY
-
+class FlagContributor(val flagPriority: Int = DEFAULT_FLAG_PRIORITY,
+ val closure: (project: Project, context: KobaltContext, currentFlags: List,
+ suffixesBeingCompiled: List) -> List) : IContributor {
companion object {
val DEFAULT_FLAG_PRIORITY = 20
}
+
+ fun flagsFor(project: Project, context: KobaltContext, currentFlags: List,
+ suffixesBeingCompiled: List) = closure(project, context, currentFlags, suffixesBeingCompiled)
}
+
+interface IFlagBase {
+ val flagPriority: Int get() = FlagContributor.DEFAULT_FLAG_PRIORITY
+}
+
+interface ICompilerFlagContributor : IContributor, IFlagBase {
+ fun compilerFlagsFor(project: Project, context: KobaltContext, currentFlags: List,
+ suffixesBeingCompiled: List): List
+}
+
+interface IDocFlagContributor : IContributor, IFlagBase {
+ fun docFlagsFor(project: Project, context: KobaltContext, currentFlags: List,
+ suffixesBeingCompiled: List): List
+}
\ No newline at end of file
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IDependencyManager.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IDependencyManager.kt
index ef7fb959..3a66f980 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IDependencyManager.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IDependencyManager.kt
@@ -1,6 +1,11 @@
package com.beust.kobalt.api
+import com.beust.kobalt.maven.aether.Filters.EXCLUDE_OPTIONAL_FILTER
+import com.beust.kobalt.maven.aether.KobaltMavenResolver
import com.beust.kobalt.maven.aether.Scope
+import com.beust.kobalt.misc.kobaltLog
+import org.eclipse.aether.graph.DependencyFilter
+import org.eclipse.aether.graph.DependencyNode
/**
* Manage the creation of dependencies and also provide dependencies for projects.
@@ -9,12 +14,12 @@ interface IDependencyManager {
/**
* Parse the id and return the correct IClasspathDependency
*/
- fun create(id: String, projectDirectory: String? = null): IClasspathDependency
+ fun create(id: String, optional: Boolean = false, projectDirectory: String? = null): IClasspathDependency
/**
* Create an IClasspathDependency from a Maven id.
*/
- fun createMaven(id: String): IClasspathDependency
+ fun createMaven(id: String, optional: Boolean = false): IClasspathDependency
/**
* Create an IClasspathDependency from a path.
@@ -24,7 +29,7 @@ interface IDependencyManager {
/**
* @return the source dependencies for this project, including the contributors.
*/
- fun dependencies(project: Project, context: KobaltContext): List
+ fun dependencies(project: Project, context: KobaltContext, scopes: List): List
/**
* @return the test dependencies for this project, including the contributors.
@@ -36,6 +41,48 @@ interface IDependencyManager {
* allDependencies is typically either compileDependencies or testDependencies
*/
fun calculateDependencies(project: Project?, context: KobaltContext,
- scopeFilters: Collection = listOf(Scope.COMPILE),
+ dependencyFilter: DependencyFilter =
+ createDependencyFilter(project, project?.compileDependencies ?: emptyList()),
+ scopes: List = listOf(Scope.COMPILE),
vararg passedDependencies: List): List
+
+ /**
+ * Create an Aether dependency filter that uses the dependency configuration included in each
+ * IClasspathDependency.
+ */
+ fun createDependencyFilter(project: Project?, dependencies: List) : DependencyFilter {
+ return DependencyFilter { p0, p1 ->
+ fun isNodeExcluded(node: DependencyNode, passedDep: IClasspathDependency) : Boolean {
+ val dep = create(KobaltMavenResolver.artifactToId(node.artifact))
+ return passedDep.excluded.any { ex -> ex.isExcluded(dep)}
+ }
+ fun isDepExcluded(node: DependencyNode, excluded: List?) : Boolean {
+ val dep = create(KobaltMavenResolver.artifactToId(node.artifact))
+ return excluded?.map { it.id }?.contains(dep.id) ?: false
+ }
+
+ val accept = dependencies.isEmpty() || dependencies.any {
+ // Is this dependency excluded?
+ val isExcluded = isNodeExcluded(p0, it) || isDepExcluded(p0, project?.excludedDependencies)
+
+ // Is the parent dependency excluded?
+ val isParentExcluded =
+ if (p1.any()) {
+ isNodeExcluded(p1[0], it) || isDepExcluded(p1[0], project?.excludedDependencies)
+ } else {
+ false
+ }
+
+ // Only accept if no exclusions were found
+ ! isExcluded && ! isParentExcluded
+ }
+
+ if (! accept) {
+ kobaltLog(2, "Excluding $p0")
+ }
+
+ if (accept) EXCLUDE_OPTIONAL_FILTER.accept(p0, p1)
+ else accept
+ }
+ }
}
\ No newline at end of file
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IRunnerContributor.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IRunnerContributor.kt
index 5ab88cb0..f8c28b52 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IRunnerContributor.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/IRunnerContributor.kt
@@ -1,10 +1,11 @@
package com.beust.kobalt.api
import com.beust.kobalt.TaskResult
-import com.beust.kobalt.api.IClasspathDependency
/**
* Plugins that can run a project (task "run" or "test") should implement this interface.
+ *
+ * Currently not used.
*/
interface IRunnerContributor : IContributor, IProjectAffinity {
/**
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ITaskContributor.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ITaskContributor.kt
index 83621451..c606d54f 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ITaskContributor.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/ITaskContributor.kt
@@ -23,9 +23,10 @@ class DynamicTask(override val plugin: IPlugin, override val name: String, overr
override fun call(): TaskResult2 {
val taskResult = closure.invoke(project)
- return TaskResult2(taskResult.success, taskResult.errorMessage, this)
+ return TaskResult2(taskResult.success, errorMessage = taskResult.errorMessage, value = this)
}
- override fun toString() = "[DynamicTask $name dependsOn=$dependsOn reverseDependsOn=$reverseDependsOn]"
+ override fun toString() =
+ "[DynamicTask ${project.name}:$name dependsOn=$dependsOn reverseDependsOn=$reverseDependsOn]"
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/InputStreamJarTemplate.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/InputStreamJarTemplate.kt
new file mode 100644
index 00000000..222a2829
--- /dev/null
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/InputStreamJarTemplate.kt
@@ -0,0 +1,55 @@
+package com.beust.kobalt.api
+
+import com.beust.kobalt.Args
+import com.beust.kobalt.misc.KFiles
+import com.beust.kobalt.misc.kobaltLog
+import java.io.*
+import java.net.URL
+import java.util.jar.JarInputStream
+
+/**
+ * Base class for templates that decompress a jar file.
+ */
+interface InputStreamJarTemplate : ITemplate {
+ val inputStream: InputStream
+
+ override fun generateTemplate(args: Args, classLoader: ClassLoader) {
+ val destDir = File(".")
+ JarInputStream(inputStream).use { ins ->
+ var entry = ins.nextEntry
+ while (entry != null) {
+ val f = File(destDir.path + File.separator + entry.name)
+ if (entry.isDirectory) {
+ f.mkdir()
+ entry = ins.nextEntry
+ continue
+ }
+
+ kobaltLog(2, " Extracting: $entry to ${f.absolutePath}")
+ FileOutputStream(f).use { fos ->
+ KFiles.copy(ins, fos)
+ }
+ entry = ins.nextEntry
+ }
+ }
+ }
+}
+
+abstract class ResourceJarTemplate(jarName: String, val classLoader: ClassLoader) : InputStreamJarTemplate {
+ override val inputStream : InputStream = classLoader.getResource(jarName).openConnection().inputStream
+}
+
+abstract class FileJarTemplate(val fileName: String) : InputStreamJarTemplate {
+ override val inputStream = FileInputStream(File(fileName))
+}
+
+abstract class HttpJarTemplate(val url: String) : InputStreamJarTemplate {
+ override val inputStream : InputStream
+ get() {
+ try {
+ return URL(url).openConnection().inputStream
+ } catch(ex: IOException) {
+ throw IllegalArgumentException("Couldn't connect to $url")
+ }
+ }
+}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/JarTemplate.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/JarTemplate.kt
deleted file mode 100644
index 1966b747..00000000
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/JarTemplate.kt
+++ /dev/null
@@ -1,41 +0,0 @@
-package com.beust.kobalt.api
-
-import com.beust.kobalt.Args
-import com.beust.kobalt.misc.KFiles
-import com.beust.kobalt.misc.log
-import java.io.File
-import java.io.FileOutputStream
-import java.util.jar.JarInputStream
-
-/**
- * Base class for templates that simply decompress a jar file to generate their project.
- */
-abstract class JarTemplate(val jarName: String) : ITemplate {
- companion object {
- fun extractFile(ins: JarInputStream, destDir: File) {
- var entry = ins.nextEntry
- while (entry != null) {
- val f = File(destDir.path + File.separator + entry.name)
- if (entry.isDirectory) {
- f.mkdir()
- entry = ins.nextEntry
- continue
- }
-
- log(2, "Extracting: $entry to ${f.absolutePath}")
- FileOutputStream(f).use { fos ->
- KFiles.copy(ins, fos)
- }
- entry = ins.nextEntry
- }
- }
- }
-
- override fun generateTemplate(args: Args, classLoader: ClassLoader) {
- log(2, "Generating template with class loader $classLoader")
- val destDir = File(".")
- val ins = JarInputStream(classLoader.getResource(jarName).openConnection().inputStream)
- extractFile(ins, destDir)
- }
-
-}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/Kobalt.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/Kobalt.kt
index 49b82050..7d37a0b8 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/Kobalt.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/Kobalt.kt
@@ -5,6 +5,7 @@ import com.beust.kobalt.HostConfig
import com.beust.kobalt.Plugins
import com.beust.kobalt.internal.PluginInfo
import com.beust.kobalt.maven.DependencyManager
+import com.beust.kobalt.maven.aether.KobaltMavenResolver
import com.google.inject.Guice
import com.google.inject.Injector
import com.google.inject.Module
@@ -35,12 +36,12 @@ class Kobalt {
*/
val repos : Set
get() {
- val settingsRepos = Kobalt.context?.settings?.defaultRepos ?: emptyList()
+ val settingsRepos = Kobalt.context?.settings?.defaultRepos?.map { HostConfig(it) } ?: emptyList()
// Repos from in the settings
val result = ArrayList(
(if (settingsRepos.isEmpty()) Constants.DEFAULT_REPOS
else settingsRepos)
- .map { HostConfig(it) })
+ )
// Repo from in the settings
Kobalt.context?.settings?.kobaltCompilerRepo?.let {
@@ -55,6 +56,9 @@ class Kobalt {
// Repos from the build file
result.addAll(reposFromBuildFiles)
+ result.forEach {
+ KobaltMavenResolver.initAuthentication(it)
+ }
return result.toHashSet()
}
@@ -118,5 +122,20 @@ class Kobalt {
get() = Duration.parse( kobaltProperties.getProperty(PROPERTY_KOBALT_VERSION_CHECK_TIMEOUT) ?: "P1D")
fun findPlugin(name: String) = Plugins.findPlugin(name)
+
+ val optionsFromBuild = arrayListOf()
+ fun addKobaltOptions(options: Array) {
+ optionsFromBuild.addAll(options)
+ }
+
+ val buildSourceDirs = arrayListOf()
+ fun addBuildSourceDirs(dirs: Array) {
+ buildSourceDirs.addAll(dirs)
+ }
+
+ fun cleanUp() {
+ buildSourceDirs.clear()
+ buildFileClasspath.clear()
+ }
}
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/KobaltContext.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/KobaltContext.kt
index c927517d..b7e5ace8 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/KobaltContext.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/KobaltContext.kt
@@ -4,6 +4,7 @@ import com.beust.kobalt.Args
import com.beust.kobalt.KobaltException
import com.beust.kobalt.Plugins
import com.beust.kobalt.Variant
+import com.beust.kobalt.internal.ILogger
import com.beust.kobalt.internal.IncrementalManager
import com.beust.kobalt.internal.KobaltSettings
import com.beust.kobalt.internal.PluginInfo
@@ -11,7 +12,7 @@ import com.beust.kobalt.maven.DependencyManager
import com.beust.kobalt.maven.MavenId
import com.beust.kobalt.maven.PomGenerator
import com.beust.kobalt.maven.SimpleDep
-import com.beust.kobalt.maven.aether.KobaltAether
+import com.beust.kobalt.maven.aether.KobaltMavenResolver
import com.beust.kobalt.misc.KobaltExecutors
import java.io.File
@@ -52,9 +53,9 @@ class KobaltContext(val args: Args) {
FileType.JAVADOC -> toQualifier(dep, "", "javadoc")
FileType.OTHER -> id
}
- val resolved = aether.resolveToArtifact(fullId)
+ val resolved = resolver.resolveToArtifact(fullId)
if (resolved != null) {
- return resolved.artifact.file
+ return resolved.file
} else {
throw KobaltException("Couldn't resolve $id")
}
@@ -80,8 +81,9 @@ class KobaltContext(val args: Args) {
lateinit var executors: KobaltExecutors
lateinit var settings: KobaltSettings
lateinit var incrementalManager: IncrementalManager
- lateinit var aether: KobaltAether
+ lateinit var resolver: KobaltMavenResolver
lateinit var pomGeneratorFactory: PomGenerator.IFactory
+ lateinit var logger: ILogger
}
class InternalContext {
@@ -103,4 +105,12 @@ class InternalContext {
* The absolute directory of the current project.
*/
var absoluteDir: File? = null
+
+ /**
+ * If true, will force a recompile of the files even if using the incremental compile
+ */
+ var forceRecompile: Boolean = false
+
+ var noIncrementalKotlin: Boolean = false
+
}
\ No newline at end of file
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/Project.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/Project.kt
index 374fb90d..e54e30ec 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/Project.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/Project.kt
@@ -3,12 +3,16 @@ package com.beust.kobalt.api
import com.beust.kobalt.TestConfig
import com.beust.kobalt.api.annotation.Directive
import com.beust.kobalt.maven.DependencyManager
+import com.beust.kobalt.maven.aether.AetherDependency
+import com.beust.kobalt.maven.aether.KobaltMavenResolver
import com.beust.kobalt.misc.KFiles
+import com.beust.kobalt.misc.kobaltLog
import org.apache.maven.model.Model
import java.io.File
import java.util.*
import java.util.concurrent.Future
import java.util.concurrent.FutureTask
+import java.util.regex.Pattern
open class Project(
@Directive open var name: String = "",
@@ -22,6 +26,7 @@ open class Project(
@Directive open var url: String? = null,
@Directive open var pom: Model? = null,
@Directive open var dependsOn: ArrayList = arrayListOf(),
+ @Directive open var testsDependOn: ArrayList = arrayListOf(),
@Directive open var packageName: String? = group)
: IBuildConfig, IDependencyHolder by DependencyHolder() {
@@ -29,13 +34,15 @@ open class Project(
this.project = this
}
+ fun allProjectDependedOn() = project.dependsOn + project.testsDependOn
+
class ProjectExtra(project: Project) {
var isDirty = false
/**
* @return true if any of the projects we depend on is dirty.
*/
- fun dependsOnDirtyProjects(project: Project) = project.dependsOn.any { it.projectExtra.isDirty }
+ fun dependsOnDirtyProjects(project: Project) = project.allProjectDependedOn().any { it.projectExtra.isDirty }
}
/**
@@ -83,7 +90,8 @@ open class Project(
@Directive
fun dependenciesTest(init: Dependencies.() -> Unit) : Dependencies {
- dependencies = Dependencies(this, testDependencies, testProvidedDependencies, compileRuntimeDependencies,
+ dependencies = Dependencies(this, testDependencies, arrayListOf(),
+ testProvidedDependencies, compileOnlyDependencies, compileRuntimeDependencies,
excludedDependencies, nativeDependencies)
dependencies!!.init()
return dependencies!!
@@ -92,6 +100,9 @@ open class Project(
val testDependencies : ArrayList = arrayListOf()
val testProvidedDependencies : ArrayList = arrayListOf()
+ fun testsDependOn(vararg projects: Project) = testsDependOn.addAll(projects)
+ fun dependsOn(vararg projects: Project) = dependsOn.addAll(projects)
+
/** Used to disambiguate various name properties */
@Directive
val projectName: String get() = name
@@ -118,6 +129,18 @@ open class Project(
return result
}
+ class Dep(val file: File, val id: String)
+
+ /**
+ * @return a list of the transitive dependencies (absolute paths to jar files) for the given dependencies.
+ * Can be used for example as `collect(compileDependencies)`.
+ */
+ @Directive
+ fun collect(dependencies: List) : List {
+ return (Kobalt.context?.dependencyManager?.transitiveClosure(dependencies) ?: emptyList())
+ .map { Dep(it.jarFile.get(), it.id) }
+ }
+
override fun toString() = "[Project $name]"
}
@@ -130,7 +153,9 @@ class Sources(val project: Project, val sources: HashSet) {
class Dependencies(val project: Project,
val dependencies: ArrayList,
+ val optionalDependencies: ArrayList,
val providedDependencies: ArrayList,
+ val compileOnlyDependencies: ArrayList,
val runtimeDependencies: ArrayList,
val excludedDependencies: ArrayList,
val nativeDependencies: ArrayList) {
@@ -142,17 +167,98 @@ class Dependencies(val project: Project,
* future tasks receive a get(), the repos will be correct.
*/
private fun addToDependencies(project: Project, dependencies: ArrayList,
- dep: Array): List>
- = with(dep.map { DependencyManager.create(it, project.directory)}) {
+ dep: Array, optional: Boolean = false, excludeConfig: ExcludeConfig? = null): List>
+ = with(dep.map {
+ val resolved =
+ if (KobaltMavenResolver.isRangeVersion(it)) {
+ // Range id
+ val node = Kobalt.INJECTOR.getInstance(KobaltMavenResolver::class.java).resolveToArtifact(it)
+ val result = KobaltMavenResolver.artifactToId(node)
+ kobaltLog(2, "Resolved range id $it to $result")
+ result
+ } else {
+ it
+ }
+ DependencyManager.create(resolved, optional, project.directory)
+ }) {
dependencies.addAll(this)
+ if (excludeConfig != null) {
+ this.forEach { it.excluded.add(excludeConfig) }
+ }
+
this.map { FutureTask { it.jarFile.get() } }
- }
+ }
@Directive
fun compile(vararg dep: String) = addToDependencies(project, dependencies, dep)
+ class ExcludeConfig {
+ val ids = arrayListOf()
+
+ @Directive
+ fun exclude(vararg passedIds: String) = ids.addAll(passedIds)
+
+ class ArtifactConfig(
+ var groupId: String? = null,
+ var artifactId: String? = null,
+ var version: String? = null
+ )
+
+ val artifacts = arrayListOf()
+
+ @Directive
+ fun exclude(groupId: String? = null, artifactId: String? = null, version: String? = null)
+ = artifacts.add(ArtifactConfig(groupId, artifactId, version))
+
+ fun match(pattern: String?, id: String) : Boolean {
+ return pattern == null || Pattern.compile(pattern).matcher(id).matches()
+ }
+
+ /**
+ * @return true if the dependency is excluded with any of the exclude() directives. The matches
+ * are performed by a regular expression match against the dependency.
+ */
+ fun isExcluded(dep: IClasspathDependency) : Boolean {
+ // Straight id match
+ var result = ids.any { match(it, dep.id) }
+
+ // Match on any combination of (groupId, artifactId, version)
+ if (! result && dep.isMaven) {
+ val mavenDep = dep as AetherDependency
+ val artifact = mavenDep.artifact
+ result = artifacts.any {
+ val match1 = it.groupId == null || match(it.groupId, artifact.groupId)
+ val match2 = it.artifactId == null || match(it.artifactId, artifact.artifactId)
+ val match3 = it.version == null || match(it.version, artifact.version)
+ match1 && match2 && match3
+ }
+ }
+
+ return result
+ }
+ }
+
@Directive
- fun provided(vararg dep: String) = addToDependencies(project, providedDependencies, dep)
+ fun compile(dep: String, init: ExcludeConfig.() -> Unit) {
+ val excludeConfig = ExcludeConfig().apply {
+ init()
+ }
+ addToDependencies(project, dependencies, arrayOf(dep), excludeConfig = excludeConfig)
+ }
+
+ @Directive
+ fun compileOnly(vararg dep: String) = addToDependencies(project, compileOnlyDependencies, dep)
+
+ @Directive
+ fun compileOptional(vararg dep: String) {
+ addToDependencies(project, optionalDependencies, dep, optional = true)
+ addToDependencies(project, dependencies, dep, optional = true)
+ }
+
+ @Directive
+ fun provided(vararg dep: String) {
+ addToDependencies(project, providedDependencies, dep)
+ }
@Directive
fun runtime(vararg dep: String) = addToDependencies(project, runtimeDependencies, dep)
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/TaskContributor.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/TaskContributor.kt
index 65eb529d..8c68be94 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/TaskContributor.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/api/TaskContributor.kt
@@ -44,6 +44,25 @@ class TaskContributor @Inject constructor(val incrementalManagerFactory: Increme
}
}
+ fun addTask(plugin: IPlugin, project: Project, taskName: String, description: String,
+ group: String = AnnotationDefault.GROUP,
+ dependsOn: List = emptyList(),
+ reverseDependsOn : List = emptyList(),
+ runBefore : List = emptyList(),
+ runAfter : List = emptyList(),
+ alwaysRunAfter: List = emptyList(),
+ runTask: (Project) -> TaskResult) {
+ dynamicTasks.add(DynamicTask(plugin, taskName, description, group, project,
+ dependsOn = dependsOn,
+ reverseDependsOn = reverseDependsOn,
+ runBefore = runBefore,
+ runAfter = runAfter,
+ alwaysRunAfter = alwaysRunAfter,
+ closure = { p: Project ->
+ runTask(project)
+ }))
+ }
+
fun addIncrementalVariantTasks(plugin: IPlugin, project: Project, context: KobaltContext, taskName: String,
group: String = AnnotationDefault.GROUP,
dependsOn: List = emptyList(),
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Archives.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Archives.kt
index 31bc254a..5334e09f 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Archives.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Archives.kt
@@ -1,30 +1,24 @@
package com.beust.kobalt.archive
-import com.beust.kobalt.Features
-import com.beust.kobalt.IFileSpec
+import com.beust.kobalt.*
import com.beust.kobalt.api.KobaltContext
import com.beust.kobalt.api.Project
import com.beust.kobalt.api.annotation.ExportedProjectProperty
-import com.beust.kobalt.misc.From
-import com.beust.kobalt.misc.IncludedFile
import com.beust.kobalt.misc.JarUtils
import com.beust.kobalt.misc.KFiles
-import com.beust.kobalt.misc.log
-import com.beust.kobalt.misc.To
+import com.beust.kobalt.misc.kobaltLog
import java.io.File
-import java.io.FileOutputStream
-import java.io.OutputStream
import java.util.*
-import java.util.zip.ZipOutputStream
class Archives {
companion object {
@ExportedProjectProperty(doc = "The name of the jar file", type = "String")
const val JAR_NAME = "jarName"
+ @ExportedProjectProperty(doc = "The name of the a jar file with a main() method", type = "String")
+ const val JAR_NAME_WITH_MAIN_CLASS = "jarNameWithMainClass"
- private val DEFAULT_STREAM_FACTORY = { os : OutputStream -> ZipOutputStream(os) }
-
- fun defaultArchiveName(project: Project) = project.name + "-" + project.version
+ fun defaultArchiveName(project: Project) = project.name +
+ if (project.version.isNullOrBlank()) "" else "-${project.version}"
fun generateArchive(project: Project,
context: KobaltContext,
@@ -32,17 +26,17 @@ class Archives {
suffix: String,
includedFiles: List,
expandJarFiles : Boolean = false,
- outputStreamFactory: (OutputStream) -> ZipOutputStream = DEFAULT_STREAM_FACTORY) : File {
+ manifest: java.util.jar.Manifest? = null) : File {
val fullArchiveName = context.variant.archiveName(project, archiveName, suffix)
val archiveDir = File(KFiles.libsDir(project))
val result = File(archiveDir.path, fullArchiveName)
- log(3, "Creating $result")
+ context.logger.log(project.name, 3, "Creating $result")
if (! Features.USE_TIMESTAMPS || isOutdated(project.directory, includedFiles, result)) {
try {
- outputStreamFactory(FileOutputStream(result)).use {
- JarUtils.addFiles(project.directory, includedFiles, it, expandJarFiles)
- log(2, text = "Added ${includedFiles.size} files to $result")
- log(1, " Created $result")
+ MetaArchive(result, manifest).use { metaArchive ->
+ JarUtils.addFiles(project.directory, includedFiles, metaArchive, expandJarFiles)
+ context.logger.log(project.name, 2, "Added ${includedFiles.size} files to $result")
+ context.logger.log(project.name, 1, " Created $result")
}
} catch (e: Throwable) {
// make sure that incomplete archive is deleted
@@ -52,11 +46,9 @@ class Archives {
}
} else {
- log(3, " $result is up to date")
+ context.logger.log(project.name, 3, " $result is up to date")
}
- project.projectProperties.put(JAR_NAME, result.absolutePath)
-
return result
}
@@ -73,7 +65,7 @@ class Archives {
File(KFiles.joinDir(directory, root.from, relFile.path))
if (file.isFile) {
if (file.lastModified() > lastModified) {
- log(3, " TS - Outdated $file and $output "
+ kobaltLog(3, " TS - Outdated $file and $output "
+ Date(file.lastModified()) + " " + Date(output.lastModified()))
return true
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Jar.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Jar.kt
index f8a47848..d5086cbd 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Jar.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Jar.kt
@@ -8,7 +8,7 @@ import com.beust.kobalt.api.annotation.Directive
*/
open class Jar(override val project: Project,
override var name : String = Archives.defaultArchiveName(project) + ".jar",
- var fatJar: Boolean = false) : Zip(project, name), AttributeHolder {
+ override var fatJar: Boolean = false) : Zip(project, name, fatJar), AttributeHolder {
@Directive
fun manifest(init: Manifest.(p: Manifest) -> Unit) : Manifest {
val m = Manifest(this)
@@ -18,7 +18,7 @@ open class Jar(override val project: Project,
// Need to specify the version or attributes will just be dropped
@Directive
- val attributes = arrayListOf(Pair("Manifest-Version", "1.0"))
+ override val attributes = arrayListOf(Pair("Manifest-Version", "1.0"))
override fun addAttribute(k: String, v: String) {
attributes.add(Pair(k, v))
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/MetaArchive.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/MetaArchive.kt
new file mode 100644
index 00000000..c217c83e
--- /dev/null
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/MetaArchive.kt
@@ -0,0 +1,125 @@
+package com.beust.kobalt.archive
+
+import com.beust.kobalt.Glob
+import com.beust.kobalt.misc.KFiles
+import org.apache.commons.compress.archivers.ArchiveEntry
+import org.apache.commons.compress.archivers.zip.ZipArchiveEntry
+import org.apache.commons.compress.archivers.zip.ZipArchiveOutputStream
+import java.io.Closeable
+import java.io.File
+import java.io.FileInputStream
+import java.io.FileOutputStream
+import java.util.jar.Manifest
+import org.apache.commons.compress.archivers.zip.ZipFile as ApacheZipFile
+
+/**
+ * Abstraction of a zip/jar/war archive that automatically manages the addition of expanded jar files.
+ * Uses ZipArchiveOutputStream for fast inclusion of expanded jar files.
+ */
+class MetaArchive(outputFile: File, val manifest: Manifest?) : Closeable {
+ companion object {
+ const val MANIFEST_MF = "META-INF/MANIFEST.MF"
+ }
+
+ private val zos= ZipArchiveOutputStream(outputFile).apply {
+ encoding = "UTF-8"
+ }
+
+ init {
+ // If no manifest was passed, create an empty one so it's the first one in the archive
+ val m = manifest ?: Manifest()
+ val manifestFile = File.createTempFile("kobalt", "tmpManifest")
+ addEntry(ZipArchiveEntry("META-INF/"), null)
+ if (manifest != null) {
+ FileOutputStream(manifestFile).use { fos ->
+ m.write(fos)
+ }
+ }
+ val entry = zos.createArchiveEntry(manifestFile, MetaArchive.MANIFEST_MF)
+ addEntry(entry, FileInputStream(manifestFile))
+ }
+
+
+ fun addFile(f: File, entryFile: File, path: String?) {
+ maybeCreateParentDirectories(f)
+ addFile2(f, entryFile, path)
+ }
+
+ private fun addFile2(f: File, entryFile: File, path: String?) {
+ val file = f.normalize()
+ FileInputStream(file).use { inputStream ->
+ val actualPath = KFiles.fixSlashes(if (path != null) path + entryFile.path else entryFile.path)
+ ZipArchiveEntry(actualPath).let { entry ->
+ maybeCreateParentDirectories(File(actualPath))
+ maybeAddEntry(entry) {
+ addEntry(entry, inputStream)
+ }
+ }
+ }
+ }
+
+ private val createdDirs = hashSetOf()
+
+ /**
+ * For an entry a/b/c/File, an entry needs to be created for each individual directory:
+ * a/
+ * a/b/
+ * a/b/c
+ * a/b/c/File
+ */
+ private fun maybeCreateParentDirectories(file: File) {
+ val toCreate = arrayListOf()
+ var current = file.parentFile
+ while (current != null && current.path != ".") {
+ if (!createdDirs.contains(current.path)) {
+ toCreate.add(0, KFiles.fixSlashes(current) + "/")
+ createdDirs.add(current.path)
+ }
+ current = current.parentFile
+ }
+ toCreate.forEach { dir ->
+ addEntry(ZipArchiveEntry(dir), null)
+ }
+ }
+
+ fun addArchive(jarFile: File) {
+ ApacheZipFile(jarFile).use { jar ->
+ val jarEntries = jar.entries
+ for (entry in jarEntries) {
+ maybeAddEntry(entry) {
+ zos.addRawArchiveEntry(entry, jar.getRawInputStream(entry))
+ }
+ }
+ }
+ }
+
+
+
+ private fun okToAdd(name: String) : Boolean {
+ val result = !KFiles.isExcluded(name,
+ Glob("META-INF/*.SF", "META-INF/*.DSA", "META-INF/*.RSA", MANIFEST_MF))
+// if (name.startsWith("META-INF")) println((if (result) "ADDING" else "NOT ADDING") + " $name")
+ return result
+ }
+
+ override fun close() = zos.close()
+
+ private fun addEntry(entry: ArchiveEntry, inputStream: FileInputStream?) {
+ zos.putArchiveEntry(entry)
+ inputStream?.use { ins ->
+ ins.copyTo(zos, 50 * 1024)
+ }
+ zos.closeArchiveEntry()
+ }
+
+ private val seen = hashSetOf()
+
+ private fun maybeAddEntry(entry: ArchiveEntry, action:() -> Unit) {
+ entry.name.let { name ->
+ if (!seen.contains(name) && okToAdd(name)) {
+ action()
+ }
+ seen.add(name)
+ }
+ }
+}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/War.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/War.kt
index 295a1987..978f21bf 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/War.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/War.kt
@@ -6,7 +6,7 @@ import com.beust.kobalt.glob
class War(override val project: Project, override var name: String = Archives.defaultArchiveName(project) + ".war")
: Jar(project, name), AttributeHolder {
init {
- include(from("src/main/webapp"),to(""), glob("**"))
+ include(from("src/main/webapp"), to(""), glob("**"))
include(from("kobaltBuild/classes"), to("WEB-INF/classes"), glob("**"))
}
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Zip.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Zip.kt
index 61237291..41957218 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Zip.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/archive/Zip.kt
@@ -1,22 +1,13 @@
package com.beust.kobalt.archive
-import com.beust.kobalt.Glob
-import com.beust.kobalt.IFileSpec
+import com.beust.kobalt.*
import com.beust.kobalt.api.Project
import com.beust.kobalt.api.annotation.Directive
-import com.beust.kobalt.misc.From
-import com.beust.kobalt.misc.IncludedFile
-import com.beust.kobalt.misc.To
-open class Zip(open val project: Project, open var name: String = Archives.defaultArchiveName(project) + ".zip") {
+open class Zip(open val project: Project, open var name: String = Archives.defaultArchiveName(project) + ".zip",
+ open var fatJar: Boolean = false): AttributeHolder, IncludeFromTo() {
val excludes = arrayListOf()
- @Directive
- fun from(s: String) = From(s)
-
- @Directive
- fun to(s: String) = To(s)
-
@Directive
fun exclude(vararg files: String) {
files.forEach { excludes.add(Glob(it)) }
@@ -28,26 +19,9 @@ open class Zip(open val project: Project, open var name: String = Archives.defau
}
@Directive
- fun include(vararg files: String) {
- includedFiles.add(IncludedFile(files.map { IFileSpec.FileSpec(it) }))
+ open val attributes = arrayListOf(Pair("Manifest-Version", "1.0"))
+
+ override fun addAttribute(k: String, v: String) {
+ attributes.add(Pair(k, v))
}
-
- @Directive
- fun include(from: From, to: To, vararg specs: String) {
- includedFiles.add(IncludedFile(from, to, specs.map { IFileSpec.FileSpec(it) }))
- }
-
- @Directive
- fun include(from: From, to: To, vararg specs: IFileSpec.GlobSpec) {
- includedFiles.add(IncludedFile(from, to, listOf(*specs)))
- }
-
- /**
- * Prefix path to be removed from the zip file. For example, if you add "build/lib/a.jar" to the zip
- * file and the excludePrefix is "build/lib", then "a.jar" will be added at the root of the zip file.
- */
- val includedFiles = arrayListOf()
-
}
-
-
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/ActorUtils.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/ActorUtils.kt
index 42aac9e2..e9b315a5 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/ActorUtils.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/ActorUtils.kt
@@ -1,7 +1,6 @@
package com.beust.kobalt.internal
import com.beust.kobalt.api.IProjectAffinity
-import com.beust.kobalt.api.ISimpleAffinity
import com.beust.kobalt.api.KobaltContext
import com.beust.kobalt.api.Project
@@ -11,19 +10,13 @@ class ActorUtils {
* Return the plug-in actor with the highest affinity.
*/
fun selectAffinityActor(project: Project, context: KobaltContext, actors: List)
- = actors.maxBy { it.affinity(project, context) }
+ = actors.maxBy { it.affinity(project, context) }
/**
* Return all the plug-in actors with a non zero affinity sorted from the highest to the lowest.
*/
fun selectAffinityActors(project: Project, context: KobaltContext, actors: List)
= actors.filter { it.affinity(project, context) > 0 }
- .sortedByDescending { it.affinity(project, context) }
-
- /**
- * Return the plug-in actor with the highest affinity.
- */
- fun , A> selectAffinityActor(actors: List, arg: A) = actors.maxBy { it.affinity(arg) }
+ .sortedByDescending { it.affinity(project, context) }
}
-
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BaseJvmPlugin.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BaseJvmPlugin.kt
index cfece392..e89fc9aa 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BaseJvmPlugin.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BaseJvmPlugin.kt
@@ -12,7 +12,7 @@ abstract class BaseJvmPlugin(open val configActor: ConfigActor) :
ICompilerFlagContributor {
companion object {
// Run before other flag contributors
- val FLAG_CONTRIBUTOR_PRIORITY = ICompilerFlagContributor.DEFAULT_FLAG_PRIORITY - 10
+ val FLAG_CONTRIBUTOR_PRIORITY = FlagContributor.DEFAULT_FLAG_PRIORITY - 10
}
protected fun maybeCompilerArgs(sourceSuffixes: List, suffixesBeingCompiled: List,
@@ -21,14 +21,17 @@ abstract class BaseJvmPlugin(open val configActor: ConfigActor) :
override val flagPriority = FLAG_CONTRIBUTOR_PRIORITY
- override fun accept(project: Project) = hasSourceFiles(project)
+ override fun accept(project: Project) = sourceFileCount(project) > 0
// IBuildConfigContributor
+ protected fun sourceFileCount(project: Project)
+ = KFiles.findSourceFiles(project.directory, project.sourceDirectories, sourceSuffixes()).size +
+ KFiles.findSourceFiles(project.directory, project.sourceDirectoriesTest, sourceSuffixes()).size
- private fun hasSourceFiles(project: Project)
- = KFiles.findSourceFiles(project.directory, project.sourceDirectories, sourceSuffixes()).size > 0
+ fun affinity(project: Project) = sourceFileCount(project)
- fun affinity(project: Project) = if (hasSourceFiles(project)) 1 else 0
+ // IDocContributor
+ open fun affinity(project: Project, context: KobaltContext) = sourceFileCount(project)
abstract fun sourceSuffixes() : List
}
\ No newline at end of file
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BaseProjectRunner.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BaseProjectRunner.kt
index 40859d83..963255bd 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BaseProjectRunner.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BaseProjectRunner.kt
@@ -1,7 +1,11 @@
package com.beust.kobalt.internal
+import com.beust.kobalt.TestResult
+import com.beust.kobalt.api.IBuildListener
+import com.beust.kobalt.api.KobaltContext
import com.beust.kobalt.api.Project
-import com.beust.kobalt.misc.log
+import com.beust.kobalt.api.ProjectBuildStatus
+import com.beust.kobalt.misc.kobaltLog
import com.google.common.annotations.VisibleForTesting
import com.google.common.collect.ArrayListMultimap
import com.google.common.collect.Multimap
@@ -13,7 +17,26 @@ abstract class BaseProjectRunner {
: TaskManager.RunTargetResult
companion object {
- val LOG_LEVEL = TaskManager.LOG_LEVEL
+ val TAG = "graph"
+
+ fun runBuildListenersForProject(project: Project, context: KobaltContext, start: Boolean,
+ status: ProjectBuildStatus = ProjectBuildStatus.SUCCESS) {
+ context.pluginInfo.buildListeners.forEach {
+ if (start) it.projectStart(project, context) else it.projectEnd(project, context, status)
+ }
+ }
+
+ fun runBuildListenersForTask(project: Project, context: KobaltContext, taskName: String, start: Boolean,
+ success: Boolean = false, testResult: TestResult? = null) {
+ context.pluginInfo.buildListeners.forEach {
+ if (start) {
+ it.taskStart(project, context, taskName)
+ } else {
+ val info = IBuildListener.TaskEndInfo(success, testResult?.shortMessage, testResult?.longMessage)
+ it.taskEnd(project, context, taskName, info)
+ }
+ }
+ }
/**
* Create a graph representing the tasks and their dependencies. That graph will then be run
@@ -34,6 +57,20 @@ abstract class BaseProjectRunner {
accept: (T) -> Boolean):
DynamicGraph {
+ /**
+ * Add an edge from @param from to all its tasks.
+ */
+ fun addEdge(result: DynamicGraph, from: String, to: String, newToProcess: ArrayList, text: String) {
+ val froms = nodeMap[from]
+ froms.forEach { f: T ->
+ nodeMap[to].forEach { t: T ->
+ kobaltLog(TAG, " Adding edge ($text) $f -> $t")
+ result.addEdge(f, t)
+ newToProcess.add(t)
+ }
+ }
+ }
+
val result = DynamicGraph()
val newToProcess = arrayListOf()
val seen = hashSetOf()
@@ -63,37 +100,23 @@ abstract class BaseProjectRunner {
while (toProcess.size > 0) {
- /**
- * Add an edge from @param from to all its tasks.
- */
- fun addEdge(result: DynamicGraph, from: String, to: String, newToProcess: ArrayList, text: String) {
- val froms = nodeMap[from]
- froms.forEach { f: T ->
- nodeMap[to].forEach { t: T ->
- log(LOG_LEVEL, " Adding edge ($text) $f -> $t")
- result.addEdge(f, t)
- newToProcess.add(t)
- }
- }
- }
-
/**
* Whenever a task is added to the graph, we also add its alwaysRunAfter tasks.
*/
fun processAlways(always: Multimap, node: T) {
- log(LOG_LEVEL, " Processing always for $node")
+ kobaltLog(TAG, " Processing always for $node")
always[toName(node)]?.let { to: Collection ->
to.forEach { t ->
nodeMap[t].forEach { from ->
- log(LOG_LEVEL, " Adding always edge $from -> $node")
+ kobaltLog(TAG, " Adding always edge $from -> $node")
result.addEdge(from, node)
}
}
- log(LOG_LEVEL, " ... done processing always for $node")
+ kobaltLog(TAG, " ... done processing always for $node")
}
}
- log(LOG_LEVEL, " Current batch to process: $toProcess")
+ kobaltLog(TAG, " Current batch to process: $toProcess")
//
// Move dependsOn + reverseDependsOn in one multimap called allDepends
@@ -115,7 +138,7 @@ abstract class BaseProjectRunner {
//
toProcess.forEach { taskInfo ->
val taskName = taskInfo.taskName
- log(LOG_LEVEL, " ***** Current node: $taskName")
+ kobaltLog(TAG, " ***** Current node: $taskName")
nodeMap[taskName].forEach {
result.addNode(it)
processAlways(always, it)
@@ -128,33 +151,36 @@ abstract class BaseProjectRunner {
addEdge(result, taskName, to, newToProcess, "dependsOn")
}
- //
- // runBefore and runAfter (task ordering) are only considered for explicit tasks (tasks that were
- // explicitly requested by the user)
- //
- passedTasks.map { it.id }.let { taskNames ->
- runBefore[taskName].forEach { from ->
- if (taskNames.contains(from)) {
- addEdge(result, from, taskName, newToProcess, "runBefore")
- }
- }
- runAfter[taskName].forEach { to ->
- if (taskNames.contains(to)) {
- addEdge(result, taskName, to, newToProcess, "runAfter")
- }
- }
- }
seen.add(taskName)
}
newToProcess.forEach { processAlways(always, it) }
toProcess.clear()
- toProcess.addAll(newToProcess.filter { !seen.contains(toName(it)) }.map { TaskManager.TaskInfo(toName(it)) })
+ toProcess.addAll(
+ newToProcess
+ .filter { !seen.contains(toName(it)) }
+ .map { TaskManager.TaskInfo(toName(it)) })
newToProcess.clear()
}
+
+ //
+ // Now that we have all the tasks tnat need to run, process runBefore/runAfter, which
+ // are not allowed to add new tasks. Therefore, we only add edges to the graph if both
+ // the from and the to are already present.
+ //
+ val finalTaskNames = result.nodes.map { TaskManager.TaskInfo(it.toString()).taskName }
+ finalTaskNames.forEach { taskName ->
+ runBefore[taskName].filter { finalTaskNames.contains(it) }.forEach { from ->
+ addEdge(result, from, taskName, newToProcess, "runBefore")
+ }
+ runAfter[taskName].filter { finalTaskNames.contains(it) }.forEach { to ->
+ addEdge(result, to, taskName, newToProcess, "runAfter")
+ }
+ }
+
+
return result
}
}
-
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BuildListeners.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BuildListeners.kt
index 96b6e10e..58d8eed8 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BuildListeners.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/BuildListeners.kt
@@ -3,60 +3,126 @@ package com.beust.kobalt.internal
import com.beust.kobalt.Args
import com.beust.kobalt.AsciiArt
import com.beust.kobalt.api.*
-import com.beust.kobalt.misc.log
+import com.beust.kobalt.misc.kobaltLog
import java.util.concurrent.ConcurrentHashMap
/**
- * Record timings and display them at the end of the build.
+ * Record timings and statuses for tasks and projects and display them at the end of the build.
*/
class BuildListeners : IBuildListener, IBuildReportContributor {
class ProfilerInfo(val taskName: String, val durationMillis: Long)
+ class ProjectInfo(val projectName: String, var durationMillis: Long = 0,
+ var shortMessage: String? = null, var longMessage: String? = null)
private val startTimes = ConcurrentHashMap()
private val timings = arrayListOf()
+ private val projectInfos = hashMapOf()
+ private var hasFailures = false
+ private val args: Args get() = Kobalt.INJECTOR.getInstance(Args::class.java)
+ private var buildStartTime: Long? = null
+ // IBuildListener
override fun taskStart(project: Project, context: KobaltContext, taskName: String) {
startTimes.put(taskName, System.currentTimeMillis())
+ if (! projectInfos.containsKey(project.name)) {
+ projectInfos.put(project.name, ProjectInfo(project.name))
+ }
}
- override fun taskEnd(project: Project, context: KobaltContext, taskName: String, success: Boolean) {
+ // IBuildListener
+ override fun taskEnd(project: Project, context: KobaltContext, taskName: String, info: IBuildListener.TaskEndInfo) {
+ val success = info.success
+ if (! success) hasFailures = true
startTimes[taskName]?.let {
- timings.add(ProfilerInfo(taskName, System.currentTimeMillis() - it))
+ val taskTime = System.currentTimeMillis() - it
+ timings.add(ProfilerInfo(taskName, taskTime))
+ projectInfos[project.name]?.let {
+ it.durationMillis += taskTime
+ if (info.shortMessage != null && it.shortMessage == null) it.shortMessage = info.shortMessage
+ if (info.longMessage != null && it.longMessage == null) it.longMessage = info.longMessage
+ }
}
}
+ private val projectStatuses = arrayListOf>()
+
+ // IBuildListener
+ override fun projectStart(project: Project, context: KobaltContext) {
+ if (buildStartTime == null) buildStartTime = System.currentTimeMillis()
+ }
+
+ // IBuildListener
+ override fun projectEnd(project: Project, context: KobaltContext, status: ProjectBuildStatus) {
+ val shortMessage = projectInfos[project.name]?.shortMessage
+ val statusText = status.toString() + (if (shortMessage != null) " ($shortMessage)" else "")
+ projectStatuses.add(Pair(project, statusText))
+ }
+
+ // IBuildReportContributor
override fun generateReport(context: KobaltContext) {
- val profiling = Kobalt.INJECTOR.getInstance(Args::class.java).profiling
+ fun formatMillis(millis: Long, format: String) = String.format(format, millis.toDouble() / 1000)
+ fun formatMillisRight(millis: Long, length: Int) = formatMillis(millis, "%1\$$length.2f")
+ fun formatMillisLeft(millis: Long, length: Int) = formatMillis(millis, "%1\$-$length.2f")
+
+ fun millisToSeconds(millis: Long) = (millis.toDouble() / 1000).toInt()
+
+ val profiling = args.profiling
if (profiling) {
- log(1, "\n" + AsciiArt.horizontalSingleLine + " Timings (in seconds)")
+ kobaltLog(1, "\n" + AsciiArt.horizontalSingleLine + " Timings (in seconds)")
timings.sortedByDescending { it.durationMillis }.forEach {
- log(1, String.format("%1$10.2f", it.durationMillis.toDouble() / 1000)
- + " " + it.taskName)
+ kobaltLog(1, formatMillisRight(it.durationMillis, 10) + " " + it.taskName)
}
- log(1, "\n")
+ kobaltLog(1, "\n")
}
+ // Calculate the longest short message so we can create a column long enough to contain it
+ val width = 12 + (projectInfos.values.map { it.shortMessage?.length ?: 0 }.maxBy { it } ?: 0)
+
fun col1(s: String) = String.format(" %1\$-30s", s)
- fun col2(s: String) = String.format(" %1\$-10s", s)
+ fun col2(s: String) = String.format(" %1\$-${width}s", s)
+ fun col3(s: String) = String.format(" %1\$-8s", s)
- val line = listOf(col1("Project"), col2("Build status")).joinToString(AsciiArt.verticalBar)
- AsciiArt.logBox(listOf(line), AsciiArt.bottomLeft2, AsciiArt.bottomRight2)
- projectStatuses.forEach { pair ->
- val cl = listOf(col1(pair.first.name), col2(pair.second.toString())).joinToString(AsciiArt.verticalBar)
- log(1, " " + AsciiArt.verticalBar + " " + cl + " " + AsciiArt.verticalBar)
+
+
+ // Only print the build report if there is more than one project and at least one of them failed
+ if (timings.any()) {
+// if (timings.size > 1 && hasFailures) {
+ val line = listOf(col1("Project"), col2("Build status"), col3("Time"))
+ .joinToString(AsciiArt.verticalBar)
+ val table = StringBuffer()
+ table.append(AsciiArt.logBox(listOf(line), AsciiArt.bottomLeft2, AsciiArt.bottomRight2, indent = 10) + "\n")
+ projectStatuses.forEach { pair ->
+ val projectName = pair.first.name
+ val cl = listOf(col1(projectName), col2(pair.second),
+ col3(formatMillisLeft(projectInfos[projectName]!!.durationMillis, 8)))
+ .joinToString(AsciiArt.verticalBar)
+ table.append(" " + AsciiArt.verticalBar + " " + cl + " " + AsciiArt.verticalBar + "\n")
+ }
+ table.append(" " + AsciiArt.lowerBox(line.length))
+ kobaltLog(1, table.toString())
+// }
}
- log(1, " " + AsciiArt.lowerBox(line.length))
- }
+ val buildTime =
+ if (buildStartTime != null)
+ millisToSeconds(System.currentTimeMillis() - buildStartTime!!)
+ else
+ 0
+ // BUILD SUCCESSFUL / FAILED message
+ val message =
+ if (hasFailures) {
+ String.format("BUILD FAILED", buildTime)
+ } else if (! args.sequential) {
+ val sequentialBuildTime = ((projectInfos.values.sumByDouble { it.durationMillis.toDouble() }) / 1000)
+ .toInt()
+ String.format("PARALLEL BUILD SUCCESSFUL (%d SECONDS), sequential build would have taken %d seconds",
+ buildTime, sequentialBuildTime)
+ } else {
+ String.format("BUILD SUCCESSFUL (%d SECONDS)", buildTime)
+ }
+ kobaltLog(1, message)
-
-// override fun projectStart(project: Project, context: KobaltContext) {}
-
- private val projectStatuses = arrayListOf>()
-
- override fun projectEnd(project: Project, context: KobaltContext, status: ProjectBuildStatus) {
- projectStatuses.add(Pair(project, status))
}
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/CompilerUtils.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/CompilerUtils.kt
index bab12aee..758a10e9 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/CompilerUtils.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/CompilerUtils.kt
@@ -3,11 +3,9 @@ package com.beust.kobalt.internal
import com.beust.kobalt.TaskResult
import com.beust.kobalt.api.*
import com.beust.kobalt.maven.DependencyManager
-import com.beust.kobalt.maven.DependencyManager2
import com.beust.kobalt.maven.aether.Scope
import com.beust.kobalt.maven.dependency.FileDependency
import com.beust.kobalt.misc.KFiles
-import com.beust.kobalt.misc.log
import com.google.inject.Inject
import java.io.File
import java.nio.file.Paths
@@ -16,14 +14,12 @@ import java.util.*
/**
* Central place to compile files, used by plug-ins and non plug-ins.
*/
-class CompilerUtils @Inject constructor(val files: KFiles,
- val dependencyManager: DependencyManager,
- val dependencyManager2: DependencyManager2) {
+class CompilerUtils @Inject constructor(val files: KFiles, val dependencyManager: DependencyManager) {
class CompilerResult(val successResults: List, val failedResult: TaskResult?)
fun invokeCompiler(project: Project, context: KobaltContext, compiler: ICompilerDescription,
- sourceDirectories: List, isTest: Boolean): CompilerResult {
+ sourceDirectories: List, isTest: Boolean, buildDirectory: File): CompilerResult {
val results = arrayListOf()
var failedResult: TaskResult? = null
val contributedSourceDirs =
@@ -34,18 +30,19 @@ class CompilerUtils @Inject constructor(val files: KFiles,
}
val sourceFiles = KFiles.findSourceFiles(project.directory,
contributedSourceDirs.map { it.path }, compiler.sourceSuffixes)
- if (sourceFiles.size > 0) {
+ if (sourceFiles.isNotEmpty()) {
// TODO: createCompilerActionInfo recalculates the source files, only compute them
// once and pass them
val info = createCompilerActionInfo(project, context, compiler, isTest,
- sourceDirectories, sourceSuffixes = compiler.sourceSuffixes)
+ sourceDirectories, sourceSuffixes = compiler.sourceSuffixes, buildDirectory = buildDirectory)
val thisResult = invokeCompiler(project, context, compiler, info)
results.addAll(thisResult.successResults)
if (failedResult == null) {
failedResult = thisResult.failedResult
}
} else {
- log(2, "${compiler.name} compiler not running on ${project.name} since no source files were found")
+ context.logger.log(project.name, 2,
+ "${compiler.name} compiler not running on ${project.name} since no source files were found")
}
return CompilerResult(results, failedResult)
@@ -68,32 +65,22 @@ class CompilerUtils @Inject constructor(val files: KFiles,
* Runs all the contributors and interceptors relevant to that task.
*/
fun createCompilerActionInfo(project: Project, context: KobaltContext, compiler: ICompilerDescription,
- isTest: Boolean, sourceDirectories: List, sourceSuffixes: List): CompilerActionInfo {
+ isTest: Boolean, sourceDirectories: List, sourceSuffixes: List, buildDirectory: File)
+ : CompilerActionInfo {
copyResources(project, context, SourceSet.of(isTest))
- val fullClasspath = dependencyManager2.resolve(project, context, isTest, listOf(Scope.COMPILE, Scope.TEST))
+ val fullClasspath = dependencyManager.calculateDependencies(project, context,
+ scopes = if (isTest) {
+ listOf(Scope.COMPILE, Scope.COMPILEONLY, Scope.TEST)
+ } else {
+ listOf(Scope.COMPILE, Scope.COMPILEONLY)
+ })
-// if (project.name == "ktor-core" && isTest) {
-// val contains = fullClasspath.filter{it.id.contains("json")}
-// val fc1 =
-// if (isTest) dependencyManager.testDependencies(project, context)
-// else dependencyManager.dependencies(project, context)
-//
-// println("DONOTCOMMIT")
-// val d2 = dependencyManager2.resolve(project, context, isTest, listOf(Scope.COMPILE, Scope.TEST))
-// }
-
- // The directory where the classes get compiled
- val buildDirectory =
- if (isTest) File(project.buildDirectory, KFiles.TEST_CLASSES_DIR)
- else File(project.classesDir(context))
File(project.directory, buildDirectory.path).mkdirs()
// Remove all the excluded dependencies from the classpath
- var classpath = fullClasspath.filter {
- ! isDependencyExcluded(it, project.excludedDependencies)
- }
+ var classpath = fullClasspath
// The classpath needs to contain $buildDirectory/classes as well so that projects that contain
// multiple languages can use classes compiled by the compiler run before them.
@@ -137,7 +124,8 @@ class CompilerUtils @Inject constructor(val files: KFiles,
// depending on the compiler's ability, sourceFiles can actually contain a list of directories
// instead of individual source files.
val projectDirectory = File(project.directory)
- val sourceFiles = if (compiler.canCompileDirectories) {
+ val sourceFiles =
+ if (compiler.canCompileDirectories) {
allSourceDirectories.map { File(projectDirectory, it.path).path }
} else {
files.findRecursively(projectDirectory, allSourceDirectories,
@@ -185,7 +173,7 @@ class CompilerUtils @Inject constructor(val files: KFiles,
// Finally, alter the info with the compiler interceptors before returning it
val initialActionInfo = CompilerActionInfo(projectDirectory.path, classpath, allSources,
sourceSuffixes, buildDirectory, emptyList() /* the flags will be provided by flag contributors */,
- emptyList())
+ emptyList(), context.internalContext.forceRecompile)
val result = context.pluginInfo.compilerInterceptors.fold(initialActionInfo, { ai, interceptor ->
interceptor.intercept(project, context, ai)
})
@@ -205,27 +193,40 @@ class CompilerUtils @Inject constructor(val files: KFiles,
val outputDir = sourceSet.outputDir
val variantSourceDirs = context.variant.resourceDirectories(project, sourceSet)
- if (variantSourceDirs.size > 0) {
- JvmCompilerPlugin.lp(project, "Copying $sourceSet resources")
+ if (variantSourceDirs.isNotEmpty()) {
+ context.logger.log(project.name, 2, "Copying $sourceSet resources")
val absOutputDir = File(KFiles.joinDir(project.directory, project.buildDirectory, outputDir))
- variantSourceDirs.map { File(project.directory, it.path) }.filter {
- it.exists()
- }.forEach {
- log(2, "Copying from $it to $absOutputDir")
- KFiles.copyRecursively(it, absOutputDir, deleteFirst = false)
- }
+ variantSourceDirs
+ .map { File(project.directory, it.path) }
+ .filter(File::exists)
+ .forEach {
+ context.logger.log(project.name, 2, "Copying from $it to $absOutputDir")
+ KFiles.copyRecursively(it, absOutputDir, replaceExisting = true)
+ }
} else {
- JvmCompilerPlugin.lp(project, "No resources to copy for $sourceSet")
+ context.logger.log(project.name, 2, "No resources to copy for $sourceSet")
}
}
+ fun sourceCompilerFlags(project: Project?, context: KobaltContext, info: CompilerActionInfo) : List {
+ val adapters = context.pluginInfo.compilerFlagContributors.map {
+ val closure = { project: Project, context: KobaltContext, currentFlags: List,
+ suffixesBeingCompiled: List
+ -> it.compilerFlagsFor(project, context, currentFlags, suffixesBeingCompiled) }
+ FlagContributor(it.flagPriority, closure)
+ }
+ return compilerFlags(project, context, info, adapters)
+ }
- /**
- * Naïve implementation: just exclude all dependencies that start with one of the excluded dependencies.
- * Should probably make exclusion more generic (full on string) or allow exclusion to be specified
- * formally by groupId or artifactId.
- */
- private fun isDependencyExcluded(id: IClasspathDependency, excluded: List)
- = excluded.any { id.id.startsWith(it.id) }
-
+ fun compilerFlags(project: Project?, context: KobaltContext, info: CompilerActionInfo,
+ adapters: List) : List {
+ val result = arrayListOf()
+ if (project != null) {
+ adapters.sortedBy { it.flagPriority }
+ adapters.forEach {
+ result.addAll(it.flagsFor(project, context, result, info.suffixesBeingCompiled))
+ }
+ }
+ return result
+ }
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/DocUrl.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/DocUrl.kt
index 8acab0ab..93010294 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/DocUrl.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/DocUrl.kt
@@ -2,7 +2,7 @@ package com.beust.kobalt.internal
class DocUrl {
companion object {
- private const val HOST = "http://beust.com/kobalt/"
+ private const val HOST = "https://beust.com/kobalt/"
private fun url(path: String) = HOST + path
val PUBLISH_PLUGIN_URL = url("plug-ins/index.html#publishing")
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/DynamicGraph.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/DynamicGraph.kt
index a363d878..a3e26afd 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/DynamicGraph.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/DynamicGraph.kt
@@ -1,21 +1,19 @@
package com.beust.kobalt.internal
-import com.beust.kobalt.KobaltException
-import com.beust.kobalt.TaskResult
-import com.beust.kobalt.misc.NamedThreadFactory
-import com.beust.kobalt.misc.error
-import com.beust.kobalt.misc.log
+import com.beust.kobalt.*
+import com.beust.kobalt.misc.*
import com.google.common.collect.HashMultimap
import java.lang.reflect.InvocationTargetException
import java.util.*
import java.util.concurrent.*
-open class TaskResult2(success: Boolean, errorMessage: String?, val value: T) : TaskResult(success, errorMessage) {
+open class TaskResult2(success: Boolean, testResult: TestResult? = null,
+ errorMessage: String? = null, val value: T) : TaskResult(success, testResult, errorMessage) {
override fun toString() = com.beust.kobalt.misc.toString("TaskResult", "value", value, "success", success)
}
class DynamicGraph {
- val VERBOSE = 2
+ val VERBOSE = 3
val values : Collection get() = nodes.map { it.value }
val nodes = hashSetOf>()
private val dependedUpon = HashMultimap.create, PrivateNode>()
@@ -63,16 +61,19 @@ class DynamicGraph {
}
}
- fun transitiveClosureGraph(roots: List, childrenFor: (T) -> List) : List>
- = roots.map { transitiveClosureGraph(it, childrenFor) }
+ fun transitiveClosureGraph(roots: List, childrenFor: (T) -> List,
+ filter: (T) -> Boolean): List>
+ = roots.map { transitiveClosureGraph(it, childrenFor, filter) }
- fun transitiveClosureGraph(root: T, childrenFor: (T) -> List, seen: HashSet = hashSetOf()) : Node {
+ fun transitiveClosureGraph(root: T, childrenFor: (T) -> List,
+ filter: (T) -> Boolean = { t: T -> true },
+ seen: HashSet = hashSetOf()) : Node {
val children = arrayListOf>()
- childrenFor(root).forEach { child ->
+ childrenFor(root).filter(filter).forEach { child ->
if (! seen.contains(child)) {
- val c = transitiveClosureGraph(child, childrenFor)
- children.add(c)
seen.add(child)
+ val c = transitiveClosureGraph(child, childrenFor, filter, seen)
+ children.add(c)
}
}
return Node(root, children)
@@ -89,7 +90,7 @@ class DynamicGraph {
}
fun removeNode(t: T) = synchronized(nodes) {
- log(VERBOSE, " Removing node $t")
+ kobaltLog(VERBOSE, " Removing node $t")
PrivateNode(t).let { node ->
nodes.remove(node)
dependingOn.removeAll(node)
@@ -129,7 +130,7 @@ class DynamicGraph {
}
}
val result = nodes.map { it.value }.filter { !nonFree.contains(it) }.toHashSet()
- log(VERBOSE, " Free nodes: $result")
+ kobaltLog(VERBOSE, " Free nodes: $result")
return result
}
}
@@ -162,6 +163,8 @@ interface IWorker : Callable> {
*/
// val tasks : List
+ val name: String
+
/**
* @return the priority of this task.
*/
@@ -181,11 +184,16 @@ interface IThreadWorkerFactory {
}
class DynamicGraphExecutor(val graph : DynamicGraph, val factory: IThreadWorkerFactory,
- threadCount: Int = 1) {
+ val threadCount: Int = 1) {
val executor : ExecutorService
= Executors.newFixedThreadPool(threadCount, NamedThreadFactory("DynamicGraphExecutor"))
val completion = ExecutorCompletionService>(executor)
+ data class HistoryLog(val name: String, val timestamp: Long, val threadId: Long, val start: Boolean)
+
+ val historyLog = arrayListOf()
+ val threadIds = ConcurrentHashMap()
+
fun run() : TaskResult {
try {
return run2()
@@ -201,7 +209,24 @@ class DynamicGraphExecutor(val graph : DynamicGraph, val factory: IThreadW
val newFreeNodes = HashSet(graph.freeNodes)
while (failedResult == null && (running > 0 || newFreeNodes.size > 0)) {
nodesRun.addAll(newFreeNodes)
- val callables : List> = factory.createWorkers(newFreeNodes)
+ val callables : List> = factory.createWorkers(newFreeNodes).map {
+ it -> object: IWorker {
+ override val priority: Int
+ get() = it.priority
+
+ override val name: String get() = it.name
+ override fun call(): TaskResult2 {
+ val threadId = Thread.currentThread().id
+ historyLog.add(HistoryLog(it.name, System.currentTimeMillis(), threadId,
+ start = true))
+ threadIds.put(threadId, threadId)
+ val result = it.call()
+ historyLog.add(HistoryLog(it.name, System.currentTimeMillis(), Thread.currentThread().id,
+ start = false))
+ return result
+ }
+ }
+ }
callables.forEach { completion.submit(it) }
running += callables.size
@@ -211,19 +236,19 @@ class DynamicGraphExecutor(val graph : DynamicGraph, val factory: IThreadW
running--
if (taskResult.success) {
nodesRun.add(taskResult.value)
- log(2, "Task succeeded: $taskResult")
+ kobaltLog(3, "Task succeeded: $taskResult")
graph.removeNode(taskResult.value)
newFreeNodes.clear()
newFreeNodes.addAll(graph.freeNodes.minus(nodesRun))
} else {
- log(2, "Task failed: $taskResult")
+ kobaltLog(3, "Task failed: $taskResult")
newFreeNodes.clear()
if (failedResult == null) {
failedResult = taskResult
}
}
} catch(ex: TimeoutException) {
- log(2, "Time out")
+ kobaltLog(3, "Time out")
} catch(ex: Exception) {
val ite = ex.cause
if (ite is InvocationTargetException) {
@@ -241,6 +266,116 @@ class DynamicGraphExecutor(val graph : DynamicGraph, val factory: IThreadW
}
return if (failedResult != null) failedResult else TaskResult()
}
+
+ fun dumpHistory() {
+ kobaltLog(1, "Thread report")
+
+ val table = AsciiTable.Builder()
+ .columnWidth(11)
+ threadIds.keys.forEach {
+ table.columnWidth(24)
+ }
+ table.header("Time (sec)")
+ threadIds.keys.forEach {
+ table.header("Thread " + it.toString())
+ }
+
+ fun toSeconds(millis: Long) = (millis / 1000).toInt().toString()
+
+ fun displayCompressedLog(table: AsciiTable.Builder) : AsciiTable.Builder {
+ data class CompressedLog(val timestamp: Long, val threadMap: HashMap)
+
+ fun compressLog(historyLog: List): ArrayList {
+ val compressed = arrayListOf()
+
+ var currentLog: CompressedLog? = null
+
+ val projectStart = hashMapOf()
+ fun toName(hl: HistoryLog) : String {
+ var duration = ""
+ if (! hl.start) {
+ val start = projectStart[hl.name]
+ if (start != null) {
+ duration = " (" + ((hl.timestamp - start) / 1000)
+ .toInt().toString() + ")"
+ } else {
+ kobaltLog(1, "DONOTCOMMIT")
+ }
+ }
+ return hl.name + duration
+ }
+
+ historyLog.forEach { hl ->
+ kobaltLog(1, "CURRENT LOG: " + currentLog + " HISTORY LINE: " + hl)
+ if (hl.start) {
+ projectStart[hl.name] = hl.timestamp
+ }
+ if (currentLog == null) {
+ currentLog = CompressedLog(hl.timestamp, hashMapOf(hl.threadId to hl.name))
+ } else currentLog?.let { cl ->
+ if (! hl.start || hl.timestamp - cl.timestamp < 1000) {
+ kobaltLog(1, " CURRENT LOG IS WITHING ONE SECOND: $hl")
+ cl.threadMap[hl.threadId] = toName(hl)
+ } else {
+ kobaltLog(1, " ADDING COMPRESSED LINE $cl")
+ compressed.add(cl)
+ currentLog = CompressedLog(hl.timestamp, hashMapOf(hl.threadId to toName(hl)))
+ }
+ }
+ }
+ return compressed
+ }
+
+ compressLog(historyLog).forEach {
+ val row = arrayListOf()
+ row.add(toSeconds(it.timestamp))
+ it.threadMap.values.forEach {
+ row.add(it)
+ }
+ table.addRow(row)
+ }
+
+ return table
+ }
+
+ fun displayRegularLog(table: AsciiTable.Builder) : AsciiTable.Builder {
+ if (historyLog.any()) {
+ if (historyLog[0] != null) {
+ val start = historyLog[0].timestamp
+ val projectStart = ConcurrentHashMap()
+ historyLog.forEach { line ->
+ val row = arrayListOf()
+ row.add(toSeconds(line.timestamp - start))
+ threadIds.keys.forEach {
+ if (line.threadId == it) {
+ var duration = ""
+ if (line.start) {
+ projectStart[line.name] = line.timestamp
+ } else {
+ val projectStart = projectStart[line.name]
+ if (projectStart != null) {
+ duration = " (" + ((line.timestamp - projectStart) / 1000)
+ .toInt().toString() + ")"
+ } else {
+ warn("Couldn't determine project start: " + line.name)
+ }
+ }
+ row.add((line.name + duration))
+ } else {
+ row.add("")
+ }
+ }
+ table.addRow(row)
+ }
+ } else {
+ warn("Couldn't find historyLog")
+ }
+ }
+ return table
+ }
+
+ kobaltLog(1, displayRegularLog(table).build())
+ }
}
fun main(argv: Array) {
@@ -258,11 +393,12 @@ fun main(argv: Array) {
return nodes.map {
object: IWorker {
override fun call(): TaskResult2? {
- log(1, " Running worker $it")
- return TaskResult2(true, null, it)
+ kobaltLog(1, " Running worker $it")
+ return TaskResult2(true, value = it)
}
override val priority: Int get() = 0
+ override val name: String = "workerName"
}
}
}
diff --git a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/GenericRunner.kt b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/GenericRunner.kt
index 77e45e49..995dba53 100644
--- a/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/GenericRunner.kt
+++ b/modules/kobalt-plugin-api/src/main/kotlin/com/beust/kobalt/internal/GenericRunner.kt
@@ -2,10 +2,9 @@ package com.beust.kobalt.internal
import com.beust.kobalt.*
import com.beust.kobalt.api.*
-import com.beust.kobalt.maven.DependencyManager2
import com.beust.kobalt.misc.KFiles
-import com.beust.kobalt.misc.log
import com.google.common.annotations.VisibleForTesting
+import com.google.inject.Inject
import java.io.File
import java.util.*
@@ -17,17 +16,31 @@ abstract class GenericTestRunner: ITestRunnerContributor {
abstract val dependencyName : String
abstract val mainClass: String
abstract val annotationPackage: String
+ abstract val runnerName: String
+ open var shortMessage: String? = null
+ open var longMessage: String? = null
+
+ @Inject
+ private lateinit var jvm: Jvm
+
abstract fun args(project: Project, context: KobaltContext, classpath: List,
testConfig: TestConfig) : List
+ open fun onFinish(project: Project) {}
+
+ open val extraClasspath: List = emptyList()
+
+ open fun filterTestClasses(project: Project, context: KobaltContext, classes: List) : List = classes
+
override fun run(project: Project, context: KobaltContext, configName: String,
- classpath: List)
- = TaskResult(runTests(project, context, classpath, configName))
+ classpath: List) : TaskResult {
+ val tr = runTests(project, context, classpath, configName)
+ return TaskResult(tr.success, testResult = tr)
+ }
override fun affinity(project: Project, context: KobaltContext) : Int {
- val td = Kobalt.INJECTOR.getInstance(DependencyManager2::class.java).resolve(project, context, isTest = true)
val result =
- if (td.any { it.id.contains(dependencyName) }) IAffinity.DEFAULT_POSITIVE_AFFINITY
+ if (project.testDependencies.any { it.id.contains(dependencyName) }) IAffinity.DEFAULT_POSITIVE_AFFINITY
else 0
return result
}
@@ -55,8 +68,8 @@ abstract class GenericTestRunner: ITestRunnerContributor {
// result
// }
- log(2, "Found ${result.size} test classes")
- return result.map { it.second }
+ context.logger.log(project.name, 2, "Found ${result.size} test classes")
+ return filterTestClasses(project, context, result.map { it.second })
}
/**
@@ -95,17 +108,20 @@ abstract class GenericTestRunner: ITestRunnerContributor {
/**
* @return true if all the tests passed
*/
- fun runTests(project: Project, context: KobaltContext, classpath: List,
- configName: String) : Boolean {
+ open fun runTests(project: Project, context: KobaltContext, classpath: List,
+ configName: String) : TestResult {
var result = false
+ context.logger.log(project.name, 1, "Running tests with $runnerName")
+
val testConfig = project.testConfigs.firstOrNull { it.name == configName }
+ var errorCode = -1
if (testConfig != null) {
val args = args(project, context, classpath, testConfig)
if (args.size > 0) {
- val java = JavaInfo.create(File(SystemProperties.javaBase)).javaExecutable
+ val java = jvm.javaExecutable
val jvmArgs = calculateAllJvmArgs(project, context, testConfig, classpath,
Kobalt.INJECTOR.getInstance (PluginInfo::class.java))
val allArgs = arrayListOf().apply {
@@ -118,24 +134,28 @@ abstract class GenericTestRunner: ITestRunnerContributor {
val pb = ProcessBuilder(allArgs)
pb.directory(File(project.directory))
pb.inheritIO()
- log(2, "Running tests with classpath size ${classpath.size}")
- log(2, "Launching " + allArgs.joinToString(" "))
+ context.logger.log(project.name, 2, "Running tests with classpath size ${classpath.size}")
+ context.logger.log(project.name, 2, "Launching " + allArgs.joinToString(" "))
val process = pb.start()
- val errorCode = process.waitFor()
- if (errorCode == 0) {
- log(1, "All tests passed")
- } else {
- log(1, "Test failures")
- }
+ errorCode = process.waitFor()
result = result || errorCode == 0
} else {
- log(1, " No tests to run")
+ context.logger.log(project.name, 1, " No tests to run")
result = true
}
} else {
throw KobaltException("Couldn't find a test configuration named \"$configName\"")
}
- return result
+
+ onFinish(project)
+
+ if (errorCode == 0) {
+ context.logger.log(project.name, 1, "All tests passed")
+ } else {
+ context.logger.log(project.name, 1, longMessage!!)
+ }
+
+ return TestResult(result, shortMessage, longMessage)
}
/*
@@ -144,11 +164,13 @@ abstract class GenericTestRunner: ITestRunnerContributor {
@VisibleForTesting
fun calculateAllJvmArgs(project: Project, context: KobaltContext,
testConfig: TestConfig, classpath: List, pluginInfo: IPluginInfo) : List {
+ val fullClasspath = classpath.map { it.jarFile.get().absolutePath } + extraClasspath
// Default JVM args
val jvmFlags = arrayListOf