Сбой компилятора Scala: что здесь не так?

Я начал писать модульный тест для класса X500PrincipalBuilder в Scala. Вот мой тестовый код:

import org.junit.runner.RunWith
import org.scalatest.WordSpec
import org.scalatest.junit.JUnitRunner
import org.scalatest.matchers.ShouldMatchers._
import org.scalatest.prop.PropertyChecks._
import javax.security.auth.x500.X500Principal

@RunWith(classOf[JUnitRunner])
class X500PrincipalBuilderTest extends WordSpec {

  "An X500 Principal Builder" when {
    "given a list of attributes" should {
      "properly build an X500 Principal" in {
        val table = Table(
          ("expected", "attributes"),
          ("CN=Duke", List("CN" -> "Duke"))
        )
        forAll (table) { (expected, attributes) =>
          val b = new X500PrincipalBuilder
          attributes foreach { case (key, value) => b addAttribute (key, value) }
          b.build should equal (new X500Principal(expected))
        }
      }
    }
  }
}

Однако при компиляции этого кода с помощью Scala 2.9.2 или 2.10.0 я получаю сбой компилятора. Вот начальная трассировка стека из 2.9.2:

[ERROR] error: java.lang.Error: no-symbol does not have owner
[INFO]  at scala.tools.nsc.symtab.SymbolTable.abort(SymbolTable.scala:34)
[INFO]  at scala.tools.nsc.symtab.Symbols$NoSymbol$.owner(Symbols.scala:2155)
[INFO]  at scala.tools.nsc.symtab.Symbols$Symbol.logicallyEnclosingMember(Symbols.scala:1251)
[INFO]  at scala.tools.nsc.transform.LambdaLift$LambdaLifter.searchIn$1(LambdaLift.scala:234)
[INFO]  at scala.tools.nsc.transform.LambdaLift$LambdaLifter.proxy(LambdaLift.scala:243)
[INFO]  at scala.tools.nsc.transform.LambdaLift$LambdaLifter.scala$tools$nsc$transform$LambdaLift$LambdaLifter$$proxyRef(LambdaLift.scala:259)
[INFO]  at scala.tools.nsc.transform.LambdaLift$LambdaLifter.postTransform(LambdaLift.scala:389)
[INFO]  at scala.tools.nsc.transform.LambdaLift$LambdaLifter.transform(LambdaLift.scala:412)
... (removed some hundred stack frames)

Вывод из 2.10.0, похоже, представляет собой некоторый непонятный псевдокод для генерации AST:

