gpt4 book ai didi

java - 如何让 JUnit 对打开的多个选项卡/窗口但只有一个 session 进行建模

转载 作者:行者123 更新时间:2023-12-01 05:43:06 26 4
gpt4 key购买 nike

我正在使用 JUnit 为 Spring Framework 2.5 编写单元测试。更具体地说,我使用 MockHttpServletRequest 来测试我的 Controller 。我在测试特定场景时遇到了问题。

应用程序正在编辑对象,并保留到 Hibernate。这些对象的工作原理如下:

Class Workflow {
... various properties and methods
Long workflowId; // primary key for this object
Long subclassId; // foreign key to another class
String subclassType; // what kind of other class?
}

Class Request {
.. various other properties and methods
Long subclassId; // primary key for this object
Long workflowId; // foreign key to Workflow
}

工作流始终附加到一个且仅一个请求,并且请求始终附加到一个且仅一个工作流。但是,Workflow 可以链接到多个对象/表,因此 subclassType 必须跟踪要查找的其他表。因此,这有点像 Request 继承/实现 Workflow,但数据库将它们存储在单独的表,因此 Java 被实现为单独的类。

这些乱七八糟的东西不是我写的。我正在维护它。它被广泛使用,具有多种不同的应用程序和子类类型,因此我可以做的“清理”工作是有限的。

相关 Web 应用程序将 Workflow 对象放入 HttpSession 中,但在表单中编辑请求。当有人打开多个窗口/选项卡,每个窗口中都有不同的请求时,最后一个打开的工作流对象将覆盖任何先前打开的工作流对象( session 附加到 cookie,域中只有一组 cookie,由Web 浏览器中的所有窗口)。如果他们提交了第一个请求的表单,则最后一个请求中的工作流对象(从 HttpSession 中提取)将与提交的请求(不是正确的请求)一起写入,从而损坏数据库中的数据。

我们最终得到了有多个工作流连接到它们的请求,以及没有连接到它们的工作流的请求。

明显的解决方法是重写应用程序,使其永远不会将工作流存储在 HttpSession 中。 flowId是Request中的字段之一;我们通过隐藏字段将其保留到表单中。是的,我们更频繁地访问数据库,但我们没有交叉连接我们的记录。

情况就是这样。现在的问题是:如何在 JUnit 中对此进行建模?这样以后就不会再有人遇到这种情况了?

我们使用 MockHttpServletRequest,插入表单值并使用 AnnotationMethodHandlerAdapter 和 Controller 对象提交它们。举个例子:

    MockHttpServletRequest request = new MockHttpServletRequest();
MockHttpServletResponse response = new MockHttpServletResponse();
AnnotationMethodHandlerAdapter handler = new AnnotationMethodHandlerAdaptor();
RequestFormController controller = new RequestFormController();

request.setMethod("GET");
request.setRequestURI("/EditRequest.do"); // handled by the RequestFormController

request.setParameter("wfId", Long.toString(wfid)); // wfId is the workflow ID of a valid Workflow with an attached Request
ModelAndView mv = handler.handle(request, response, controller);

request.setParameter("wfId", mv.getModel().get("wfId");
request.setParameter("requestText", "blah blah blah");

request.setMethod("POST");
request.setRequestURI("/UpdateRequest.do"); // handled by the RequestFormController
mv = handler.handle(request, response, controller);

这都是非常简单的 Spring/JUnit 内容。问题是,要在两个不同的选项卡/窗口中打开某些内容,您必须有两个不同的请求,每个请求都有自己的 HttpSession。但是,使用 Web 浏览器时,这两个不同的请求共享相同的 cookie,这意味着它们共享相同的 session 信息。如何让 JUnit 对其进行建模?

我可以将 cookie 从一个复制到另一个,但这似乎并没有暴露错误。 MockHttpServletRequest 仍然将 HttpSession 信息分开。我很难将 HttpSession 属性从一个复制到另一个。尽管我已经这样做了,但我似乎也无法暴露错误。

有什么建议吗?这是一个非常现实的问题,MockHttpServletRequest 和我们正在做的所有其他与 JUnit 相关的事情并没有准确地对 Tomcat 和 Web 浏览器的现实世界进行建模。

和平。

最佳答案

编写测试时,您可以显式创建 session 并将所有请求设置为使用同一 session 。

MockHttpSession session = new MockHttpSession();
request.setSession(session);

关于java - 如何让 JUnit 对打开的多个选项卡/窗口但只有一个 session 进行建模,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6625764/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com