2

I'm using Gradle (v2.3) for a project, that contains multiple Scala sub-projects. Generating the ScalaDoc for each sub-project individually works as expected (running gradle :project-a:scaladoc or gradle :project-b:scaladoc).

But how do a get a single ScalaDoc of all Scala sub-projects? The minimal project below leads to a "Cannot invoke method withInputStream() on null object" error, when executing
gradle scaladoc --info:

Starting Build
Compiling settings file '/tmp/gradle-scaladoc-test/settings.gradle' using StatementExtractingScriptTransformer.
Compiling settings file '/tmp/gradle-scaladoc-test/settings.gradle' using BuildScriptTransformer.
Settings evaluated using settings file '/tmp/gradle-scaladoc-test/settings.gradle'.
Projects loaded. Root project using build file '/tmp/gradle-scaladoc-test/build.gradle'.
Included projects: [root project 'some project', project ':project-a', project ':project-b']
Evaluating root project 'some project' using build file '/tmp/gradle-scaladoc-test/build.gradle'.
Compiling build file '/tmp/gradle-scaladoc-test/build.gradle' using StatementExtractingScriptTransformer.
Compiling build file '/tmp/gradle-scaladoc-test/build.gradle' using BuildScriptTransformer.
Evaluating project ':project-a' using empty build file.
Evaluating project ':project-b' using empty build file.
All projects evaluated.
Selected primary task 'scaladoc' from project :
Tasks to be executed: [task ':scaladoc', task ':project-a:compileJava', task ':project-a:compileScala', task ':project-a:processResources', task ':project-a:classes', task ':project-a:scaladoc', task ':project-b:compileJava', task ':project-b:compileScala', task ':project-b:processResources', task ':project-b:classes', task ':project-b:scaladoc']
:scaladoc (Thread[main,5,main]) started.
:scaladoc
Executing task ':scaladoc' (up-to-date check took 0.448 secs) due to:
  No history is available.
:scaladoc FAILED
:scaladoc (Thread[main,5,main]) completed. Took 0.768 secs.

FAILURE: Build failed with an exception.

* What went wrong:
Execution failed for task ':scaladoc'.
> Cannot invoke method withInputStream() on null object

* Try:
Run with --stacktrace option to get the stack trace. Run with --debug option to get more log output.

BUILD FAILED

Total time: 5.198 secs
Stopped 0 compiler daemon(s).

Directory structure:

├─ project-a/
│   └─ src/
│       └─ main/
│           └─ scala/
│               └─ package_a/
│                   └─ A.scala
├─ project-b/
│   └─ src/
│       └─ main/
│           └─ scala/
│               └─ package_b/
│                   └─ B.scala
├─ build.gradle
└─ settings.gradle

build.gradle

subprojects {
  repositories { 
    mavenCentral() 
  }

  apply plugin: 'scala'

  dependencies {
    compile 'org.scala-lang:scala-library:2.11.6'
  }

  tasks.withType(ScalaCompile) {
    scalaCompileOptions.additionalParameters = ['-unchecked', '-deprecation', '-feature']
    scalaCompileOptions.useAnt = false
  }
}


task scaladoc(type: org.gradle.api.tasks.scala.ScalaDoc) {
  group = 'Documentation'
  description = 'Aggregated ScalaDoc documentation.'
  title = 'Title of documentation'
  destinationDir = new File(buildDir, "aggregated-api")

  source subprojects.collect { project ->
    project.sourceSets.main.allScala
  }

  scalaClasspath = files(subprojects.collect { project ->
    project.sourceSets.main.compileClasspath
  })

  classpath = files(subprojects.collect { project ->
    project.sourceSets.main.compileClasspath
  })
}

settings.gradle

rootProject.name = 'some project'
include 'project-a', 'project-b'

A.scala

package package_a
case class A(value: Int)

B.scala

package package_b
case class B(value: Int)    

The only similar problem I could find is "Gradle Fails to Compile Basic Scala Project", but it doesn't help here (and is about Gradle 1.3).

Community
  • 1
  • 1
hiddenbit
  • 2,233
  • 14
  • 25
  • I know you are not asking this, but if it was an sbt build, you could leverage the [sbt-unidoc plugin](https://github.com/sbt/sbt-unidoc) – 0__ Apr 06 '15 at 20:57
  • Try adapting one of the many [gradle aggregate javadoc](https://www.google.com/search?q=gradle+aggregate+javadoc) examples. – Ben Manes Apr 06 '15 at 21:43
  • Thanks for the suggestions. sbt is no option, it has to be done with Gradle. – hiddenbit Apr 09 '15 at 19:08

2 Answers2

3

I know it's been a while since this has been asked, but since I just had the exact same problem...

I solved it the following way:

task aggregatedScaladocs(type: ScalaDoc, description: 'Generate scaladocs from all child projects as if it were a single project', group: 'Documentation') {
destinationDir = file("$buildDir/docs/scaladoc")
title = "$project.name $version API"

subprojects.each { proj ->
    proj.tasks.withType(ScalaDoc).each {
        source += proj.sourceSets.main.allJava
        source += proj.sourceSets.main.allScala
        classpath += proj.sourceSets.main.compileClasspath
        excludes += scaladoc.excludes
        includes += scaladoc.includes
    }
}

}

Hope it'll help somebody at some point.

juwi
  • 389
  • 3
  • 16
0

For Gradle 6.x I needed to use slightly modified version of answer by @juwi.

task aggregatedScaladocs(type: ScalaDoc, description: 'Generate scaladocs from all child projects as if it were a single project', group: 'Documentation') {
    destinationDir = file("$buildDir/docs/scaladoc")
    title = "$project.name $version API"
    classpath = project.files([])
    scalaClasspath = project.files([])

    subprojects.each { proj ->
        proj.tasks.withType(ScalaDoc).each {
            source proj.sourceSets.main.allJava
            source proj.sourceSets.main.allScala
            classpath += proj.scaladoc.classpath
            scalaClasspath += proj.scaladoc.scalaClasspath
            exclude proj.scaladoc.excludes
            include proj.scaladoc.includes
        }
    }
}
Ján Čabala
  • 184
  • 1
  • 7