[INFO]
[INFO]      while compiling: /Users/christian/code/truelicense~v2/truelicense-core/src/test/scala/net/java/truelicense/core/util/X500PrincipalBuilderTest.scala
[INFO]         during phase: global=lambdalift, atPhase=constructors
[INFO]      library version: version 2.10.0
[INFO]     compiler version: version 2.10.0
[INFO]   reconstructed args: -deprecation -feature -classpath /Users/christian/code/truelicense~v2/truelicense-core/target/classes:/Users/christian/.m2/repository/net/java/truelicense/truelicense-obfuscate/2.3-SNAPSHOT/truelicense-obfuscate-2.3-SNAPSHOT.jar:/Users/christian/.m2/repository/commons-codec/commons-codec/1.8/commons-codec-1.8.jar:/Users/christian/.m2/repository/org/slf4j/slf4j-simple/1.7.5/slf4j-simple-1.7.5.jar:/Users/christian/.m2/repository/org/slf4j/slf4j-api/1.7.5/slf4j-api-1.7.5.jar:/Users/christian/.m2/repository/com/google/code/findbugs/annotations/2.0.1/annotations-2.0.1.jar:/Users/christian/.m2/repository/junit/junit/4.11/junit-4.11.jar:/Users/christian/.m2/repository/org/hamcrest/hamcrest-core/1.3/hamcrest-core-1.3.jar:/Users/christian/.m2/repository/org/mockito/mockito-core/1.9.5/mockito-core-1.9.5.jar:/Users/christian/.m2/repository/org/objenesis/objenesis/1.3/objenesis-1.3.jar:/Users/christian/.m2/repository/org/scalatest/scalatest_2.10.0/1.8/scalatest_2.10.0-1.8.jar:/Users/christian/.m2/repository/org/scala-lang/scala-library/2.10.0/scala-library-2.10.0.jar:/Users/christian/.m2/repository/org/scala-lang/scala-actors/2.10.0/scala-actors-2.10.0.jar:/Users/christian/.m2/repository/org/scala-lang/scala-reflect/2.10.0/scala-reflect-2.10.0.jar:/Users/christian/.m2/repository/org/scalacheck/scalacheck_2.10.0/1.10.0/scalacheck_2.10.0-1.10.0.jar:/Users/christian/.m2/repository/org/scala-tools/testing/test-interface/0.5/test-interface-0.5.jar:/Users/christian/code/truelicense~v2/truelicense-core/target/test-classes -d /Users/christian/code/truelicense~v2/truelicense-core/target/test-classes
[INFO]
[INFO]   last tree to typer: term $outer
[INFO]               symbol: value $outer (flags: <synthetic> <paramaccessor> <triedcooking> private[this])
[INFO]    symbol definition: private[this] val $outer: net.java.truelicense.core.util.X500PrincipalBuilderTest
[INFO]                  tpe: <notype>
[INFO]        symbol owners: value $outer -> anonymous class $anonfun$1 -> value <local X500PrincipalBuilderTest> -> class X500PrincipalBuilderTest -> package util
[INFO]       context owners: value key -> value $anonfun -> value $anonfun -> method apply -> anonymous class $anonfun$apply$mcV$sp$2 -> method apply$mcV$sp -> anonymous class $anonfun$apply$mcV$sp$1 -> method apply$mcV$sp -> anonymous class $anonfun$1 -> value <local X500PrincipalBuilderTest> -> class X500PrincipalBuilderTest -> package util
[INFO]
[INFO] == Enclosing template or block ==
[INFO]
[INFO] DefDef( // final def apply(x$1: Tuple2): Unit
[INFO]   <method> final <triedcooking>
[INFO]   "apply"
[INFO]   []
[INFO]   // 1 parameter list
[INFO]   ValDef( // x0$2: Tuple2
[INFO]     <param> <synthetic> <triedcooking>
[INFO]     "x0$2"
[INFO]     <tpt> // tree.tpe=Tuple2
[INFO]     <empty>
[INFO]   )
[INFO]   <tpt> // tree.tpe=Unit
[INFO]   Block( // tree.tpe=Unit
[INFO]     // 3 statements
[INFO]     ValDef( // case val x1: Tuple2
[INFO]       case <synthetic> <triedcooking>
[INFO]       "x1"
[INFO]       <tpt> // tree.tpe=Tuple2
[INFO]       "x0$2" // x0$2: Tuple2, tree.tpe=Tuple2
[INFO]     )
[INFO]     LabelDef( // case def case4(): Unit, tree.tpe=Unit
[INFO]       ()
[INFO]       If( // tree.tpe=Unit
[INFO]         Apply( // final def ne(x$1: Object): Boolean in class Object, tree.tpe=Boolean
[INFO]           "x1"."ne" // final def ne(x$1: Object): Boolean in class Object, tree.tpe=(x$1: Object)Boolean
[INFO]           null
[INFO]         )
[INFO]         Block( // tree.tpe=Unit
[INFO]           // 2 statements
[INFO]           ValDef( // val key: String
[INFO]             <triedcooking>
[INFO]             "key"
[INFO]             <tpt> // tree.tpe=String
[INFO]             Apply( // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=String
[INFO]               TypeApply( // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=()String
[INFO]                 x1._1()."$asInstanceOf" // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=[T0 >: ? <: ?]()T0
[INFO]                 <tpt> // tree.tpe=String
[INFO]               )
[INFO]               Nil
[INFO]             )
[INFO]           )
[INFO]           ValDef( // val value: String
[INFO]             <triedcooking>
[INFO]             "value"
[INFO]             <tpt> // tree.tpe=String
[INFO]             Apply( // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=String
[INFO]               TypeApply( // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=()String
[INFO]                 x1._2()."$asInstanceOf" // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=[T0 >: ? <: ?]()T0
[INFO]                 <tpt> // tree.tpe=String
[INFO]               )
[INFO]               Nil
[INFO]             )
[INFO]           )
[INFO]           Apply( // case def matchEnd3(x: runtime.BoxedUnit): Unit, tree.tpe=Unit
[INFO]             "matchEnd3" // case def matchEnd3(x: runtime.BoxedUnit): Unit, tree.tpe=(x: runtime.BoxedUnit)Unit
[INFO]             Block( // tree.tpe=runtime.BoxedUnit
[INFO]               Apply( // def addAttribute(x$1: String,x$2: String): Unit in class X500PrincipalBuilder, tree.tpe=Unit
[INFO]                 "b"."addAttribute" // def addAttribute(x$1: String,x$2: String): Unit in class X500PrincipalBuilder, tree.tpe=(x$1: String, x$2: String)Unit
[INFO]                 // 2 arguments
[INFO]                 "key" // val key: String, tree.tpe=String
[INFO]                 "value" // val value: String, tree.tpe=String
[INFO]               )
[INFO]               "scala"."runtime"."BoxedUnit"."UNIT" // final val UNIT: runtime.BoxedUnit in object BoxedUnit, tree.tpe=runtime.BoxedUnit
[INFO]             )
[INFO]           )
[INFO]         )
[INFO]         Apply( // case def case5(): Unit, tree.tpe=Unit
[INFO]           "case5" // case def case5(): Unit, tree.tpe=()Unit
[INFO]           Nil
[INFO]         )
[INFO]       )
[INFO]     )
[INFO]     LabelDef( // case def case5(): Unit, tree.tpe=Unit
[INFO]       ()
[INFO]       Apply( // case def matchEnd3(x: runtime.BoxedUnit): Unit, tree.tpe=Unit
[INFO]         "matchEnd3" // case def matchEnd3(x: runtime.BoxedUnit): Unit, tree.tpe=(x: runtime.BoxedUnit)Unit
[INFO]         Throw( // tree.tpe=Nothing
[ERROR]           Apply( // def <init>(obj: Object): MatchError in class MatchError, tree.tpe=MatchError
[ERROR]             new MatchError."<init>" // def <init>(obj: Object): MatchError in class MatchError, tree.tpe=(obj: Object)MatchError
[INFO]             "x1" // case val x1: Tuple2, tree.tpe=Tuple2
[INFO]           )
[INFO]         )
[INFO]       )
[INFO]     )
[INFO]     LabelDef( // case def matchEnd3(x: runtime.BoxedUnit): Unit, tree.tpe=Unit
[INFO]       "x" // x: runtime.BoxedUnit, tree.tpe=runtime.BoxedUnit
[INFO]       ()
[INFO]     )
[INFO]   )
[INFO] )
[INFO]
[INFO] == Expanded type of tree ==
[INFO]
[INFO] <notype>
[INFO]
[INFO] unhandled exception while transforming X500PrincipalBuilderTest.scala
[ERROR] error:
[INFO]      while compiling: /Users/christian/code/truelicense~v2/truelicense-core/src/test/scala/net/java/truelicense/core/util/X500PrincipalBuilderTest.scala
[INFO]         during phase: lambdalift
[INFO]      library version: version 2.10.0
[INFO]     compiler version: version 2.10.0
[INFO]   reconstructed args: -deprecation -feature -classpath /Users/christian/code/truelicense~v2/truelicense-core/target/classes:/Users/christian/.m2/repository/net/java/truelicense/truelicense-obfuscate/2.3-SNAPSHOT/truelicense-obfuscate-2.3-SNAPSHOT.jar:/Users/christian/.m2/repository/commons-codec/commons-codec/1.8/commons-codec-1.8.jar:/Users/christian/.m2/repository/org/slf4j/slf4j-simple/1.7.5/slf4j-simple-1.7.5.jar:/Users/christian/.m2/repository/org/slf4j/slf4j-api/1.7.5/slf4j-api-1.7.5.jar:/Users/christian/.m2/repository/com/google/code/findbugs/annotations/2.0.1/annotations-2.0.1.jar:/Users/christian/.m2/repository/junit/junit/4.11/junit-4.11.jar:/Users/christian/.m2/repository/org/hamcrest/hamcrest-core/1.3/hamcrest-core-1.3.jar:/Users/christian/.m2/repository/org/mockito/mockito-core/1.9.5/mockito-core-1.9.5.jar:/Users/christian/.m2/repository/org/objenesis/objenesis/1.3/objenesis-1.3.jar:/Users/christian/.m2/repository/org/scalatest/scalatest_2.10.0/1.8/scalatest_2.10.0-1.8.jar:/Users/christian/.m2/repository/org/scala-lang/scala-library/2.10.0/scala-library-2.10.0.jar:/Users/christian/.m2/repository/org/scala-lang/scala-actors/2.10.0/scala-actors-2.10.0.jar:/Users/christian/.m2/repository/org/scala-lang/scala-reflect/2.10.0/scala-reflect-2.10.0.jar:/Users/christian/.m2/repository/org/scalacheck/scalacheck_2.10.0/1.10.0/scalacheck_2.10.0-1.10.0.jar:/Users/christian/.m2/repository/org/scala-tools/testing/test-interface/0.5/test-interface-0.5.jar:/Users/christian/code/truelicense~v2/truelicense-core/target/test-classes -d /Users/christian/code/truelicense~v2/truelicense-core/target/test-classes
[INFO]
[INFO]   last tree to typer: term $outer
[INFO]               symbol: value $outer (flags: <synthetic> <paramaccessor> <triedcooking> private[this])
[INFO]    symbol definition: private[this] val $outer: net.java.truelicense.core.util.X500PrincipalBuilderTest
[INFO]                  tpe: <notype>
[INFO]        symbol owners: value $outer -> anonymous class $anonfun$1 -> value <local X500PrincipalBuilderTest> -> class X500PrincipalBuilderTest -> package util
[INFO]       context owners: value key -> value $anonfun -> value $anonfun -> method apply -> anonymous class $anonfun$apply$mcV$sp$2 -> method apply$mcV$sp -> anonymous class $anonfun$apply$mcV$sp$1 -> method apply$mcV$sp -> anonymous class $anonfun$1 -> value <local X500PrincipalBuilderTest> -> class X500PrincipalBuilderTest -> package util
[INFO]
[INFO] == Enclosing template or block ==
[INFO]
[INFO] DefDef( // final def apply(x$1: Tuple2): Unit
[INFO]   <method> final <triedcooking>
[INFO]   "apply"
[INFO]   []
[INFO]   // 1 parameter list
[INFO]   ValDef( // x0$2: Tuple2
[INFO]     <param> <synthetic> <triedcooking>
[INFO]     "x0$2"
[INFO]     <tpt> // tree.tpe=Tuple2
[INFO]     <empty>
[INFO]   )
[INFO]   <tpt> // tree.tpe=Unit
[INFO]   Block( // tree.tpe=Unit
[INFO]     // 3 statements
[INFO]     ValDef( // case val x1: Tuple2
[INFO]       case <synthetic> <triedcooking>
[INFO]       "x1"
[INFO]       <tpt> // tree.tpe=Tuple2
[INFO]       "x0$2" // x0$2: Tuple2, tree.tpe=Tuple2
[INFO]     )
[INFO]     LabelDef( // case def case4(): Unit, tree.tpe=Unit
[INFO]       ()
[INFO]       If( // tree.tpe=Unit
[INFO]         Apply( // final def ne(x$1: Object): Boolean in class Object, tree.tpe=Boolean
[INFO]           "x1"."ne" // final def ne(x$1: Object): Boolean in class Object, tree.tpe=(x$1: Object)Boolean
[INFO]           null
[INFO]         )
[INFO]         Block( // tree.tpe=Unit
[INFO]           // 2 statements
[INFO]           ValDef( // val key: String
[INFO]             <triedcooking>
[INFO]             "key"
[INFO]             <tpt> // tree.tpe=String
[INFO]             Apply( // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=String
[INFO]               TypeApply( // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=()String
[INFO]                 x1._1()."$asInstanceOf" // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=[T0 >: ? <: ?]()T0
[INFO]                 <tpt> // tree.tpe=String
[INFO]               )
[INFO]               Nil
[INFO]             )
[INFO]           )
[INFO]           ValDef( // val value: String
[INFO]             <triedcooking>
[INFO]             "value"
[INFO]             <tpt> // tree.tpe=String
[INFO]             Apply( // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=String
[INFO]               TypeApply( // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=()String
[INFO]                 x1._2()."$asInstanceOf" // final def $asInstanceOf[T0 >: ? <: ?](): T0 in class Object, tree.tpe=[T0 >: ? <: ?]()T0
[INFO]                 <tpt> // tree.tpe=String
[INFO]               )
[INFO]               Nil
[INFO]             )
[INFO]           )
[INFO]           Apply( // case def matchEnd3(x: runtime.BoxedUnit): Unit, tree.tpe=Unit
[INFO]             "matchEnd3" // case def matchEnd3(x: runtime.BoxedUnit): Unit, tree.tpe=(x: runtime.BoxedUnit)Unit
[INFO]             Block( // tree.tpe=runtime.BoxedUnit
[INFO]               Apply( // def addAttribute(x$1: String,x$2: String): Unit in class X500PrincipalBuilder, tree.tpe=Unit
[INFO]                 "b"."addAttribute" // def addAttribute(x$1: String,x$2: String): Unit in class X500PrincipalBuilder, tree.tpe=(x$1: String, x$2: String)Unit
[INFO]                 // 2 arguments
[INFO]                 "key" // val key: String, tree.tpe=String
[INFO]                 "value" // val value: String, tree.tpe=String
[INFO]               )
[INFO]               "scala"."runtime"."BoxedUnit"."UNIT" // final val UNIT: runtime.BoxedUnit in object BoxedUnit, tree.tpe=runtime.BoxedUnit
[INFO]             )
[INFO]           )
[INFO]         )
[INFO]         Apply( // case def case5(): Unit, tree.tpe=Unit
[INFO]           "case5" // case def case5(): Unit, tree.tpe=()Unit
[INFO]           Nil
[INFO]         )
[INFO]       )
[INFO]     )
[INFO]     LabelDef( // case def case5(): Unit, tree.tpe=Unit
[INFO]       ()
[INFO]       Apply( // case def matchEnd3(x: runtime.BoxedUnit): Unit, tree.tpe=Unit
[INFO]         "matchEnd3" // case def matchEnd3(x: runtime.BoxedUnit): Unit, tree.tpe=(x: runtime.BoxedUnit)Unit
[INFO]         Throw( // tree.tpe=Nothing
[ERROR]           Apply( // def <init>(obj: Object): MatchError in class MatchError, tree.tpe=MatchError
[ERROR]             new MatchError."<init>" // def <init>(obj: Object): MatchError in class MatchError, tree.tpe=(obj: Object)MatchError
[INFO]             "x1" // case val x1: Tuple2, tree.tpe=Tuple2
[INFO]           )
[INFO]         )
[INFO]       )
[INFO]     )
[INFO]     LabelDef( // case def matchEnd3(x: runtime.BoxedUnit): Unit, tree.tpe=Unit
[INFO]       "x" // x: runtime.BoxedUnit, tree.tpe=runtime.BoxedUnit
[INFO]       ()
[INFO]     )
[INFO]   )
[INFO] )
[INFO]
[INFO] == Expanded type of tree ==
[INFO]
[INFO] <notype>
[INFO]
[INFO] uncaught exception during compilation: java.lang.IllegalArgumentException
[ERROR] error: java.lang.IllegalArgumentException: Could not find proxy for case val x1: Tuple2 in List(value x1, method apply, anonymous class $anonfun$apply$1, method apply, anonymous class $anonfun$apply$mcV$sp$3, method apply$mcV$sp, anonymous class $anonfun$apply$mcV$sp$2, method apply$mcV$sp, anonymous class $anonfun$apply$mcV$sp$1, method apply$mcV$sp, anonymous class $anonfun$1, value <local X500PrincipalBuilderTest>, class X500PrincipalBuilderTest, package util, package core, package truelicense, package java, package net, package <root>) (currentOwner= value key )
[INFO]  at scala.tools.nsc.transform.LambdaLift$LambdaLifter.scala$tools$nsc$transform$LambdaLift$LambdaLifter$$searchIn$1(LambdaLift.scala:303)
[INFO]  at scala.tools.nsc.transform.LambdaLift$LambdaLifter.scala$tools$nsc$transform$LambdaLift$LambdaLifter$$searchIn$1(LambdaLift.scala:308)
[INFO]  at scala.tools.nsc.transform.LambdaLift$LambdaLifter.scala$tools$nsc$transform$LambdaLift$LambdaLifter$$searchIn$1(LambdaLift.scala:308)
[INFO]  at scala.tools.nsc.transform.LambdaLift$LambdaLifter.scala$tools$nsc$transform$LambdaLift$LambdaLifter$$searchIn$1(LambdaLift.scala:308)
... (removed some hundred stack frames)

WTF это означает, и как я могу обойти это?

1 ответ

Решение

Этот сбой, по-видимому, вызван SI-6317, где используется анонимная частичная функция (с case (...) =>иногда убивает компилятор

В заявке есть исправление, но похоже, что оно не дошло до 2.10.1 (см. Adriaanm/scala@08dffa3 против scala/scala@2.10.1), но оно дает обходной путь:

дающий foreach функция, которая делает match сам вместо того, чтобы дать ему PartialFunction:

attributes foreach { _ match { case (key, value) => b addAttribute (key, value)}}
Другие вопросы по тегам