与查询参数关联的转换/验证失败时执行重定向

时间:2014-04-13 23:58:45

标签: jsf primefaces jsf-2.2

以下是<f:viewAction>的简单用例。

<f:metadata>
    <f:viewParam name="id" value="#{testManagedBean.id}" maxlength="20"/>
    <f:viewAction action="#{testManagedBean.viewAction}"/>
</f:metadata>

涉及托管bean。

@ManagedBean
@ViewScoped
public final class TestManagedBean implements Serializable {

    private static final long serialVersionUID = 1L;
    private Long id; //Getter and setter.

    public void viewAction() {
        System.out.println("viewAction() called : " + id);
    }
}

参数id通过URL传递。如果通过相关网址传递了非xxx这样的非数字值,并且未调用与viewAction()的侦听器关联的<f:viewAction>方法,则会出现转换错误。

在这种情况下,id的值为null。我希望重定向到另一个页面,当id无法转换为所需的目标类型时(如本例所示)或id未根据指定的验证标准进行验证,以避免可能发生的潜在异常只要在相应的托管bean中尝试访问这些参数,就会抛出PrimeFaces的LazyDataModel#load()方法或相关托管bean中的其他位置。为此,应调用viewAction()方法。

如何处理?我应该使用

<f:event type="preRenderView">

<f:viewAction>

一起使用

2 个答案:

答案 0 :(得分:8)

这是specified行为。当PROCESS_VALIDATIONS阶段以validation failure结尾时,UPDATE_MODEL_VALUESINVOKE_APPLICATION阶段都会被跳过。与<h:form>的“常规”形式完全相同。将<f:viewParam>视为<h:inputText>,将<f:viewAction>视为<h:commandButton>,它将变得更加清晰。

根据您的特定要求,在转换/验证失败时执行重定向,至少有3个解决方案:

  1. 如您所知,添加<f:event listener>。我宁愿挂钩postValidate事件而不是更好的自我可记录性。

    <f:metadata>
        <f:viewParam name="id" value="#{bean.id}" maxlength="20" />
        <f:event type="postValidate" listener="#{bean.redirectIfNecessary}" />
        <f:viewAction action="#{bean.viewAction}" />
    </f:metadata>
    
    public void redirectIfNecessary() throws IOException {
        FacesContext context = FacesContext.getCurrentInstance();
    
        if (!context.isPostback() && context.isValidationFailed()) {
            context.getExternalContext().redirect("some.xhtml");
        }
    }
    

    FacesContext#isPostback()的检查可防止在同一视图(如果有)中“常规”表单的验证失败时执行重定向。


  2. 扩展内置LongConverter,您可以在getAsObject()中执行重定向(验证器不可用,因为Long的默认转换器在非数字输入上已经失败;如果转换器失败,验证器永远不会被解雇)。然而,这是一种糟糕的设计(紧耦合)。

    <f:metadata>
        <f:viewParam name="id" value="#{bean.id}" converter="idConverter" />
        <f:viewAction action="#{bean.viewAction}" />
    </f:metadata>
    
    @FacesConverter("idConverter")
    public class IdConverter extends LongConverter {
    
        @Override
        public Object getAsObject(FacesContext context, UIComponent component, String value) {
            if (value == null || !value.matches("[0-9]{1,20}")) {
                try {
                    context.getExternalContext().redirect("some.xhtml");
                    return null;
                }
                catch (IOException e) {
                    throw new FacesException(e);
                }
            }
            else {
                return super.getAsObject(context, component, value);
            }
        }
    
    }
    

    如果需要,您可以在<f:attribute>内使用<f:viewParam>将“参数”传递给转换器。

    <f:viewParam name="id" value="#{bean.id}" converter="idConverter">
        <f:attribute name="redirect" value="some.xhtml" />
    </f:viewParam>
    
    String redirect = (String) component.getAttributes().get("redirect");
    context.getExternalContext().redirect(redirect);
    

  3. 创建一个自定义标记处理程序,它与<f:event listener>基本相同,但不需要额外的辅助bean方法。

    <html ... xmlns:my="http://example.com/ui">
    
    <f:metadata>
        <f:viewParam name="id" value="#{bean.id}" maxlength="20" />
        <my:viewParamValidationFailed redirect="some.xhtml" />
        <f:viewAction action="#{bean.viewAction}" />
    </f:metadata>
    

    com.example.taghandler.ViewParamValidationFailed

    public class ViewParamValidationFailed extends TagHandler implements ComponentSystemEventListener {
    
        private String redirect;
    
        public ViewParamValidationFailed(TagConfig config) {
            super(config);
            redirect = getRequiredAttribute("redirect").getValue();
        }
    
        @Override
        public void apply(FaceletContext context, UIComponent parent) throws IOException {
            if (parent instanceof UIViewRoot && !context.getFacesContext().isPostback()) {
                ((UIViewRoot) parent).subscribeToEvent(PostValidateEvent.class, this);
            }
        }
    
        @Override
        public void processEvent(ComponentSystemEvent event) throws AbortProcessingException {
            FacesContext context = FacesContext.getCurrentInstance();
    
            if (context.isValidationFailed()) {
                try {
                    context.getExternalContext().redirect(redirect);
                }
                catch (IOException e) {
                    throw new AbortProcessingException(e);
                }
            }
        }
    
    }
    

    /WEB-INF/my.taglib.xml

    <?xml version="1.0" encoding="UTF-8"?>
    <facelet-taglib
        xmlns="http://java.sun.com/xml/ns/javaee"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-facelettaglibrary_2_0.xsd"
        version="2.0"
    >
        <namespace>http://example.com/ui</namespace>
    
        <tag>
            <tag-name>viewParamValidationFailed</tag-name>
            <handler-class>com.example.taghandler.ViewParamValidationFailed</handler-class>
        </tag>  
    </facelet-taglib>
    

    /WEB-INF/web.xml

    <context-param>
        <param-name>javax.faces.FACELETS_LIBRARIES</param-name>
        <param-value>/WEB-INF/my.taglib.xml</param-value>
    </context-param>
    

    是的,这是一些代码,但它最终是干净且可重复使用的<my:viewParamValidationFailed>代码,实际上非常适合新的OmniFaces功能。

答案 1 :(得分:1)

为什么不自行验证id

@ManagedBean
@ViewScoped
public final class TestManagedBean implements Serializable
{
    private String id;     //Getter and setter.
    private Long validId;  //Getter and setter.

    public void viewAction() {
        try {
            validId = Long.parseLong(id);
        } catch (NumberFormatException ex) {
            FacesContext facesContext = FacesContext.getCurrentInstance();
            String outcome = "redirect.xhtml";
            facesContext.getApplication().getNavigationHandler().handleNavigation(facesContext, null, outcome);
        }
    }
}