如何在单元测试中模拟缺乏网络连接

how to simulate lack of network connectivity in unit testing

一般问题是如何模拟(作为 JUnit 测试用例套件的一部分)缺乏网络连接,因为这是某些测试用例中的重要考虑因素。有没有办法通过 Java API(或通过 JVM 选项)这样做,以便某些测试用例可以 运行 在网络断开连接的情况下? (模拟还是真实?)。

更具体的情况(如果没有普遍适用的解决方案)是我正在做一堆XML-文件处理(包括XSD-验证),我需要确保没有通过网络获取任何内容,特别是未使用 xsi:schemaLocation 属性值(提示),并且所有 XSD 实际上都是从类路径中获取的。我正在使用 Validator with a custom LSResourceResolver 来加载类路径中可能需要的任何 XSD。

我想我可以实现 LSResourceResolverresolveResource 方法,以便永远不会 return null(因此,永远不会退回到打开与资源的常规 URI 连接的默认行为),但我不确定这是否足够,无论如何,如果我可以 运行 我的 JUnit 在模拟岛中进行测试,我会更有信心-模式(无需手动关闭我机器上的接口)。

更新

接受的答案,即 -DsocksProxyHost 方法提供了我需要的解决方案,因为 JUnit task 可以接受 VM 参数(如果 fork 设置为 true)所以我可以在我的 Ant 文件中包含以下内容:

<junit printsummary="true" showoutput="true" fork="true" maxmemory="256m">
    <jvmarg value="-DsocksProxyHost=127.0.0.1"/>
    ...

... 包含在 contrib:if 中,因此我可以从命令行控制是否 运行 JUnit 在网络连接条件下进行测试。

我不建议这样做,但是... 您可以尝试使用安全管理器。默认情况下 JVM 不使用安全管理器,通过设置 -Djava.security.manager 您将激活 默认安全管理器。 AFAIR,安全管理器的默认行为是阻止任何连接(只要在安全策略中未明确授予权限)。但。每次连接被阻止时,您都会收到 java.net.NetPermission 异常。您还可能遇到本地文件访问、反射调用等问题。

另一种可能性是将网络代理设置为不存在的地址,例如-DsocksProxyHost=127.0.0.1。然后任何 TCP 套接字都会尝试使用 SOCKS 服务器并失败。

Sniffy allows you to block outgoing network connections 在您的 JUnit 测试中 - 每当您尝试建立新连接时它都会抛出 ConnectException

@Rule public SniffyRule sniffyRule = new SniffyRule();

@Test
@DisableSockets
public void testDisableSockets() throws IOException {
    try {
        new Socket("google.com", 22);
        fail("Sniffy should have thrown ConnectException");
    } catch (ConnectException e) {
        assertNotNull(e);
    }
}

它还允许您测试您的应用程序在 runtime 中连接中断时的行为。只需将 -javaagent:sniffy.jar=5559 添加到您的 JVM 参数并将您的浏览器指向 localhost:5559 - 它将打开一个网页,其中包含所有已发现的到下游系统的连接和控件以禁用某些连接。

免责声明:我是 Sniffy 的作者