如何在JSF中定制数据库唯一约束错误消息



我有一个带有UNI约束的Name,当用户注册一个重复的Name时,它会抛出一个异常:

Caused by: java.sql.SQLIntegrityConstraintViolationException: Duplicate entry 'John' for key 'UK_t622yodh32su1s2wseebkimwp'

在JSF视图页面中,异常处理程序捕获的异常显示给用户:

Duplicate entry 'John' for key 'UK_t622yodh32su1s2wseebkimwp'
代码:

public void register (Student student) {
   ...
catch (Exception e) {
            String errorMessage = getRootErrorMessage(e);
            FacesMessage m = new FacesMessage(FacesMessage.SEVERITY_ERROR, errorMessage, "Registration unsuccessful");
            facesContext.addMessage(null, m);
        }
}
private String getRootErrorMessage(Exception e) {
        String errorMessage = "Registration failed.";
        if (e == null) {
            return errorMessage;
        }
        Throwable t = e;
        while (t != null) {
            errorMessage = t.getLocalizedMessage();
            t = t.getCause();
        }
        return errorMessage;
    }

Registration.xhtml:

<p>
            <h:panelGrid columns="2">
                <h:commandButton id="register"
                    action="#{registrationController.register(registrationView.student)}" value="Register"
                    styleClass="register" />
                <h:messages styleClass="messages" errorClass="invalid"
                    infoClass="valid" warnClass="warning" globalOnly="true" />
            </h:panelGrid>
        </p>
这里的问题是,它只显示异常错误代码,如上所述。我希望错误信息是这样的:
The user name has been registered, Please use another one. Thanks.

与SQL约束违反错误消息相比,这更加用户友好。如何做到这一点?

我真的不确定你到底是什么意思"异常错误代码"?

在你的代码中,你显式地把Java异常消息放在FacesMessage中。如果你想要一些不同的东西显示给用户,那么你应该在FacesMessage中写一些不同的东西。

但是,我建议在尝试持久化用户之前,先检查该用户是否已经存在。您应该使用以下简单的预检查来覆盖大多数情况:

public void register (Student student) {
    if(service.exists(student)) {
        String message = "The user name has been registered, Please use another one. Thanks.";
        FacesMessage m = new FacesMessage(FacesMessage.SEVERITY_ERROR, message, 
                                         "Registration unsuccessful");
        facesContext.addMessage(null, m);
    } else {
        try {
            service.create(student);
        } catch (Exception e) {
            String errorMessage = getRootErrorMessage(e);
            FacesMessage m = new FacesMessage(FacesMessage.SEVERITY_ERROR, errorMessage, 
                                              "Registration unsuccessful");
            facesContext.addMessage(null, m);
        }
    }
}

除此之外,您还可以显式地检查SQLException中是否违反完整性约束,但是您无法通过这种方式区分不同类型的完整性约束违反。

public static boolean isConstraintViolation(SQLException e) {
    return e.getSQLState().startsWith("23");
}

您也可以检查错误消息是否包含字符串Duplicate entry,但这不是很可靠。

我更喜欢第一个建议的precheck尝试。如果用户已经存在,添加一个适当的FacesMassages到FacesContext。它将涵盖大多数情况。在非常不可能的情况下,当两个用户同时注册相同的用户名时,向用户显示"Unexpected error"。

我认为在添加消息vs jsf生命周期中有一些错误尝试创建一个PhaseListener来在faces上下文中添加消息,像这样:

public class FacesMessagesListener implements PhaseListener {
    @Override
    public void afterPhase(final PhaseEvent event) {
           //add my messages to facesContext
    }
    @Override
    public void beforePhase(final PhaseEvent event) {
           //add my messages to facesContext
    }
    @Override
    public PhaseId getPhaseId() {
        return PhaseId.ANY_PHASE;
    }
}

相关内容

  • 没有找到相关文章

最新更新