noober
noober

Reputation: 164

Mocking Scala Objects and Functions

I have a Scala object and it contains a few utility functions. These functions are called by other functions present in the same object or from other classes/objects. Is it possible to mock this object or the functions so that I can unit test the classes where they are being called.

Example:

object Util {

  def methodA() = { 
    //other code
    methodB() 
    //other code
  }

  def methodB() = { 
    //other code
    methodC() 
    //other code
  }

  def methodC() = { ... }

}

And here I'm calling the object function from another class

class Data {

  //other code

  def call() = {
    //other code
    Util.methodA()
    //other code
  }
}

How can I go about unit testing the function call() of class Data ? In Java I can create a mocked object for Util and set expectations for a call to methodA(), but this is not possible in Scala since no mocking library supports mocking of a Scala object.

Upvotes: 7

Views: 12990

Answers (2)

UnitasBrooks
UnitasBrooks

Reputation: 1092

This might not be a perfect fit for all scenarios but if you have a function you are passing into other classes say and you want to mock it in your unit tests you can create a class that extends a function definition and mock that:

  class Increment extends (Int => Int) {
    def apply(i: Int): Int = {
      i + 1
    }
  }
  case class IncrementFactory(incrementer: Int => Int)

  val incFactory = IncrementFactory(new Increment)
  incFactory.incrementer(3)

  def testIncrementFactory: Unit = {
    val incrementer = mock[Increment]
    when(incrementer.apply(any())).thenReturn(4)
    
    val factory = IncrementFactory(incrementer)
    assert(factory.incrementer(3) == 4) 
  }

Upvotes: 0

Keith Nordstrom
Keith Nordstrom

Reputation: 354

The problem in some sense is that you've used a pattern that is similar to a static class in Java. You can't mock static classes, which is why Java patterns often don't favor that approach.

On the other hand, in Scala you can. Make Util extend a trait and implement it.

trait UtilityMethods {
 def methodA()
 def methodB()
}

object Utils extends UtilityMethods {
  def methodA() = {}
  def methodB() = {}
}

And then in your test:

val mockedUtils = mock[UtilityMethods]

(using mockito). Obviously you will have to pass UtilityMethods around your code instead of Utils, just like you would wth an interface in Java.

In general, you can also use implicit scoping to inject functions and thus mock those dependencies without a static-like object approach, but the above is pretty close to Java patterns and should feel pretty comfortable.

Upvotes: 13

Related Questions