2014-12-04 14 views
16

Sto usando mockito con scalatest. Ho il seguente problema quando utilizzo il matcher con la classe value.Mockito matchers, scala value class e NullPointerException

import org.scalatest.FlatSpec 
import org.scalatest.mock.MockitoSugar 
import org.mockito.BDDMockito._ 
import org.mockito.Matchers.any 

case class FirstId(val value: String) extends AnyVal 
case class SecondId(val value: String) extends AnyVal 

trait MockedClass { 
    def someMethods(firstId: FirstId, secondId: SecondId): Int 
} 

class ValueClassSpec() extends FlatSpec with MockitoSugar { 

    val mockedClass = mock[MockedClass] 
    val secondId = SecondId("secondId") 

    "Matchers" should "work for value class" in { 
    // given 
    given(mockedClass.someMethods(any[FirstId], org.mockito.Matchers.eq(secondId))).willReturn(3) 
    // when 
    val result = mockedClass.someMethods(FirstId("firstId"), secondId) 
    // then 
    assert(result == 3) 
    } 

} 

e il risultato è:

ValueClassSpec: 
Matchers 
- should work for value class *** FAILED *** 
    java.lang.NullPointerException: 
    at io.scalac.fow.party.ValueClassSpec$$anonfun$1.apply$mcV$sp(ValueClassSpec.scala:22) 
    at io.scalac.fow.party.ValueClassSpec$$anonfun$1.apply(ValueClassSpec.scala:20) 
    at io.scalac.fow.party.ValueClassSpec$$anonfun$1.apply(ValueClassSpec.scala:20) 
    at org.scalatest.Transformer$$anonfun$apply$1.apply(Transformer.scala:22) 
    at org.scalatest.Transformer$$anonfun$apply$1.apply(Transformer.scala:22) 
    at org.scalatest.OutcomeOf$class.outcomeOf(OutcomeOf.scala:85) 
    at org.scalatest.OutcomeOf$.outcomeOf(OutcomeOf.scala:104) 
    at org.scalatest.Transformer.apply(Transformer.scala:22) 
    at org.scalatest.Transformer.apply(Transformer.scala:20) 
    at org.scalatest.FlatSpecLike$$anon$1.apply(FlatSpecLike.scala:1639) 
    ... 

ho trovato simile domanda (Scala Value classes and Mockito Matchers don't play together), ma senza alcun preavviso.

Esiste la possibilità di utilizzare i mockito matcher con classe di scala di valore?

versioni lib: Scala 2.11.2, 1.10.8, Mockito ScalaTest 2.1.6

+0

afaik 'any' e' eq' non possono essere usati insieme. – Ashalynd

+0

@Ashalynd - da [documentazione di Mockito] (http://mockito.github.io/mockito/docs/current/org/mockito/Mockito.html): ** Avviso sugli argomenti corrispondenti: ** Se si utilizza gli argomenti corrispondenti, tutti gli argomenti devono essere forniti dai corrispondenti. E.g: (l'esempio mostra la verifica ma lo stesso vale per lo stub): verify (mock) .someMethod (anyInt(), anyString(), eq ("terzo argomento")); // above is correct - eq() è anche un argomento matcher –

risposta

19

La soluzione corretta è:

case class StringValue(val text: String) extends AnyVal 
case class LongValue(val value: Long) extends AnyVal 

val eqFirst: StringValue = StringValue(org.mockito.Matchers.eq("first")) 
val anySecond: StringValue = StringValue(org.mockito.Matchers.any[String]) 

val eqFirst: LongValue = LongValue(org.mockito.Matchers.eq(1L)) 
val anySecond: LongValue = LongValue(org.mockito.Matchers.any[Long]) 
+0

Quando faccio questo, ho un sacco di avvertimenti e gli errori di non usare matchers al di fuori di sradicamento: val anyProjectName = org.mockito.ArgumentMatchers.any [stringa] .asInstanceOf [ProjectName] org.mockito.exceptions. abuso.InvalidUseOfMatchersException: Corrispondenza argomento errato o maltrattato rilevato qui: –

9

ho trovato la soluzione:

val anyFirstId: FirstId = any[String].asInstanceOf[FirstId] 
val eqSecondId: SecondId = org.mockito.Matchers.eq[String](secondId.value).asInstanceOf[SecondId] 
given(mockedClass.someMethods(anyFirstId, eqSecondId)).willReturn(3) 
+0

Sarebbe bello se implementaste il vostro matcher per AnyVals e lo metteste su GitHub. –

+0

Non è facile scrivere eq matcher generico per AnyVal. –

+0

Ha funzionato per te? Ottengo un ClassCastException. –

1

questo funziona per tutti i tipi di classi di valore extends AnyVal e non richiede abbinamenti speciali:

given(mockedClass.someMethods(FirstId(anyString), SecondId(org.mockito.Matchers.eq(secondId.value)))).willReturn(3) 
0

Se hai informe nelle vostre dipendenze si potrebbe prendere in considerazione il mio piccolo metodo di supporto: https://gist.github.com/Fristi/bbc9d0e04557278f8d19976188a0b733

Invece di scrivere

UserId(is(context.userId.value))

è possibile scrivere

isAnyVal(context.userId)

Quale è un po 'più conveniente :-)

Problemi correlati