Więc, po tym jak się trochę nad tym zastanowiłem, znalazłem rozwiązanie. Najpierw musisz podzielić projekt na dwa podrzędne projekty. gen
zawiera całe źródło zawierające kod generatora. use
zależy od gen
i używa generatora.
import sbt._
import Keys._
import java.io.{ File ⇒ JFile, FileOutputStream }
object OverallBuild extends Build {
lazy val root = Project(id = "overall", base = file(".")).aggregate(gen, use)
lazy val gen = Project(id = "generate", base = file("gen"))
val myCodeGenerator = TaskKey[Seq[File]]("mycode-generate", "Generate My Awesome Code")
lazy val use = Project(id = "use", base = file("use"),
settings = Defaults.defaultSettings ++ Seq(
sourceGenerators in Compile <+= (myCodeGenerator in Compile),
myCodeGenerator in Compile <<=
(javaSource in Compile, dependencyClasspath in Runtime in gen) map {
(javaSource, cp) ⇒ runMyCodeGenerator(javaSource, cp.files)
})).dependsOn(gen)
def runMyCodeGenerator(javaSource: File, cp: Seq[File]): Seq[File] = {
val mainClass = "com.yourcompany.myCodeGenerator"
val tmp = JFile.createTempFile("sources", ".txt")
val os = new FileOutputStream(tmp)
try {
val i = new Fork.ForkScala(mainClass).fork(None, Nil, cp,
Seq(javaSource.toString),
None,
false,
CustomOutput(os)).exitValue()
if (i != 0) {
error("Trouble with code generator")
}
} finally {
os.close()
}
scala.io.Source.fromFile(tmp).getLines.map(f ⇒ file(f)).toList
}
}
W tym przypadku, byłem generowania plików .java więc zdałem w javaSource
do generatora.
Ważne jest, aby nie było tak, że gdy używamy sourceGenerators, jak jesteśmy tutaj, wykonane zadanie musi zwrócić Seq[File]
ze wszystkich plików, które zostały wygenerowane, aby sbt mógł nimi zarządzać. W tej implementacji nasz generator wyprowadza pełne nazwy plików ścieżek do standardowego wyjścia i zapisujemy je do pliku tymczasowego.
Podobnie jak w przypadku wszystkich rzeczy Scala i na pewno SBT, możesz zrobić wszystko, wystarczy się w to zagłębić.
Ja też zmagałem się z tym dokładnym scenariuszem. Nie mam dla ciebie odpowiedzi, wciąż nowicjusz sbt. Ale będzie też czekał na odpowiedź. –