2017-04-07 13 views
0

質問:CXF JAX-RSサービスエンドポイントの単体テスト中にHockpServletResponseを実際に注入するにはどうすればよいですか?ユニットテスト中に模擬HttpServletResponse

package services.example; 

import org.springframework.http.MediaType; 
import javax.servlet.http.HttpServletResponse; 
import javax.ws.rs.GET; 
import javax.ws.rs.POST; 
import javax.ws.rs.PUT; 
import javax.ws.rs.Path; 
import javax.ws.rs.PathParam; 
import javax.ws.rs.Produces; 
import javax.ws.rs.core.Context; 

@RestController 
@Path("/base") 
@Produces(MediaType.APPLICATION_JSON_VALUE); 
public class ExampleService { 

    @Autowired 
    SomeInjectedService injectedService; 

    @POST 
    @Path("/endpoint") 
    public Object testEndpoint(Object exampleInput, @Context HttpServletResponse response) { 
    if (exampleInput.someMethod != null) { 
     // Exception thrown here during tests, response is null!!! 
     response.setStatus(301); 
     return exampleInput; 
    } 

    return new injectedService.returnOutputFromInput(exampleInput); 
    } 
} 

Testクラス::私たちのサービスが設定されているかの

例は、(私は非テスト構造を変更するいかなる制御/権限を持っていません注意してください)

package test.services.example 

// A gazillion imports. Mockito, JUnit, Hamcrest, Javax.ws.rs, org.apache.cxf, Jackson providers. 

public class testExampleService { 

    @Mock 
    private HttpServletResponse mockResponse; // Should I do this? 

    @Mock 
    SomeInjectedService mockInjectionService; 

    @InjectMocks 
    ExampleService service = new ExampleService(); 

    @Before 
    public void setUpStuff() { 
    MockitoAnnotations.initMocks(this); 

    JAXRSServerFactoryBean sf = new JAXRSServerFactoryBean(); 
    sf.setResourceClasses(ExampleService.class); 

    List<Object> providers = new ArrayList<Object>; 
    JacksonJsonProvider provider = new JacksonJsonProvider(); 
    providers.add(provider); 

    sf.setProviders(providers); 

    sf.setResourceProvider(ExampleService.class, new SingletonResourceProvider(service, true)); 
    sf.setAddress("local://someuri"); 

    // This is where I'm trying to set up my injected Response context. Help? 
    sf.setInvoker(new Invoker() { 
     Invoker jarsInvoker = new JAXRSInvoker(); 

     @Override 
     public Object invoke(Exchange exchange, Object o) { 
     // What goes in here? I've seen an example for setting a mock 
     // Request context, but no for a Response. Here's my current 
     // (broken) attempt to do this... 
     exchange.getInMessage().put(AbstractHTTPDestination.HTTP_RESPONSE, mockResponse); 
     return jarsInvoker.invoke(exchange, o); 
     } 
    }); 
    } 

    @Test 
    public void testEndpointShouldReturn301() { 
    List<Object> providers = new ArrayList<Object>(); 
    JacksonJaxbJsonProvider provider = new JacksonJaxbJsonProvider(); 
    providers.add(provider); 

    WebClient client = WebClient.create("local://someuri", providers); 
    WebClient.getConfig(client) 
      .getRequestContext() 
      .put(LocalConduit.DIRECT_DISPATCH, Boolean.TRUE); 

    // Exception thrown here, Response context doesn't get injected! 
    client.path("/base/endpoint"); 
    SomeTestObject obj = new SomeTestObject(); 
    Response response = client.type(MediaType.APPLICATION_JSON).post(obj); 

    // Never gets run because an exception was thrown :(
    assertThat(response.getStatus(), is(301)); 
    } 

} 

それは私がいる可能性がありますここではいくつかのことを間違えていますが、エンドポイントにHttpServletResponseを適切に注入するために、私のカスタムInvoker内で正しい "魔法"が必要なだけであることを期待しています。注入されたContextを持たない私のエンドポイントでは、テストが正しく動作することに注意します(カスタム呼び出し者ビットを削除したとき)。

答えて

0

あなたはこれを維持、コンストラクタで例のサービスを初期化する必要はありません:あなたは@Path(「/エンドポイント」)をテストする場合、それはあなたが何をしますかそうです

@InjectMocks 
ExampleService service; 

が、奇妙な。 あなたのコードをテストしたい場合は、単に次のように実行します。

@Test 
    public void testEndpointShouldReturn301() { 
    service.testEndpoint(someObject, mockResponse); 
    assertThat(response.getStatus(), is(301)); 
    } 
+0

答えるためのおかげで、しかし、テストの目的は、特定のエンドポイントへのREST呼び出しが予想されるコールバック関数を打つことを確認することです。テスト中に、私の例では、 "/ endpoint"へのPOSTは常にtestEndpoint()メソッドをトリガーすることを確認することが重要です。 – Wrinn

関連する問題