如何从战争中提供角度应用

时间:2016-05-06 06:15:47

标签: java angularjs maven

我有一个使用maven构建为战争的Web应用程序。此Web应用程序包含我的后端作为休息服务。

我还有一个角色应用程序,我想使用它来使用所述Web应用程序(它使用npm和webpack进行捆绑)。

如何从包含我后端的同一战争中提供我的角度应用程序?说,在路径... / index.html?

2 个答案:

答案 0 :(得分:1)

您正在讨论来自Web应用程序的提供,因此您的角度应用程序应嵌入WAR中以便执行此操作。如果您有2个项目的原因是因为您希望将问题分开,那么为什么Web应用需要为角度应用提供服务?我会选择你想要的项目设计模式并坚持下去。

  • 在REST应用程序内部构建/打包角度应用程序。您的servlet可以提供初始index.html,然后处理REST API的剩余请求。这允许您为两者共享相同的上下文根,而无需任何其他干预。
  • 将它们分开并让一个小的Web应用程序为您的角度前端提供服务(也可以使用节点),然后只在部署的WAR的路径上访问您的REST后端。这意味着您将同时拥有2个应用程序(1个用于提供角度资源,1个用于提供REST服务),每个应用程序都有自己的上下文根。

基于提供的最小细节,您可以使用几乎任何servlet框架完成所要做的事情,但是考虑到您没有提供诸如servlet实现/配置(即Spring?vanilla HttpServlets?)等信息,也没有你有没有提供项目的目录结构,我现在不能提供更多。

答案 1 :(得分:1)

为了让我的Angular项目在WAR中运行,我创建了一个Maven项目,其中只有2个文件(在Angular文件旁边):

  • 的pom.xml
  • 的src /主/ JAVA / COM /计算器/ AngularFilter.java

pom.xml 的内容:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.stackoverflow.example</groupId>
    <artifactId>angular-example</artifactId>
    <version>1.0.0</version>
    <packaging>war</packaging>

    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-war-plugin</artifactId>
                <configuration>
                    <warSourceDirectory>${basedir}/src/angular/dist/angular</warSourceDirectory>
                </configuration>
            </plugin>
        </plugins>
    </build>

    <dependencies>
        <dependency>
            <groupId>javax</groupId>
            <artifactId>javaee-web-api</artifactId>
            <version>7.0</version>
            <scope>provided</scope>
        </dependency>
    </dependencies>
</project>

warSourceDirectory 元素中,我引用了Angular构建输出。 (Angular文件必须使用正确的 - base-href 值构建。)

pom.xml 文件足以让Angular应用程序正常工作。  但是,如果没有 AngularFilter.java 文件,刷新Angular路由将不起作用。使用以下过滤器类,还可以刷新Angular路径:

package com.stackoverflow;

import java.io.IOException;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;

@WebFilter("/*")
public class AngularFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {}

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        HttpServletRequest httpRequest = ((HttpServletRequest) request);
        String contextPath = httpRequest.getContextPath();
        String requestURI = httpRequest.getRequestURI();
        String angularPath = requestURI.substring(contextPath.length());
        if (isAngularRoute(angularPath)) {
            request.getRequestDispatcher("/index.html").forward(request, response);
        } else {
            chain.doFilter(request, response);
        }
    }

    private boolean isAngularRoute(String angularPath) {
        return
                /* Do not reroute files in the Angular root, such as index.html,
                   main.js, polyfills.js, runtime.js, styles.css and favicon.ico */
                ((angularPath.indexOf('/', 1) != -1) || (angularPath.indexOf('.') == -1))
                /* Do not reroute static files */
            && !angularPath.startsWith("/assets/");
    }

    @Override
    public void destroy() {}
}

还有其他解决方案使用Spring MVC来启用刷新Angular路由,如下所述:Springboot/Angular2 - How to handle HTML5 urls?

但是,我认为如果单个Filter类可以完成这项工作,那么使用Spring是不合适的。