package scala.tools
package reflect
import java.lang.reflect.{ Method, Proxy, InvocationHandler }
trait Mock extends (Invoked => AnyRef) {
mock =>
def interfaces: List[Class[_]]
def classLoader: ClassLoader
def apply(invoked: Invoked): AnyRef
def newProxyInstance(handler: InvocationHandler): AnyRef =
Proxy.newProxyInstance(classLoader, interfaces.toArray, handler)
def newProxyInstance(): AnyRef =
newProxyInstance(newInvocationHandler())
def newInvocationHandler() = new InvocationHandler {
def invoke(proxy: AnyRef, method: Method, args: Array[AnyRef]) =
mock(Invoked(proxy, method, args))
}
}
object Mock {
def fromInterfaces(clazz: Class[_], clazzes: Class[_]*)(pf: PartialFunction[Invoked, AnyRef]): AnyRef = {
val ints = clazz :: clazzes.toList
require(ints forall (_.isInterface), "All class objects must represent interfaces")
val mock = new Mock {
val interfaces = ints
def classLoader = clazz.getClassLoader
def apply(invoked: Invoked) =
if (pf.isDefinedAt(invoked)) pf(invoked)
else if (invoked.isObjectMethod) invoked invokeOn this
else throw new NoSuchMethodException("" + invoked)
}
mock.newProxyInstance()
}
def fromClass(clazz: Class[_])(pf: PartialFunction[Invoked, AnyRef]): AnyRef = allInterfaces(clazz) match {
case Nil => sys.error(clazz + " implements no interfaces.")
case x :: xs => fromInterfaces(x, xs: _*)(pf)
}
}