Ir al contenido principal

Unit-Testing SOAP: ¿Qué hacemos con los Holder?

En los últimos meses he tenido que integrarme con algunos Servicios Web (SOAP). Suerte que frameworks como CXF consiguen que la tarea sea algo más sencilla de lo que era hace unos años...

El problema vino cuando intenté probar uno de mis métodos: las clases que necesitaba llamar (auto-generadas con CXF) tenían como parámetros instancias de Holder (parámetro de entrada/salida). Y yo necesitaba utilizar los resultados de los Holder para continuar...

Veamos un ejemplo. Mi código tiene que llamar a esta interfaz:

package external.soap;

import javax.xml.ws.Holder;

// Omito todas las anotaciones y demás meta-información...
public interface IWebService {
    public String execute(int param, Holder<String> innerResult);
}

E imaginemos que tiene que hacer algo tan sencillo como concatenar la salida del método, con el resultado contenido en innerResult. El test que lo comprobara podría ser tan sencillo como:

[...]
import org.jmock.*;
import org.junit.*;
import static org.junit.Assert.*;

import javax.xml.ws.Holder;
import external.soap.IWebService;
    
public class MyServiceTest {
    
    private MyService myService = null;

    private Mockery mockery = new Mockery();
    private IWebService webServiceMock = null;

    @Before
    public void setUp() {
        myService = new MyService();

        webServiceMock = mockery.mock(IWebService.class);
        myService.setWebService(webServiceMock);
    }

    @After
    public void tearDown() {
        mockery.assertIsSatisfied();
    }

    @Test
    public void testBothResultsAreConcatenated() throws Exception {
        mockery.checking(new Expectations() {
          {
            exactly(1).of(webServiceMock).execute(with(any(Integer.class)),
                    with(any(Holder.class)));
            will(returnValue("outer"));
          }
        });

        String result = myService.executeOperation(1);
        assertEquals("outer-inner", result);
    }
}

Por muchos "inventos" que hagamos en el método executeOperation(), no habrá manera de hacer funcionar ningún test que tenga Holders, ya que del mismo modo que especificamos que el método IWebService.execute() tiene que retornar la respuesta "outer", no estamos declarando en ningún sitio que además debería rellenar el Holder con la respuesta "inner".

¿Se puede hacer algo al respecto? Aunque el capítulo 15 del JMock Cookbook explique la manera de realizar acciones concretas dentro de los objetos mock utilizando BeanShell, creo que es una solución demasiado compleja. Sin embargo, sí que es posible encontrar una solución sencilla y fácil de implementar para estas situaciones: extender las Action. Utilizando la propia jerarquía de clases de JMock, se puede redefinir el comportamiento de un objeto, implementando la clase Action.

A la hora de la verdad, yo prefiero utilizar la subclase VoidAction, creando una clase anónima y redefiniendo su método invoke:

[...]
import org.jmock.*;
import org.junit.*;
import static org.junit.Assert.*;

import javax.xml.ws.Holder;
import external.soap.IWebService;
    
public class MyServiceTest {
    
    private MyService myService = null;

    private Mockery mockery = new Mockery();
    private IWebService webServiceMock = null;

    @Before
    public void setUp() {
        myService = new MyService();

        webServiceMock = mockery.mock(IWebService.class);
        myService.setWebService(webServiceMock);
    }

    @After
    public void tearDown() {
        mockery.assertIsSatisfied();
    }

    @Test
    public void testBothResultsAreConcatenated() throws Exception {
        mockery.checking(new Expectations() {
          {
            exactly(1).of(webServiceMock).execute(with(any(Integer.class)),
                    with(any(Holder.class)));
            will(new VoidAction() {
                @Override
                public Object invoke(Invocation invocation) throws Throwable {
                    Holder<String> holder = (Holder) invocation.getParameter(1);
                    holder.value = "inner";
                    return "outer";
                }
            });
          }
        });

        String result = myService.executeOperation(1);
        assertEquals("outer-inner", result);
    }
}

Como se puede ver, aunque la sintaxis es algo extraña, se consigue definir el comportamiento exacto de cada uno de los parámetros de entrada/salida. Sólo hay que tener en cuenta que el primer parámetro es invocation.getParameter(0), el segundo es invocation.getParameter(1), y así sucesivamente.

