The Scala Plugin
- Usage
- Tasks
- Project layout
- Dependency management
- Automatic configuration of scalaClasspath
- Configuring the Zinc compiler
- Adding plugins to the Scala compiler
- Convention properties
- Source set properties
- Target bytecode level and Java APIs version
- Compiling in external process
- Incremental compilation
- Eclipse Integration
- IntelliJ IDEA Integration
The Scala plugin extends the Java plugin to add support for Scala projects. The plugin also supports joint compilation, which allows you to freely mix and match Scala and Java code with dependencies in both directions. For example, a Scala class can extend a Java class that in turn extends a Scala class. This makes it possible to use the best language for the job, and to rewrite any class in the other language if needed.
Note that if you want to benefit from the API / implementation separation, you can also apply the java-library
plugin to your Scala project.
Usage
To use the Scala plugin, include the following in your build script:
plugins {
scala
}
plugins {
id 'scala'
}
Tasks
The Scala plugin adds the following tasks to the project. Information about altering the dependencies to Java compile tasks are found here.
compileScala
— ScalaCompile-
Depends on:
compileJava
Compiles production Scala source files.
compileTestScala
— ScalaCompile-
Depends on:
compileTestJava
Compiles test Scala source files.
compileSourceSetScala
— ScalaCompile-
Depends on:
compileSourceSetJava
Compiles the given source set’s Scala source files.
scaladoc
— ScalaDoc-
Generates API documentation for the production Scala source files.
The ScalaCompile
and ScalaDoc
tasks support Java toolchains out of the box.
The Scala plugin adds the following dependencies to tasks added by the Java plugin.
Task name | Depends on |
---|---|
|
|
|
|
|
|
Project layout
The Scala plugin assumes the project layout shown below. All the Scala source directories can contain Scala and Java code. The Java source directories may only contain Java source code. None of these directories need to exist or have anything in them; the Scala plugin will simply compile whatever it finds.
src/main/java
-
Production Java source.
src/main/resources
-
Production resources, such as XML and properties files.
src/main/scala
-
Production Scala source. May also contain Java source files for joint compilation.
src/test/java
-
Test Java source.
src/test/resources
-
Test resources.
src/test/scala
-
Test Scala source. May also contain Java source files for joint compilation.
src/sourceSet/java
-
Java source for the source set named sourceSet.
src/sourceSet/resources
-
Resources for the source set named sourceSet.
src/sourceSet/scala
-
Scala source files for the given source set. May also contain Java source files for joint compilation.
Changing the project layout
Just like the Java plugin, the Scala plugin allows you to configure custom locations for Scala production and test source files.
sourceSets {
main {
scala {
setSrcDirs(listOf("src/scala"))
}
}
test {
scala {
setSrcDirs(listOf("test/scala"))
}
}
}
sourceSets {
main {
scala {
srcDirs = ['src/scala']
}
}
test {
scala {
srcDirs = ['test/scala']
}
}
}
Dependency management
Scala projects need to declare a scala-library
dependency. This dependency will then be used on compile and runtime class paths. It will also be used to get hold of the Scala compiler and Scaladoc tool, respectively.[1]
If Scala is used for production code, the scala-library
dependency should be added to the implementation
configuration:
repositories {
mavenCentral()
}
dependencies {
implementation("org.scala-lang:scala-library:2.13.12")
testImplementation("junit:junit:4.13")
}
repositories {
mavenCentral()
}
dependencies {
implementation 'org.scala-lang:scala-library:2.13.12'
testImplementation 'junit:junit:4.13'
}
If you want to use Scala 3 instead of the scala-library
dependency you should add the scala3-library_3
dependency:
plugins {
scala
}
repositories {
mavenCentral()
}
dependencies {
implementation("org.scala-lang:scala3-library_3:3.0.1")
testImplementation("org.scalatest:scalatest_3:3.2.9")
testImplementation("junit:junit:4.13")
}
dependencies {
implementation("commons-collections:commons-collections:3.2.2")
}
plugins {
id 'scala'
}
repositories {
mavenCentral()
}
dependencies {
implementation 'org.scala-lang:scala3-library_3:3.0.1'
implementation 'commons-collections:commons-collections:3.2.2'
testImplementation 'org.scalatest:scalatest_3:3.2.9'
testImplementation 'junit:junit:4.13'
}
If Scala is only used for test code, the scala-library
dependency should be added to the testImplementation
configuration:
dependencies {
testImplementation("org.scala-lang:scala-library:2.13.12")
}
dependencies {
testImplementation 'org.scala-lang:scala-library:2.13.12'
}
Automatic configuration of scalaClasspath
The ScalaCompile
and ScalaDoc
tasks consume Scala code in two ways: on their classpath
, and on their scalaClasspath
. The former is used to locate classes referenced by the source code, and will typically contain scala-library
along with other libraries. The latter is used to load and execute the Scala compiler and Scaladoc tool, respectively, and should only contain the scala-compiler
library and its dependencies.
Unless a task’s scalaClasspath
is configured explicitly, the Scala (base) plugin will try to infer it from the task’s classpath
. This is done as follows:
-
If a
scala-library
jar is found onclasspath
, and the project has at least one repository declared, a correspondingscala-compiler
repository dependency will be added toscalaClasspath
. -
Otherwise, execution of the task will fail with a message saying that
scalaClasspath
could not be inferred.
Configuring the Zinc compiler
The Scala plugin uses a configuration named zinc
to resolve the Zinc compiler and its dependencies.
Gradle will provide a default version of Zinc, but if you need to use a particular Zinc version, you can change it.
Gradle supports version 1.6.0 of Zinc and above.
scala {
zincVersion = "1.10.4"
}
scala {
zincVersion = "1.10.4"
}
The Zinc compiler itself needs a compatible version of scala-library
that may be different from the version required by your application.
Gradle takes care of specifying a compatible version of scala-library
for you.
You can diagnose problems with the version of the Zinc compiler selected by running dependencyInsight for the zinc
configuration.
Gradle version | Supported Zinc versions | Zinc coordinates | Required Scala version | Supported Scala compilation version |
---|---|---|---|---|
7.5 and newer |
SBT Zinc. Versions 1.6.0 and above. |
|
Scala |
Scala |
6.0 to 7.5 |
SBT Zinc. Versions 1.2.0 and above. |
|
Scala |
Scala |
1.x through 5.x |
Deprecated Typesafe Zinc compiler. Versions 0.3.0 and above, except for 0.3.2 through 0.3.5.2. |
|
Scala |
Scala |
Adding plugins to the Scala compiler
The Scala plugin adds a configuration named scalaCompilerPlugins
which is used to declare and resolve optional compiler plugins.
dependencies {
implementation("org.scala-lang:scala-library:2.13.12")
scalaCompilerPlugins("org.typelevel:kind-projector_2.13.12:0.13.2")
}
dependencies {
implementation "org.scala-lang:scala-library:2.13.12"
scalaCompilerPlugins "org.typelevel:kind-projector_2.13.12:0.13.2"
}
Convention properties
The Scala plugin does not add any convention properties to the project.
Source set properties
The Scala plugin adds the following extensions to each source set in the project. You can use these in your build script as though they were properties of the source set object.
scala
— SourceDirectorySet (read-only)-
The Scala source files of this source set. Contains all
.scala
and.java
files found in the Scala source directories, and excludes all other types of files. Default value: non-null. scala.srcDirs
—Set<File>
-
The source directories containing the Scala source files of this source set. May also contain Java source files for joint compilation. Can set using anything described in Understanding implicit conversion to file collections. Default value:
[projectDir/src/name/scala]
. allScala
— FileTree (read-only)-
All Scala source files of this source set. Contains only the
.scala
files found in the Scala source directories. Default value: non-null.
These extensions are backed by an object of type ScalaSourceSet.
The Scala plugin also modifies some source set properties:
Property name | Change |
---|---|
|
Adds all |
|
Adds all source files found in the Scala source directories. |
Target bytecode level and Java APIs version
When running the Scala compile task, Gradle will always add a parameter to configure the Java target for the Scala compiler that is derived from the Gradle configuration:
-
When using toolchains, the
-release
option, ortarget
for older Scala versions, is selected, with a version matching the Java language level of the toolchain configured. -
When not using toolchains, Gradle will always pass a
target
flag — with exact value dependent on the Scala version — to compile to Java 8 bytecode.
This means that using toolchains with a recent Java version and an old Scala version can result in failures because Scala only supported Java 8 bytecode for some time. The solution is then to either use the right Java version in the toolchain or explicitly downgrade the target when needed. |
The following table explains the values computed by Gradle:
Scala version | Toolchain in use | Parameter value |
---|---|---|
version < |
yes |
|
no |
|
|
|
yes |
|
no |
|
|
|
yes |
|
no |
|
|
|
yes |
|
no |
|
Setting any of these flags explicitly, or using flags containing java-output-version
, on ScalaCompile.scalaCompileOptions.additionalParameters
disables that logic in favor of the explicit flag.
Compiling in external process
Scala compilation takes place in an external process.
Memory settings for the external process default to the defaults of the JVM. To adjust memory settings, configure the scalaCompileOptions.forkOptions
property as needed:
tasks.withType<ScalaCompile>().configureEach {
scalaCompileOptions.forkOptions.apply {
memoryMaximumSize = "1g"
jvmArgs = listOf("-XX:MaxMetaspaceSize=512m")
}
}
tasks.withType(ScalaCompile) {
scalaCompileOptions.forkOptions.with {
memoryMaximumSize = '1g'
jvmArgs = ['-XX:MaxMetaspaceSize=512m']
}
}
Incremental compilation
By compiling only classes whose source code has changed since the previous compilation, and classes affected by these changes, incremental compilation can significantly reduce Scala compilation time. It is particularly effective when frequently compiling small code increments, as is often done at development time.
The Scala plugin defaults to incremental compilation by integrating with Zinc, a standalone version of sbt's incremental Scala compiler. If you want to disable the incremental compilation, set force = true
in your build file:
tasks.withType<ScalaCompile>().configureEach {
scalaCompileOptions.apply {
isForce = true
}
}
tasks.withType(ScalaCompile) {
scalaCompileOptions.with {
force = true
}
}
Note: This will only cause all classes to be recompiled if at least one input source file has changed. If there are no changes to the source files, the compileScala
task will still be considered UP-TO-DATE
as usual.
The Zinc-based Scala Compiler supports joint compilation of Java and Scala code. By default, all Java and Scala code under src/main/scala
will participate in joint compilation. Even Java code will be compiled incrementally.
Incremental compilation requires dependency analysis of the source code. The results of this analysis are stored in the file designated by scalaCompileOptions.incrementalOptions.analysisFile
(which has a sensible default). In a multi-project build, analysis files are passed on to downstream ScalaCompile
tasks to enable incremental compilation across project boundaries. For ScalaCompile
tasks added by the Scala plugin, no configuration is necessary to make this work. For other ScalaCompile
tasks that you might add, the property scalaCompileOptions.incrementalOptions.publishedCode
needs to be configured to point to the classes folder or Jar archive by which the code is passed on to compile class paths of downstream ScalaCompile
tasks. Note that if publishedCode
is not set correctly, downstream tasks may not recompile code affected by upstream changes, leading to incorrect compilation results.
Note that Zinc’s Nailgun based daemon mode is not supported. Instead, we plan to enhance Gradle’s own compiler daemon to stay alive across Gradle invocations, reusing the same Scala compiler. This is expected to yield another significant speedup for Scala compilation.
Eclipse Integration
When the Eclipse plugin encounters a Scala project, it adds additional configuration to make the project work with Scala IDE out of the box. Specifically, the plugin adds a Scala nature and dependency container.
IntelliJ IDEA Integration
When the IDEA plugin encounters a Scala project, it adds additional configuration to make the project work with IDEA out of the box. Specifically, the plugin adds a Scala SDK (IntelliJ IDEA 14+) and a Scala compiler library that matches the Scala version on the project’s class path. The Scala plugin is backwards compatible with earlier versions of IntelliJ IDEA and it is possible to add a Scala facet instead of the default Scala SDK by configuring targetVersion
on IdeaModel.
idea {
targetVersion = "13"
}
idea {
targetVersion = '13'
}