实际上我正在根据http://www.scala-lang.org/node/140上的文章为Scala开发一个编译器插件。
以下是插件的代码:
package localhost
import scala.tools.nsc
import nsc.Global
import nsc.Phase
import nsc.plugins.Plugin
import nsc.plugins.PluginComponent
class DivByZero(val global: Global) extends Plugin {
import global._
val name = "divbyzero"
val description = "checks for division by zero"
val components = List[PluginComponent](Component)
private object Component extends PluginComponent {
val global: DivByZero.this.global.type = DivByZero.this.global
val runsAfter = "refchecks"
// Using the Scala Compiler 2.8.x the runsAfter should be written as below
// val runsAfter = List[String]("refchecks");
val phaseName = DivByZero.this.name
def newPhase(_prev: Phase) = new DivByZeroPhase(_prev)
class DivByZeroPhase(prev: Phase) extends StdPhase(prev) {
override def name = DivByZero.this.name
def apply(unit: CompilationUnit) {
for ( tree @ Apply(Select(rcvr, nme.DIV), List(Literal(Constant(0)))) <- unit.body;
if rcvr.tpe <:< definitions.IntClass.tpe)
{
unit.error(tree.pos, "definitely division by zero")
}
}
}
}
}
我正在做那里提到的事情并编写了一些makefile来编译所有内容,然后创建一个jar文件。然后我使用以下命令使用testfile加载插件jar文件:
scalac -Xplugin:myplugin.jar test.scala
看看输出是什么。我不喜欢这种方式,因为我从红宝石中知道如何做tdd和bdd。我安装了Scalatest http://www.scalatest.org/。是否有可能测试jar文件或类divbyzero?我知道插件将在使用文件执行时首先加载。我非常关注并且不知道是否可以直接测试插件类而不创建jar文件(或者甚至可以测试jar文件的某些函数和类)?
如果没有人可以帮助我,我可以像过去那样继续发展
感谢您的时间和帮助 的Matthias
答案 0 :(得分:18)
您可以使用以下代码以编程方式调用Scala编译器和插件:
import scala.tools.nsc.{Settings, Global}
import scala.tools.nsc.io.VirtualDirectory
import scala.tools.nsc.reporters.ConsoleReporter
import scala.tools.nsc.util.BatchSourceFile
// prepare the code you want to compile
val code = "object Foo extends Application { println(42 / 0) }"
val sources = List(new BatchSourceFile("<test>", code))
val settings = new Settings
// save class files to a virtual directory in memory
settings.outputDirs.setSingleOutput(new VirtualDirectory("(memory)", None))
val compiler = new Global(settings, new ConsoleReporter(settings)) {
override protected def computeInternalPhases () {
super.computeInternalPhases
for (phase <- new DivByZero(this).components)
phasesSet += phase
}
}
new compiler.Run() compileSources(sources)
请注意,此代码要求在执行代码时scala-compiler.jar
和scala-library.jar
位于类路径上。如果您正在从像SBT这样的东西中运行测试,那很可能不是这种情况。
为了让事情从SBT中运行,你必须做一些跳跃:
val settings = new Settings
val loader = getClass.getClassLoader.asInstanceOf[URLClassLoader]
val entries = loader.getURLs map(_.getPath)
// annoyingly, the Scala library is not in our classpath, so we have to add it manually
val sclpath = entries find(_.endsWith("scala-compiler.jar")) map(
_.replaceAll("scala-compiler.jar", "scala-library.jar"))
settings.classpath.value = ClassPath.join((entries ++ sclpath) : _*)
如果您在其他构建环境中运行,您可能会发现scala-library.jar
已经在类路径中,或者如果您真的很幸运,那么您需要的所有内容都在标准的Java类路径中,其中你可以用以下内容替换上面的内容:
val settings = new Settings
settings.usejavacp.value = true
您可以使用System.getProperty("java.class.path")
打印出Java类路径的值,当然可以从上面的代码中打印出entries
,以查看加载测试代码的类加载器使用的类路径
答案 1 :(得分:5)
我想补充一下samskivert的回答。覆盖computeInternalPhases
强制将插件的各个阶段注入整个phaseSet,但是,编译器不会将它们视为插件的一部分。例如,如果您想使用以下选项将选项传递给插件"-P:divbyzero:someoption"
:
settings.pluginOptions.appendToValue("divbyzero:someoption")
您将收到以下编译错误:
error: bad option: -P:divbyzero:someoption
这是因为编译器对名为divbyzero
的插件一无所知。
添加插件的更合适的方法是覆盖loadRoughPluginsList
方法并在那里添加插件,而不是手动将插件的每个阶段注入编译阶段:
override protected def loadRoughPluginsList: List[Plugin] =
new DivByZero(this) :: super.loadRoughPluginsList
答案 2 :(得分:1)