Java HtmlUnit类强制转换异常:无法将TextPage强制转换为HtmlPage

Java HtmlUnit类强制转换异常:无法将TextPage强制转换为HtmlPage,java,spring,htmlunit,Java,Spring,Htmlunit,我正在尝试使用SpringMVC测试和HTMLUnit测试我的网页。我需要得到Html页面,以便我可以在我的页面中设置值并提交它,但我得到了以下异常。我怎样才能做到这一点?我走对了吗?请把我当成是TDD的新手。 错误堆栈跟踪: java.lang.ClassCastException: com.gargoylesoftware.htmlunit.TextPage cannot be cast to com.gargoylesoftware.htmlunit.html.HtmlPage a

我正在尝试使用SpringMVC测试和HTMLUnit测试我的网页。我需要得到Html页面,以便我可以在我的页面中设置值并提交它,但我得到了以下异常。我怎样才能做到这一点?我走对了吗?请把我当成是TDD的新手。

错误堆栈跟踪:

java.lang.ClassCastException: com.gargoylesoftware.htmlunit.TextPage cannot be cast to com.gargoylesoftware.htmlunit.html.HtmlPage
    at com.demo.htmlunit.test.LoginControllerHtmlUnitTest.userLoginTest(LoginControllerHtmlUnitTest.java:51)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
    at java.lang.reflect.Method.invoke(Unknown Source)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:47)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:44)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26)
    at org.springframework.test.context.junit4.statements.RunBeforeTestMethodCallbacks.evaluate(RunBeforeTestMethodCallbacks.java:74)
    at org.junit.internal.runners.statements.RunAfters.evaluate(RunAfters.java:27)
    at org.springframework.test.context.junit4.statements.RunAfterTestMethodCallbacks.evaluate(RunAfterTestMethodCallbacks.java:83)
    at org.springframework.test.context.junit4.statements.SpringRepeat.evaluate(SpringRepeat.java:72)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:233)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:87)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:238)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:63)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:236)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:53)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:229)
    at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
    at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:71)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:309)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:176)
    at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50)
    at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)
我的测试课程是:

import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.test.context.web.WebAppConfiguration;
import org.springframework.test.web.servlet.MockMvc;
import org.springframework.test.web.servlet.htmlunit.MockMvcWebConnection;
import org.springframework.test.web.servlet.setup.MockMvcBuilders;
import org.springframework.web.context.WebApplicationContext;

import com.gargoylesoftware.htmlunit.WebClient;
import com.gargoylesoftware.htmlunit.html.HtmlForm;
import com.gargoylesoftware.htmlunit.html.HtmlPage;
import com.gargoylesoftware.htmlunit.html.HtmlPasswordInput;
import com.gargoylesoftware.htmlunit.html.HtmlSubmitInput;
import com.gargoylesoftware.htmlunit.html.HtmlTextInput;

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
@WebAppConfiguration
public class LoginControllerHtmlUnitTest {
    @Autowired
    private WebApplicationContext context;

    private WebClient webClient;

    @Before
    public void setup() {
        System.out.println("In set Up");
        MockMvc mockMvc = MockMvcBuilders.webAppContextSetup(context).build();
        webClient = new WebClient();
        webClient.setWebConnection(new MockMvcWebConnection(mockMvc));
    }

    @After
    public void cleanup() {
        this.webClient.closeAllWindows();
    }

    @Test
    public void userLoginTest() {
        try {
            // Load the Login Form
            System.out.println("In User Login Test");
            HtmlPage createLoginFormPage = webClient
                    .getPage("http://localhost:8080/htmlunitdemo/login");
            System.out.println("createLoginFormPage:"+createLoginFormPage);
            // Setting Values in the Login form
            HtmlForm form = createLoginFormPage.getHtmlElementById("loginForm");
            HtmlTextInput usernameInput = createLoginFormPage
                    .getHtmlElementById("username");
            usernameInput.setValueAttribute("admin");
            HtmlPasswordInput passwordInput = createLoginFormPage
                    .getHtmlElementById("passcode");
            passwordInput.setText("admin123");
            HtmlSubmitInput submit = form.getOneHtmlElementByAttribute("input",
                    "type", "submit");
            // Submitting Form
            HtmlPage newPage = submit.click();


            System.out.println("New Page:" + newPage.asXml());

        } catch (Exception e) {
            e.printStackTrace();
        }

    }
}
applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
    xmlns:mvc="http://www.springframework.org/schema/mvc"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd
    http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd"
    default-lazy-init="true">

    <context:component-scan base-package="com.demo.htmlunit.controller" />
    <mvc:annotation-driven />

    <bean id="viewResolver"
        class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix" value="/WEB-INF/pages/" />
        <property name="suffix" value=".jsp" />
    </bean>
</beans>

HtmlUnit根本不关心
applicationContext.xml
或Java服务器代码。它所关心的只是服务器生成的输出。如果输出是随机文本,则HtmlUnit将假定它正在获取文本文件,并将创建一个文本页面来处理它。如果输出是HTML,那么HtmlUnit将假定它得到一个HTML文件,并创建一个HtmlPage来保存结果。根据你的问题,你显然需要后者

因此,请验证服务器应用程序的输出(
http://localhost:8080/htmlunitdemo/login
)并确保它输出有效的HTML代码。

替换代码行:

 HtmlPage newPage = submit.click();
与:

 TextPage newPage = submit.click();

我不知道我是否已经找到了解决办法。当按钮是Submit(提交)时会发生错误,而不是按钮类型。 我通过添加一个按钮并执行它来解决这个问题

HtmlElement buttonCustom = (HtmlElement) page.createElement("button");
            buttonCustom.setAttribute("type", "submit");
            buttonCustom.setAttribute("name", "submit");
            buttonCustom.setAttribute("value", "Load");
            form.appendChild(buttonCustom);

问题是HTML单元无法转换不完整的HTML页面(例如,一些未关闭的标记)。因此,我可以使用HTMLPasser解决这个错误,它包含在HTMLUnit的包中(我使用的是2.36.0v)。HtmlPasser完成并处理此类强制转换错误

//Web client creation.
Page page = webClient.getPage(url);
HtmlPage tmpPage = HTMLParser.parseHtml(page.getWebResponse(), webClient.getCurrentWindow());
// use tmpPage here

URL“”是有效的html文件。如果我注释//webClient.setWebConnection(新的MockMvcWebConnection(mockMvc));此代码位于setUp()部分。代码运行良好。问题完全相同。HtmlUnit不关心请求来自哪里。将其设置为URL或模拟连接,如果内容不是有效的HTML,则无法创建HtmlPage。现在,如果没有那行代码就可以正常工作。。。为什么不删除它?实际上,我想模拟WebConnection对象。如果我删除了代码,连接就不会被模拟,服务器必须启动才能运行测试,但测试没有结果。
//Web client creation.
Page page = webClient.getPage(url);
HtmlPage tmpPage = HTMLParser.parseHtml(page.getWebResponse(), webClient.getCurrentWindow());
// use tmpPage here