Файлы JSF внутри каталога WEB-INF, как к ним получить доступ?


Я хочу поместить мои файлы JSF 2.0 xhtml в WEB-INFjsf. Как же мне тогда получить к ним доступ? Я знаю, что все, что находится внутри WEB-INF, не подвергается воздействию извне, поэтому мне нужен контроллер, чтобы перенаправить меня на соответствующий JSP, верно? (Это также модель 2 pattern iirc).

Могу ли я достичь этого с помощью параметра inside web.XML для лица-конфиг.в XML? Я думаю, что FacesServlet является контроллером моего веб-приложения, поэтому он должен служить этой цели?

И еще один вопрос для понимания Образец Модели 2. Должно ли каждое действие сначала идти к сервлету, который затем обрабатывает следующий возможный шаг? Итак, простой <a href="anotherPage.html" /> запрещен в этом шаблоне, поскольку он не идет к управляющему сервлету?

2 11

2 ответа:

я хочу поместить мои файлы JSF 2.0 xhtml в WEB-INF\jsf. Как же мне тогда получить к ним доступ?

Вы не можете. Файлы в папке /WEB-INF недоступны напрямую.

Существует два варианта решения проблемы общедоступности исходных файлов JSF.

  1. Сопоставьте FacesServlet на *.xhtml вместо *.jsf.

  2. Или ограничьте прямой доступ к *.xhtml через <security-constraint> в web.xml.

    <security-constraint>
        <display-name>Restrict direct access to XHTML files</display-name>
        <web-resource-collection>
            <web-resource-name>XHTML files</web-resource-name>
            <url-pattern>*.xhtml</url-pattern>
        </web-resource-collection>
        <auth-constraint />
    </security-constraint> 
    

См. также:


и еще один вопрос для понимания модели 2. Должно ли каждое действие сначала идти к сервлету, который затем обрабатывает следующий возможный шаг?

FacesServlet уже делает это. Это контроллер. С JSF вы уже заканчиваете с простой javabean как модель и файл JSP / Facelets как представление. FacesServlet как контроллер уже взял всю неприятную работу по сбору параметров запроса, проверке, преобразованию, обновлению модели и навигации из ваших рук.

См. также:


таким образом, простой <a href="anotherPage.html" /> запрещен в этом шаблоне, так как он не идет к управляющему сервлету?

Нет, все в полном порядке. Контроллер будет работать в любое время, когда это необходимо. Если ресурс не нуждается в контроллере (то есть статическом ресурсе), то вам также не нужно пропускать его через какой-то контроллер.

В будущем, пожалуйста, задавайте несколько вопросов в отдельных вопросах переполнения стека.

Для доступа к страницам xhtml внутри папки WEB-INF/jsf Вы можете сделать следующее:

  1. переместить папку xhtml pages из webapp root в WEB-INF
  2. ввести шаблон "диспетчерский вид " в проект
  3. Map "Front Controller " servlet to url на основе страниц из приложения
  4. Map Faces Servlet to ".xhtml"
  5. внутри "диспетчер " переадресация запроса на страницу из "WEB-INF/jsf/<name>.xhtml"
  6. переопределить jsf ViewHandler getActionUrl, чтобы исключить "WEB-INF " из генерируется action url (из form, link, button)

Например, страницы xhtml находятся в корневой папке webapp " jsf". Все url между страницами похожи на jsf/<pageName>.xhtml. Поэтому мы делаем следующее:

  1. Переместить <webapp root>/jsf в <webapp root>/WEB-INF/jsf

  2. Создать FrontController сервлет:

"

public class FrontController extends HttpServlet {

        @Override
        protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            process(req, resp);
        }

        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            process(req, resp);
        }    

        private void process(HttpServletRequest request, HttpServletResponse response) {
             Dispatcher dispatcher = Dispatcher.getInstance();
             dispatcher.dispatch(request, response);
        }
}
  1. отображение переднего сервлета контроллера в web.xml в url на основе страниц:
<servlet>
    <servlet-name>Front Controller</servlet-name>
    <servlet-class>controllers.FrontController</servlet-class>
</servlet>
<servlet-mapping>
    <servlet-name>Front Controller</servlet-name>
    <url-pattern>/jsf/*</url-pattern>
</servlet-mapping>
  1. карта сталкивается с сервлетом в web.xml to .xhtml
<servlet>
    <servlet-name>Faces Servlet</servlet-name>
    <servlet-class>javax.faces.webapp.FacesServlet</servlet-class>
    <load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
    <servlet-name>Faces Servlet</servlet-name>
    <url-pattern>*.xhtml</url-pattern>
</servlet-mapping>
  1. создайте диспетчер , который перенаправляет request для исправления xhtml страницы:

"

public class Dispatcher {

    public void dispatch(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        String pageBase = "/WEB-INF/jsf/";
        String pagePath = null;
        String errorPage = "/WEB-INF/jsf/error.xthml";

        //here could be complicated logic to analyze if the page should be visible for security reasons, authorisation etc, business logic            
       //requested page could be taken from parsing requested URI
        //pageName = findPageNameFromURI(request.getRequestURI());

        pagePath = pageBase + pageName;

        //if page should not be visible
        pagePath = errorPage;            

        //forward to page inside WEB-INF/jsf
        request.getServletContext().getRequestDispatcher(pagePath).
                                   forward(request, response);        
    }   

}

Так что если url для страницы был /myapp/jsf/home.xhtml, то диспетчер перешлет его в myapp/WEB-INF/jsf/home.xhtml. А сервлет Faces будет обрабатывать запрос " .xhtml". Но если на странице используются jsf компоненты типа h:form, h:link, h:button и т. д., которые генерируют action или url, то url будет действительно включать "/WEB-INF". Поэтому, чтобы исключить его, нам нужно следующее шаг.

  1. Исключить "/WEB-INF " из jsf сгенерированного url (для формы jsf, ссылки, кнопки). Для этого:

    6.1 создать подкласс jsf ViewHandler и переопределить getActionUrl:

"

public class HiddenPageViewHandler extends ViewHandlerWrapper {

    private static final String WEB_INF = "/WEB-INF";

    private ViewHandler parent;

    public HiddenPageViewHandler(ViewHandler parent) {
        this.parent = parent;
    }

    @Override
    public String getActionURL(FacesContext context, String viewId) {
        String actionUrl = super.getActionURL(context, viewId);

        if (actionUrl != null && actionUrl.contains(WEB_INF)) {
            actionUrl = actionUrl.replace(WEB_INF, "");
        }        

        return actionUrl;
    }

    @Override
    public ViewHandler getWrapped() {
        return parent;
    }

}
6.2 сконфигурируйте jsf для использования указанного ViewHandler. В faces-config.xml добавить следующее:
   <application>
    ...
        <view-handler>
            controllers.HiddenPageViewHandler
        </view-handler>
   </application>