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

Posted

技术标签:

【中文标题】如何在单元测试中模拟缺乏网络连接【英文标题】:how to simulate lack of network connectivity in unit testing 【发布时间】:2015-04-08 10:32:24 【问题描述】:

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

更具体的情况(如果没有普遍适用的解决方案)是我正在进行大量 XML 文件处理(包括 XSD 验证),并且我需要确保不会通过网络获取任何内容,具体而言,即xsi:schemaLocation 属性值(提示)未使用,所有 XSD 实际上都是从类路径中获取的。我正在使用带有自定义 LSResourceResolver 的 Validator,它可以从类路径加载任何可能需要的 XSD。

我想我可以实现LSResourceResolverresolveResource 方法,从而永远不会返回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 选项,那将是一个集成测试。对于正确的单元测试,您的网络连接将是一个可以模拟的独立依赖项。听起来您对 java 网络 api 有很强的依赖关系。 取决于您的代码发生了什么,但如果这不合适,我会模拟 socket() 调用(请参阅***.com/questions/3202675/… 进行讨论)或在单独的 VM 中运行测试 -例如,如果您使用的是一个完全独立的微服务,它不会通过在被测代码中模拟来改变 【参考方案1】:

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

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

【讨论】:

【参考方案2】:

Sniffy 允许您在 JUnit 测试中使用 block outgoing network connections - 每当您尝试建立新连接时,它都会抛出 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 的作者

【讨论】:

以上是关于如何在单元测试中模拟缺乏网络连接的主要内容,如果未能解决你的问题,请参考以下文章

如何在用于运行 Java 单元测试的内存数据库中模拟 Oracle (+) 外连接表示法?

不断开连接HttpUrlConnection意味着资源缺乏或网络滞后

在 javascript 执行期间测试 Internet 连接丢失

在javascript执行期间测试互联网连接的丢失

iOS/Xcode:如何模拟两个 iOS 模拟器之间的网络连接

我如何模拟慢速网络连接[重复]