Por último, si vemos que tenemos que probar muchas veces el mismo comportamiento, y se nos repite en varios tests, siempre podemos optar por crear una inner class dentro de nuestra clase de pruebas, creando una nueva instancia desde cada uno de los tests:

    [...] 

    @Test
    public void testBothResultsAreConcatenated() throws Exception {
        mockery.checking(new Expectations() {
          {
            exactly(1).of(webServiceMock).execute(with(any(Integer.class)),
                    with(any(Holder.class)));
            will(new OuterInnerAction("outer", "inner"));
          }
        });

        String result = myService.executeOperation(1);
        assertEquals("outer-inner", result);
    }

    // ... other tests.

    private class OuterInnerAction extends VoidAction {
        private String returnValue = null;
        private String holderValue = null;

        public OuterInnerAction(String returnValue, String holderValue) {
            this.returnValue = returnValue;
            this.holderValue = holderValue;
        }

        @Override
        public Object invoke(Invocation invocation) throws Throwable {
            HolderHolder<String> holder = (Holder) invocation.getParameter(1);
            holder.value = holderValue;
            return returnValue;
        }
    }

    [...]

Aunque la solución no sea tan elegante como utilizar BeanShell, creo que es una manera rápida y sencilla de dotar a cada test de un comportamiento "a medida".

¡Suerte!

Comentarios

Entradas populares de este blog

Unit Tests (I): "mocking" la interacción

Indispensables. De entre todos los tipos de tests, son los más sencillos de escribir, aunque son los que permiten encontrar (a tiempo) la mayor cantidad de bugs. ¿Qué es un Unit Test? Mejor contar qué hacen y no lo que son: Un Unit Test debe probar un ÚNICO método de una ÚNICA clase . Simple. Sin excepciones (*) . El método a probar SIEMPRE debe tener un estado inicial conocido y un resultado esperado. ¡Siempre! Si cumple ambas premisas, el test podrá ser ejecutado de forma automática y sin depender de nadie (**) . Considero que probar la interacción de una clase con el resto de componentes es uno de los sistemas que mejores resultados ofrece. Estas interacciones se "simulan" en el mismo test, utilizando mock-objects. Antes de empezar con algún ejemplo de Unit Test con Mock Objects, hay algunos consejos que nos pueden facilitar el trabajo: Cada clase debe tener su Interfaz (***) , exceptuando las java.lang.Exception y poco más. Conocer el patrón de Inyección de Depen...

¿Introducción?

Hace tiempo que las siglas TDD están de moda. Test Driven Development por todos lados. Pero cuando uno se decide a ponerlo en práctica, lo único que encuentra por Mr. Google son ejemplos sencillos (HelloWorld, PetClinic, el Hospital y sus pacientes, o la última gran incorporación: Twitter en 15 minutos). Por desgracia, cuando un developer decide probar su código, nunca (enfatizo: ¡nunca! ) sirven de nada estos casos tan sencillos. A la hora de la verdad, tenemos varias bases de datos, repositorios de contenidos, conexiones a servicios SOAP (escalofrío), a servicios REST ... Casos difíciles de probar, que agradecen un conjunto de buenas prácticas o ejemplos parecidos. Es muy difícil encontrar información clara sobre por dónde empezar. Sobre dónde buscar. Sobre qué debería hacer un test (o lo que nunca debería hacer). Uno acaba teniendo la impresión que probar código de manera correcta y eficaz es un arte que se aprende con el tiempo, a base de pruebas y (muchos) errores. Cansad...

Matchers y jMock

A medida que la complejidad de nuestro código aumenta, también tiende a aumentar la complejidad de los tests. Al principio nos va bien con asegurarnos que la salida del método es la esperada. Pero llegado el momento en que necesitamos comprobar qué valores contiene cada campo, o qué información se utiliza para llamar a las otras clases, es hora de empezar a utilizar Matchers . Los Matchers nos permiten especificar las restricciones o condiciones que deben cumplir los parámetros de entrada. En anteriores posts, cada vez que escribíamos with(any(String.class)) , estábamos utilizando un Matcher (uno muy genérico, que permite cualquier cadena como parámetro de entrada). Imaginemos que queremos asegurarnos que nuestro método getWeatherForecast() invoca al método getWeather() con una instancia de país, cuyo nombre debe ser el mismo que el recibido. Para ello vamos a utilizar el método hasProperty() de la clase org.hamcrest.Matchers : [...] import static org.hamcrest.Matchers.*; im...