我试图从Powermock运行this baseline unit test。使用Mockito和Junit运行代码As-Is关闭该链接,它运行时没有错误。不幸的是,我需要Mockito和TestNG,这让我在10次测试中有6次出错:
Mockito cannot mock this class: class replica.java.util.UUID$$PowerMock5
Mockito can only mock visible & non-final classes.
对不起,这是一个长代码帖,但我想完成。第一次测试通过,第二次,第5次,第6次,第7次,第9次和第10次全部失败。这可以在TestNG下工作吗?
编辑 - 添加版本号
以下是我的小修改代码:
/*
* Copyright 2008 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import static org.mockito.BDDMockito.given;
import static org.mockito.Matchers.anyLong;
import static org.mockito.Mockito.times;
import static org.powermock.api.mockito.PowerMockito.doNothing;
import static org.powermock.api.mockito.PowerMockito.mock;
import static org.powermock.api.mockito.PowerMockito.mockStatic;
import static org.powermock.api.mockito.PowerMockito.spy;
import static org.powermock.api.mockito.PowerMockito.verifyStatic;
import static org.powermock.api.mockito.PowerMockito.when;
import java.net.URL;
import java.net.URLConnection;
import java.net.URLEncoder;
import java.util.Collections;
import java.util.LinkedList;
import java.util.List;
import java.util.UUID;
import org.powermock.core.classloader.annotations.PrepareForTest;
import org.powermock.modules.testng.PowerMockObjectFactory;
import org.testng.Assert;
import org.testng.IObjectFactory;
import org.testng.annotations.ObjectFactory;
import org.testng.annotations.Test;
/**
* Demonstrates PowerMockito's ability to mock non-final and final system
* classes. To mock a system class you need to prepare the calling class for
* testing. I.e. let's say you're testing class A which interacts with
* URLEncoder then you would do:
*
* <pre>
*
* @PrepareForTest({A.class})
*
* </pre>
*/
// @RunWith(PowerMockRunner.class)
@PrepareForTest({ SystemClassUser.class })
public class SystemClassUserTest {
@ObjectFactory
public IObjectFactory getObjectFactory() {
return new PowerMockObjectFactory();
}
@Test
public void assertThatMockingOfNonFinalSystemClassesWorks() throws Exception {
mockStatic(URLEncoder.class);
when(URLEncoder.encode("string", "enc")).thenReturn("something");
Assert.assertEquals("something", new SystemClassUser().performEncode());
}
@Test
public void assertThatMockingOfTheRuntimeSystemClassWorks() throws Exception {
mockStatic(Runtime.class);
Runtime runtimeMock = mock(Runtime.class);
Process processMock = mock(Process.class);
when(Runtime.getRuntime()).thenReturn(runtimeMock);
when(runtimeMock.exec("command")).thenReturn(processMock);
Assert.assertSame(processMock, new SystemClassUser().executeCommand());
}
@Test
public void assertThatMockingOfFinalSystemClassesWorks() throws Exception {
mockStatic(System.class);
when(System.getProperty("property")).thenReturn("my property");
Assert.assertEquals("my property", new SystemClassUser().getSystemProperty());
}
@Test
public void assertThatPartialMockingOfFinalSystemClassesWorks() throws Exception {
spy(System.class);
when(System.nanoTime()).thenReturn(2L);
new SystemClassUser().doMoreComplicatedStuff();
Assert.assertEquals("2", System.getProperty("nanoTime"));
}
@Test
public void assertThatMockingOfCollectionsWork() throws Exception {
List<?> list = new LinkedList<Object>();
mockStatic(Collections.class);
Collections.shuffle(list);
new SystemClassUser().shuffleCollection(list);
verifyStatic(times(2));
Collections.shuffle(list);
}
@Test
public void assertThatPartialMockingOfFinalSystemClassesWorksForNonVoidMethods() throws Exception {
spy(System.class);
when(System.getProperty("property")).thenReturn("my property");
final SystemClassUser systemClassUser = new SystemClassUser();
systemClassUser.copyProperty("to", "property");
}
@Test
public void assertThatMockingStringWorks() throws Exception {
mockStatic(String.class);
final String string = "string";
final String args = "args";
final String returnValue = "returnValue";
when(String.format(string, args)).thenReturn(returnValue);
final SystemClassUser systemClassUser = new SystemClassUser();
Assert.assertEquals(systemClassUser.format(string, args), returnValue);
}
@Test
public void mockingStaticVoidMethodWorks() throws Exception {
mockStatic(Thread.class);
doNothing().when(Thread.class);
Thread.sleep(anyLong());
long startTime = System.currentTimeMillis();
final SystemClassUser systemClassUser = new SystemClassUser();
systemClassUser.threadSleep();
long endTime = System.currentTimeMillis();
Assert.assertTrue(endTime - startTime < 5000);
}
@Test
public void mockingURLWorks() throws Exception {
URL url = mock(URL.class);
URLConnection urlConnectionMock = mock(URLConnection.class);
when(url.openConnection()).thenReturn(urlConnectionMock);
URLConnection openConnection = url.openConnection();
Assert.assertSame(openConnection, urlConnectionMock);
}
@Test
public void mockingUUIDWorks() throws Exception {
// given
final UUID mock = mock(UUID.class);
mockStatic(UUID.class);
given(UUID.randomUUID()).willReturn(mock);
// when
String actual = new SystemClassUser().generatePerishableToken();
// then
Assert.assertEquals("00000000000000000000000000000000", actual);
}
}
如果它有帮助,这里是一个精简版本,执行一个失败的测试: [我不确定我是否应该编辑这个问题,我很抱歉,如果这是我应该做的,但是已经太久了,这种方法似乎更有意义。]
import static org.powermock.api.mockito.PowerMockito.mockStatic;
import static org.powermock.api.mockito.PowerMockito.when;
import org.powermock.core.classloader.annotations.PrepareForTest;
import org.powermock.modules.testng.PowerMockObjectFactory;
import org.testng.Assert;
import org.testng.IObjectFactory;
import org.testng.annotations.ObjectFactory;
import org.testng.annotations.Test;
@PrepareForTest({ SystemClassUser.class })
public class SystemClassUserTest {
@ObjectFactory
public IObjectFactory getObjectFactory() {
return new PowerMockObjectFactory();
}
@Test
public void assertThatMockingOfFinalSystemClassesWorks() throws Exception {
mockStatic(System.class);
when(System.getProperty("property")).thenReturn("my property");
Assert.assertEquals("my property", new SystemClassUser().getSystemProperty());
}
}
现在这里是同样剥离的SystemClassUser类:
import java.io.IOException;
public class SystemClassUser {
public String getSystemProperty() throws IOException {
return System.getProperty("property");
}
}
在此例外中运行结果:
FAILED:assertThatMockingOfFinalSystemClassesWorks org.mockito.exceptions.base.MockitoException: Mockito无法模拟这个类:class replica.java.lang.System $$ PowerMock0 Mockito只能模拟可见&amp;非最终课程。
答案 0 :(得分:8)
我只是将测试用例修改为
public class SystemClassUserTest extends PowerMockTestCase {
...
}
并且工作正常
编辑(一年后):我最近找到了System Rules library,我没有使用getProperty()
进行尝试,但是System.exit()
完全符合我的预期...
答案 1 :(得分:0)
我不确定TestNG会以什么方式与此错误有任何关系。您确定在两种情况下都使用相同版本的Mockito吗?