我有一个 Spring Boot 应用程序,它以通常的方式使用 Spring MVC,带有一堆 @RequestMapping
方法、Freemarker 定义等。这一切都与 WebMvcConfigurerAdapter
类捆绑在一起。
我想提供一项服务,用户提交有效 URL 列表,web 应用程序将计算出将调用哪个 Controller ,传入参数,并返回每个 URL 的组合结果——所有这些都合而为一请求。
这将使用户不必进行数百次 HTTP 调用,但仍允许他们在需要时发出一次性请求。理想情况下,我只需要注入(inject)一个自动配置的 Spring bean,这样我就不必重复 Spring 内部进行的 URL 解析、调整和处理,并且 Controller 的其他 Controller 列表永远不会与 < em>真正的 Controller 列表。
我希望写这样的东西(简化为只处理一个 URL,这毫无意义但更容易理解):
@Autowired BeanThatSolvesAllMyProblems allMappings;
@PostMapping(path = "/encode", consumes = MediaType.TEXT_PLAIN_VALUE)
@ResponseBody
public String encode(@RequestBody String inputPath) {
if (allMappings.hasMappingForPath(inputPath)) {
return allMappings.getMapping(inputPath).execute();
} else {
return "URL didn't match, sorry";
}
}
相反,我不得不定义 Spring bean 我不知道它们做了什么,并且一直在重复一些 Spring 打算为我做的事情,我担心这不会起作用 与用户自己调用电话时的情况相同:
// these two are @Beans, with just their default constructor called.
@Autowired RequestMappingHandlerMapping handlers;
@Autowired RequestMappingHandlerAdapter adapter;
@PostMapping(path = "/encode", consumes = MediaType.TEXT_PLAIN_VALUE)
@ResponseBody
public String encode(@RequestBody String inputText) {
final HttpServletRequest mockRequest = new MockHttpServletRequest(null, inputText);
final StringBuilder result = new StringBuilder();
this.handlers.getHandlerMethods().forEach((requestMappingInfo, handlerMethod) -> {
if (requestMappingInfo.getPatternsCondition().getMatchingCondition(mockRequest) != null) {
try {
final MockHttpServletResponse mockResponse = new MockHttpServletResponse();
result.append("Result: ").append(adapter.handle(mockRequest, mockResponse, handlerMethod));
result.append(", ").append(mockResponse.getContentAsString());
result.append("\n");
} catch (Exception e) {
logger.error(e.getMessage(), e);
}
}
});
return result.toString();
}
我以为我在这条路上做得很好,但它因 Missing URI template variable
错误而失败,而且我不仅不知道如何将请求参数放入(另一件事哪个 Spring 可以自行处理),但我什至不确定这是执行此操作的正确方法。那么我如何从 webapp 本身内部“反射性地”模拟 Spring MVC 请求?
最佳答案
JSON API
规范通过允许每个请求发送多个操作来解决这个问题。甚至有一个相当成熟的实现支持这个功能,称为 Elide
.但我想这可能不能完全满足您的要求。
无论如何,这是您可以做的。
你必须考虑到 DispatcherServlet
持有 handlerMappings
用于检测适当的列表 request
处理程序和 handlerAdaptors
.两个列表的选择策略都是可配置的(参见 DispatcherServlet#initHandlerMappings
和 #initHandlerAdapters
)。
您应该想出一种您更喜欢检索 handlerMappings
列表的方法/initHandlerAdapters
并与DispatcherServlet
保持同步.
之后您可以实现自己的 HandlerMapping
/HandlerAdaptor
(或在您的示例中呈现一个 Controller
方法)将处理 request
至 /encode
小路。
顺便说一句,HandlerMapping
正如 javadoc 所说的那样
Interface to be implemented by objects that define a mapping between requests and handler objects
或者简单地说如果我们取DefaultAnnotationHandlerMapping
这将映射我们的 HttpServletRequests
至 @Controller
用 @RequestMapping
注释的方法.有这个映射 HandlerAdapter
准备传入请求以使用 Controller 方法,f.ex。提取请求 params
, body
并使用它们调用 Controller 的方法。
有了这个,你可以提取URL
来自主要request
, 创建 stub 列表 HttpRequests
保存进一步处理所需的信息并通过它们循环调用:
HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
for (HandlerMapping hm : this.handlerMappings) {
if (logger.isTraceEnabled()) {
logger.trace(
"Testing handler map [" + hm + "] in DispatcherServlet with name '" + getServletName() + "'");
}
HandlerExecutionChain handler = hm.getHandler(request);
if (handler != null) {
return handler;
}
}
return null;
}
有一个handlerMapping
你叫
HandlerAdapter getHandlerAdapter(Object handler) throws ServletException {
for (HandlerAdapter ha : this.handlerAdapters) {
if (logger.isTraceEnabled()) {
logger.trace("Testing handler adapter [" + ha + "]");
}
if (ha.supports(handler)) {
return ha;
}
}
然后你终于可以打电话了
ha.handle(processedRequest, response, mappedHandler.getHandler());
这反过来会使用参数执行 Controller 的方法。
但是考虑到所有这些,我不建议采用这种方法,而是考虑使用 JSON API
规范或任何其他。
关于java - 如何解析 URL 并使用 Spring MVC 'reflectively' 运行方法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39212895/