如何在单元测试中模拟缺乏网络连接
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。
我想我可以实现 LSResourceResolver
的 resolveResource
方法,以便永远不会 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 的作者
一般问题是如何模拟(作为 JUnit
测试用例套件的一部分)缺乏网络连接,因为这是某些测试用例中的重要考虑因素。有没有办法通过 Java API(或通过 JVM 选项)这样做,以便某些测试用例可以 运行 在网络断开连接的情况下? (模拟还是真实?)。
更具体的情况(如果没有普遍适用的解决方案)是我正在做一堆XML-文件处理(包括XSD-验证),我需要确保没有通过网络获取任何内容,特别是未使用 xsi:schemaLocation
属性值(提示),并且所有 XSD 实际上都是从类路径中获取的。我正在使用 Validator with a custom LSResourceResolver 来加载类路径中可能需要的任何 XSD。
我想我可以实现 LSResourceResolver
的 resolveResource
方法,以便永远不会 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 的作者