Генерация контекста лица вручную - PullRequest
5 голосов
/ 27 февраля 2009

У меня есть две системы, которые я пытаюсь интегрировать. Один построен на сырых сервлетах, новый - на JSF с IceFaces. Я пытаюсь облегчить межсистемный вход. Идея состоит в том, что у меня есть кнопка в старой системе, которая помещает соответствующую информацию на новый сайт и регистрирует их.

Ну, в идеале, я бы хотел использовать обычный старый сервлет для облегчения этого на новом сайте. Перейдите к сервлету нового сайта, сделайте то, что нужно, и перешли на панель управления.

Наша безопасность обрабатывается с помощью управляемого компонента. Однако к тому времени, когда вы доберетесь до сервлета, контекста лиц не будет. Итак, как бы я создал новый контекст лиц?

У меня есть план резервного копирования, в котором я всегда могу сослаться на фиктивную страницу .iface, которая создаст для меня FacesContext, а затем создаст вспомогательный компонент, который будет выполнять какие-то действия при его создании и пересылать на главную страницу. Но это очень похоже на взломать.

Любая помощь будет оценена!

РЕДАКТИРОВАТЬ: Я пошел с резервным путем. По сути, я размещаю на странице примерно так:

<f:view>
   <ice:outputText value="#{EntryPoint}"/>
</f:view

Фасоль выглядит так ...

public EntryPoint() {
      try {
         HttpServletRequest request = (HttpServletRequest) FacesContext.getCurrentInstance().getExternalContext().getRequest();
         HttpServletResponse response = (HttpServletResponse) FacesContext.getCurrentInstance().getExternalContext().getResponse(); 
         String loginID = request.getParameter("loginID");
         //Do some code to load the user/permissions
         response.sendRedirect(
            //The appropriate page
         );
      } catch (IOException ex) {
         logger.error(null, ex);
      } catch (SQLException ex) {
         logger.error(null, ex);
      }
   }

Это все еще похоже на взлом, но я не уверен, как обойти это. В идеале я бы отправил сервлету POST, получил бы loginID, собрал бы пользователя и поместил его прямо в управляемый бин. Но FacesContext не существует в этой точке.

Есть еще идеи?

1 Ответ

4 голосов
/ 27 февраля 2009

Я не уверен, что вы подразумеваете под "сайтом" в этом контексте.

Пара заметок:

  • Управляемые bean-компоненты никогда не будут доступны вне веб-приложения (WAR), в котором они определены.
  • Экземпляры объекта FacesContext в конечном итоге создаются и удаляются FacesServlet.service , поэтому запросы должны проходить через этот сервлет. Попытка создать контекст при других обстоятельствах может привести к неопределенному поведению.

Имея это в виду, вы можете создать последовательность запросов, подобную этой:

FacesServlet (mapping: /faces/*)
 -> /faces/jsfPage.jsp (a JSP with JSF controls)
    -> DispatchBean (calls ExternalContext.dispatch("/AnotherServlet")
       -> AnotherServlet

jsfPage.jsp содержит:

<f:view>
    <h:outputText value="#{dispatchBean.dispatch}" />
</f:view>

Свойство «dispatch» разрешается методом bean «getDispatch»:

public String getDispatch() {
    FacesContext context = FacesContext.getCurrentInstance();
    try {
        context.getExternalContext().dispatch("/FacesClientServlet");
    } catch (IOException e) {
        throw new FacesException(e);
    }
    return null;
}

Который отправляет сервлету:

public class FacesClientServlet extends javax.servlet.http.HttpServlet
        implements javax.servlet.Servlet {

    static final long serialVersionUID = 1L;

    @Override
    protected void doGet(HttpServletRequest request,
            HttpServletResponse response) throws ServletException, IOException {

        FacesContext context = FacesContext.getCurrentInstance();
        ELContext elContext = context.getELContext();
        ExpressionFactory expressionFactory = context.getApplication()
                .getExpressionFactory();
        ValueExpression expression = expressionFactory.createValueExpression(
                elContext, "#{myBean.text}", Object.class);
        Object value = expression.getValue(elContext);

        ResponseWriter writer = context.getResponseWriter();
        writer.write("" + value);

    }

}

Который выдает значение из управляемого компонента "myBean":

public class MyBean {

    private final String text = "Hello, World!";

    public String getText() {
        return text;
    }

}

Это все очень запутанно, и я бы не стал охотно делать это.


Альтернативой, которая может иметь свои собственные последствия, является создание собственного контекста, подобного этому:

public class ContextServlet extends javax.servlet.http.HttpServlet implements
        javax.servlet.Servlet {
    static final long serialVersionUID = 1L;

    private FacesContextFactory facesContextFactory;
    private Lifecycle lifecycle;

    @Override
    public void init(ServletConfig config) throws ServletException {
        super.init(config);

        LifecycleFactory lifecycleFactory = (LifecycleFactory) FactoryFinder
                .getFactory(FactoryFinder.LIFECYCLE_FACTORY);
        facesContextFactory = (FacesContextFactory) FactoryFinder
                .getFactory(FactoryFinder.FACES_CONTEXT_FACTORY);
        lifecycle = lifecycleFactory
                .getLifecycle(LifecycleFactory.DEFAULT_LIFECYCLE);
    }

    @Override
    protected void doGet(HttpServletRequest request,
            HttpServletResponse response) throws ServletException, IOException {

        FacesContext context = facesContextFactory.getFacesContext(
                getServletContext(), request, response, lifecycle);
        try {
            ELContext elContext = context.getELContext();
            ExpressionFactory expressionFactory = context.getApplication()
                    .getExpressionFactory();
            ValueExpression expression = expressionFactory
                    .createValueExpression(elContext, "#{myBean.text}",
                            Object.class);
            Object value = expression.getValue(elContext);

            PrintWriter pw = response.getWriter();
            try {
                pw.write("" + value);
            } finally {
                pw.close();
            }
        } finally {
            context.release();
        }
    }

}

Опять же, я бы избегал такого подхода, если это возможно.

...