Динамическая компиляция с помощью scala 2.12 - из теста sbt

Я хочу протестировать инструмент, который я пишу, который преобразует некоторый исходный код scala, я хочу проверить, что преобразованный код компилируется из теста sbt (используя scalatest).

Я хотел бы программно вызвать компилятор scala для строки с моим источником (все находится в отдельном файле).

Я добился некоторого прогресса с момента первоначального вопроса. Этот код работает в моей среде IDE (IntelliJ), но не в сеансе sbt test

import java.io.File
import scala.reflect.internal.util.BatchSourceFile
import scala.tools.nsc.{GenericRunnerSettings, Global}

object Compilation {
  val settings = new GenericRunnerSettings(System.out.println _)
//  val sbtClasspath = System.getProperty("sbt-classpath")
//  val s = File.pathSeparator
//  val classPath = s".${s}$sbtClasspath"
//  settings.classpath.append(classPath)
  settings.usejavacp.value = true
  val global = new Global(settings)

  def compiles(code: String): Boolean = {
    val r = new global.Run
    r.compileSources(List(new BatchSourceFile("<partest>", code)))
    val errors = global.reporter.hasErrors
    if (errors) r.reporting.summarizeErrors()
    !errors
  }
}

Вот мой build.sbt

name := "CodinGame-Scala-Kit"
version := "0.1.0"
scalaVersion := "2.12.2"
libraryDependencies += "org.scalatest" %% "scalatest" % "3.0.1" % "test"
libraryDependencies += "org.scala-lang" % "scala-compiler" % "2.12.2" % "test"

enablePlugins(JmhPlugin)

//val sbtcp = taskKey[Unit]("sbt-classpath")
//
//sbtcp := {
//  val files: Seq[File] = (fullClasspath in Compile).value.files
//  val sbtClasspath: String = files.map(x => x.getAbsolutePath).mkString(java.io.File.pathSeparator)
//  println("Set SBT classpath to 'sbt-classpath' environment variable")
//  println(sbtClasspath)
//  System.setProperty("sbt-classpath", sbtClasspath)
//}
//
//compile <<= (compile in Compile).dependsOn(sbtcp)

Первоначальная ошибка

scala.reflect.internal.MissingRequirementError: object scala.annotation.Annotation in compiler mirror not found.
[info]   at scala.reflect.internal.MissingRequirementError$.signal(MissingRequirementError.scala:17)
[info]   at scala.reflect.internal.MissingRequirementError$.notFound(MissingRequirementError.scala:18)
[info]   at scala.reflect.internal.Mirrors$RootsBase.$anonfun$getModuleOrClass$4(Mirrors.scala:54)
[info]   at scala.reflect.internal.Mirrors$RootsBase.getModuleOrClass(Mirrors.scala:54)
[info]   at scala.reflect.internal.Mirrors$RootsBase.getModuleOrClass(Mirrors.scala:45)
[info]   at scala.reflect.internal.Mirrors$RootsBase.getModuleOrClass(Mirrors.scala:66)
[info]   at scala.reflect.internal.Mirrors$RootsBase.getClassByName(Mirrors.scala:101)
[info]   at scala.reflect.internal.Mirrors$RootsBase.getRequiredClass(Mirrors.scala:104)
[info]   at scala.reflect.internal.Mirrors$RootsBase.requiredClass(Mirrors.scala:107)
[info]   at scala.reflect.internal.Definitions$DefinitionsClass.AnnotationClass$lzycompute(Definitions.scala:1141)

Я добавил строки, прокомментированные в build.sbt и в объекте Compilation после этот ответ. Это ничего не меняет в IDE.

Теперь у меня есть эта ошибка в тесте sbt (компиляция правильно печатает путь к классам):

 scala.reflect.internal.FatalError: class StringContext does not have a member f
[info]   at scala.reflect.internal.Definitions$DefinitionsClass.fatalMissingSymbol(Definitions.scala:1236)
[info]   at scala.reflect.internal.Definitions$DefinitionsClass.$anonfun$getMember$1(Definitions.scala:1259)
[info]   at scala.reflect.internal.Definitions$DefinitionsClass.getMember(Definitions.scala:1254)
[info]   at scala.reflect.internal.Definitions$DefinitionsClass.getMemberMethod(Definitions.scala:1288)
[info]   at scala.reflect.internal.Definitions$DefinitionsClass$RunDefinitions.StringContext_f$lzycompute(Definitions.scala:1477)
[info]   at scala.reflect.internal.Definitions$DefinitionsClass$RunDefinitions.StringContext_f(Definitions.scala:1477)
[info]   at scala.tools.reflect.FastTrack.$anonfun$fastTrackCache$1(FastTrack.scala:53)
[info]   at scala.reflect.internal.SymbolTable$perRunCaches$$anon$1.apply(SymbolTable.scala:395)
[info]   at scala.tools.reflect.FastTrack.contains(FastTrack.scala:20)
[info]   at scala.tools.nsc.typechecker.Namers$Namer.methodSig(Namers.scala:1388)

person Michel Daviot    schedule 27.05.2017    source источник


Ответы (2)


Второй подход похож на то, что сработало для меня. Очевидно, вам где-то не хватает зависимости. Пустые настройки не будут работать. Либо изменить настройки bootclasspath

или просто используйте settings.usejavacp.value = true

person Meir Maor    schedule 27.05.2017

Наконец-то я нашел надежный способ скомпилировать любую строку:

import scala.reflect.runtime.currentMirror
import scala.tools.reflect.ToolBox

object Compiler {

  def compile[A](string: String): A = {
    val toolbox = currentMirror.mkToolBox()
    val tree    = toolbox.parse(string)
    toolbox.eval(tree).asInstanceOf[A]
  }
}

с

libraryDependencies += "org.scala-lang" % "scala-compiler" % "2.12.10"
person Michel Daviot    schedule 18.04.2020