Scalaシングルトンオブジェクトをモックしようとしています。特に、サービスコンポーネント(テスト中のクラス)内で使用されるオブジェクトplay.api.libs.ws.WS
をモックする必要があります。Mockitoを使用すると、これは不可能です。 、テストの実行は次のように失敗します。
[error] MockitoException: :
[error] Cannot mock/spy class play.api.libs.ws.WS$
[error] Mockito cannot mock/spy following:
[error] - final classes
[error] - anonymous classes
[error] - primitive types (GeolocationSpec.scala:18)
ここ を読んで、Scalamockがそれを行うことを許可しているようです:
スタンドアロンのシングルトンオブジェクトをモックするには、
org.scalamock.annotation.mockObject
を使用します。
私のサービスコンポーネントは次のようなものです。
trait GeolocationService {
def wsClient = WS
def getPath(Origin: Location, destination: Location): Future[Route]
}
class DefaultGeolocationService extends GeolocationService {
val serviceProviderEndpoint = Play.current.configuration.getString("api.directions.endpoint")
override def getPath(Origin: Location, destination: Location): Future[Route] = {
val params = Seq(
"Origin" -> s"${Origin.lat},${Origin.lon}",
"destination" -> s"${destination.lat},${destination.lon}"
);
val resp = wsClient.url(serviceProviderEndpoint.get).withQueryString(params: _*).get()
resp.map {
// omitted code
}
}
}
私のbuild.sbtには、次のすべての依存関係があります。
[...]
"org.scalatest" %% "scalatest" % "2.2.1",
"org.specs2" %% "specs2" % "2.3.13" % "test",
"org.scalamock" %% "scalamock-specs2-support" % "3.0.1" % "test",
"org.scalamock" %% "scalamock-scalatest-support" % "3.0.1" % "test",
"org.scalamock" %% "scalamock" % "3.0.1",
[...]
しかし、私はこれを見つけることができません:org.scalamock.annotation.mockObject
おそらくこれはEasyMockとPowerMockを使用しても実行できますが、Scalaサンプルコードが見つかりません。
何か案が?
ScalaMock 3を使用してシングルトンオブジェクトをモックすることは不可能ですが、PaulButcherはこの機能をScalaMock4に再導入する予定です( http:// paulbutcherを参照) .com/2014/04/15/scalamock-status-report / )
シングルトンをあざけるな。 WSの代わりに、サービスコンポーネントをそれを隠す薄いファサードに依存させます。
trait GeolocationService {
def ws: (String, Seq[String]) => Promise[Response] = { (url, params) =>
wsClient.url(serviceProviderEndpoint.get).withQueryString(params: _*).get()
}
def getPath(Origin: Location, destination: Location): Future[Route]
}
テストでは、wsメソッドをモックでオーバーライドするだけです。これで簡単に作成できます。
val mockedWs = mock[(String, Seq[String]) => Promise[Response]]
// TODO specify mock's behavior here
val service = new DefaultGeolocationService() {
override def ws = mockedWs
}
なぜケーキパターンを使わないのですか?
少し冗長ですが、モックの問題は解決します。
trait GeolocationServiceComponent {
val geolocationService:GeolocationService
trait GeolocationService {
def wsClient
def getPath(Origin:Location, destination: Location): Future[Route]
}
}
trait GeolocationServiceComponentImpl {
val geolocationService = new GeolocationService {
override def wsClient = WS
}
}
class DefaultGeolocationService extends GeolocationServiceComponent ...
//Defined into your test class
trait MockGeolocationServiceComponent {
val geolocationService = Mock[GeolocationService]
//Here you define your mock logic
}
モナドを使用してこれを行うこともできますが、実装したことはありません。ここで説明します: Scala依存性注入:暗黙的なパラメーターの代替
このような制限に対応するためにデザインを変更するのは好きではありません。基本的に私が得ることができるのは、デザインを変更してからモックフレームワークを使用することです
//since mocking frameworks on SCALA is not support mocking singleton
//hacks are either required to change the design or ...
//I'd rather putting the mocking logic here by myself
var testUser:Option[User] = None;
def getCurrentUser(request: Request[Object]) = {
if( testUser != None ){
testUser;
}
それが役に立てば幸い。