Разбиение скалярного плагина на несколько файлов - PullRequest
6 голосов
/ 12 апреля 2011

Я хотел бы разбить мой плагин scalac на несколько файлов. Это звучит просто, но мне не удалось осуществить его из-за проблем с типом, зависящим от пути, которые возникают из строки import global._.

Вот пример плагина Lex Spoon:

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")
          }
      }
    }
  }
}

Как я могу поместить Component и DivByZeroPhase в их собственные файлы, не имея import global._ в области видимости?

Ответы [ 3 ]

4 голосов
/ 12 апреля 2011

Вот действительно старый проект, в котором я сделал то же самое:

https://github.com/jsuereth/osgi-scalac-plugin/blob/master/src/main/scala/scala/osgi/compiler/OsgiPlugin.scala

Если вам не нужно передавать зависимые от пути типы из глобальных, не беспокойтесь о попытках сохранить актуальность его частей "this.global".

3 голосов
/ 12 апреля 2011

В библиотеке Scala Refactoring я решил эту проблему с помощью свойства CompilerAccess:

trait CompilerAccess { 
  val global: tools.nsc.Global
}

Теперь все остальные черты, к которым требуется доступ global, просто объявляют CompilerAccess как зависимость:

trait TreeTraverser {
  this: CompilerAccess =>
  import global._

  ...
}

и затем есть класс, который смешивает во всех этих чертах и ​​обеспечивает экземпляр global:

trait SomeRefactoring extends TreeTraverser with OtherTrait with MoreTraits {
  val global = //wherever you get your global from
}

Эта схема работала довольно хорошо для меня.

2 голосов
/ 12 апреля 2011

Вы можете создать отдельный класс для вашего компонента и передать глобальный в:

class TemplateComponent(val global: Global) extends PluginComponent {

  import global._

  val runsAfter = List[String]("refchecks")

  val phaseName = "plugintemplate"

  def newPhase(prev: Phase) = new StdPhase(prev) {

    override def name = phaseName

    def apply(unit:CompilationUnit) = {
    }    
  }
}
...