OkHttp MockWebServer在重新启动时无法接受连接

Nit*_*mer 6 java android unit-testing okhttp mockwebserver

我正在使用OkHttp MockWebServer来模拟我的服务器响应以进行单元测试.

它适用于第一次测试,但在第二次测试时,我的客户端失败了:

无法连接到localhost/0:0:0:0:0:0:0:1:63631

即使第二次测试与第一次测试完全相同,也会发生这种情况.
这是我正在做的事情:

@RunWith(RobolectricTestRunner.class)
@Config(shadows = MyClassTest.MyNetworkSecurityPolicy.class,
        manifest = "src/main/AndroidManifest.xml",
        constants = BuildConfig.class,
        sdk = 16)
public class MyClassTest {
    private MockWebServer mockServer;
    private MyServerApi serverApi;

    @Before
    public void setUp() throws Exception {
        System.out.println("\ntest start");
        this.mockServer = new MockWebServer();
        this.mockServer.start();

        this.serverApi = new MyServerApi(this.mockServer.url("/").toString());
    }

    @Test
    public void testOne() throws Exception {
        final String responseBody = // read response from file
        this.mockServer.enqueue(new MockResponse().setResponseCode(200).setBody(responseBody));

        final Waiter waiter = new Waiter();
        this.serverApi.getData("some_id", new Callback<MyResponseData> {
            @Override
            public void onResponse(final Call<MyResponseData> call, final Response<MyResponseData> response) {
                waiter.assertEquals("some_value", response.body().getValue());
                waiter.resume();
            }

            @Override
            public void onFailure(final Call<T> call, final Throwable error) {
                waiter.fail(error);
            }
        });

        waiter.await();

        final RecordedRequest recordedRequest = this.mockServer.takeRequest();
        assertEquals("GET", recordedRequest.getMethod());
    }

    @Test
    public void testTwo() throws Exception {
        final String responseBody = // read response from file
        this.mockServer.enqueue(new MockResponse().setResponseCode(200).setBody(responseBody));

        final Waiter waiter = new Waiter();
        this.serverApi.getData("some_id", new Callback<MyResponseData> {
            @Override
            public void onResponse(final Call<MyResponseData> call, final Response<MyResponseData> response) {
                waiter.assertEquals("some_value", response.body().getValue());
                waiter.resume();
            }

            @Override
            public void onFailure(final Call<T> call, final Throwable error) {
                waiter.fail(error);
            }
        });

        waiter.await();

        final RecordedRequest recordedRequest = this.mockServer.takeRequest();
        assertEquals("GET", recordedRequest.getMethod());
    }

    @After
    public void tearDown() throws Exception {
        System.out.println("test end\n");
        this.mockServer.shutdown();
    }

    @Implements(NetworkSecurityPolicy.class)
    public static class MyNetworkSecurityPolicy {
        @Implementation
        public static NetworkSecurityPolicy getInstance() {
            try {
                Class<?> shadow = MyNetworkSecurityPolicy.class.forName("android.security.NetworkSecurityPolicy");
                return (NetworkSecurityPolicy) shadow.newInstance();
            } catch (Exception e) {
                throw new AssertionError();
            }
        }

        @Implementation
        public boolean isCleartextTrafficPermitted() {
            return true;
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

第一个测试按原样通过,但第二个测试因上面写的消息而失败.
控制台中的输出是:

test start
okhttp3.mockwebserver.MockWebServer$3 execute
INFO: MockWebServer[63631] starting to accept connections
WARNING: no system properties value for gsm.sim.operator.alpha
okhttp3.mockwebserver.MockWebServer$4 processOneRequest
INFO: MockWebServer[63631] received request: GET REQUEST_PATH HTTP/1.1 and responded: HTTP/1.1 200 OK
okhttp3.mockwebserver.MockWebServer$3 acceptConnections
test end

INFO: MockWebServer[63631] done accepting connections: Socket closed

test start
okhttp3.mockwebserver.MockWebServer$3 execute
INFO: MockWebServer[63649] starting to accept connections
okhttp3.mockwebserver.MockWebServer$3 acceptConnections
INFO: MockWebServer[63649] done accepting connections: Socket closed
on error: Failed to connect to localhost/0:0:0:0:0:0:0:1:63631
test end
Run Code Online (Sandbox Code Playgroud)

(Waiter对象是来自concurrentunit lib)

知道为什么会这样吗?

Jes*_*son 6

您的第二个请求是使用第一个MockWebServer实例的URL.(每个实例都有一个不同的URL.)

  • 有趣,但我不明白这是怎么回事。据我了解,`setUp` 方法发生在每个测试方法之前,这意味着创建了一个模拟服务器的新实例,并创建了我的服务器 api 的一个新实例以及新创建的模拟服务器的 url。还是我在这里遗漏了什么? (2认同)
  • 好吧,事实证明你是对的。在我发布的示例中不应该是这种情况,因为它是一个简单的示例,但是在我的实际代码中,我将第一个模拟服务器的地址缓存在我的服务器 api 实例中。谢谢! (2认同)