我遇到了一个复杂的场景,我们正在使用带有复合键的旧数据库,客户希望能够改变两个数据库pks," expenseDate"和" adjustmentNumber"。为了能够更改pks,我必须使用HQL查询。当然,这已经导致了另一系列验证问题。我的工作是填充一个域,以便我可以验证它。
到目前为止,一切正常,直到我们有一个验证错误,我想返回到UI。
我有以下网址,该网址使用recoveryDetail
控制器和edit
操作来呈现网页。
http://localhost:8080/pisr/recoveryDetail/edit?division=ALBANY&peid=PI0003&orgkey=14046701&expenseDate=07-22-2015&adjustmentNumber=1
修改操作
def edit() {
//Parse clean url expense date
params.expenseDate = new SimpleDateFormat('MM-dd-yyyy').parse(params.expenseDate)
def recoveryDetailInstance = RecoveryDetail.get(new RecoveryDetail(params))
if(recoveryDetailInstance == null) {
redirect(uri:'/')
return
}
[recoveryDetailInstance: recoveryDetailInstance, disabled: isdisabled(recoveryDetailInstance.batchOverride)]
}
以下更新操作。
更新操作
@Transactional
def update() {
params.pk_expenseDate = getDateParser(params.pk_expenseDate)
params.expenseDate = getDateParser(params.expenseDate)
params.adjustmentNumber = getAdjustementNumber(params)
RecoveryDetail recoveryDetailInstance = new RecoveryDetail(params);
recoveryDetailInstance.division = params.pk_division
recoveryDetailInstance.peid = params.pk_peid
recoveryDetailInstance.orgkey = params.pk_orgkey
recoveryDetailInstance .validate()
if(recoveryDetailInstance .hasErrors()) {
flash.message = "test"
respond view: "edit", model:[recoveryDetailInstance:recoveryDetailInstance]
return
} else {
def sqlParams = [
pk_division:params.pk_division,
pk_peid:params.pk_peid,
pk_orgkey:params.pk_orgkey,
pk_expenseDate:params.pk_expenseDate,
pk_adjustmentNumber:params.int('pk_adjustmentNumber'),
approved:YesNoTypes.valueOf(params.approved),
batchOverride:YesNoTypes.valueOf(params.batchOverride),
adjustmentFlag:params.adjustmentFlag,
adjustmentNumber:params.adjustmentNumber,
projectHours:new BigDecimal(params.projectHours),
percentEffort:new BigDecimal(params.percentEffort),
totalHours:new BigDecimal(params.totalHours),
expenseDate:params.expenseDate
]
RecoveryDetail.executeUpdate(recoveryDetailQuery, sqlParams)
}
编辑gsp
<g:form class="form-horizontal" url="[resource:recoveryDetailInstance, action:'update']" method="PUT">
<!-- hidden fields contain params from url (composite key)-->
<g:hiddenField name="pk_division" value="${recoveryDetailInstance?.division}"/>
<g:hiddenField name="pk_peid" value="${recoveryDetailInstance?.peid}"/>
<g:hiddenField name="pk_orgkey" value="${recoveryDetailInstance?.orgkey}"/>
<g:hiddenField name="pk_expenseDate" value="${formatDate(format:'MM/dd/yyyy',date:recoveryDetailInstance?.expenseDate)}" />
<g:hiddenField name="pk_adjustmentNumber" value="${recoveryDetailInstance?.adjustmentNumber}"/>
<div class="row">
<div class="col-md-6">
<g:render template="form" model="[recoveryDetailInstance: recoveryDetailInstance, 'mode':'edit']"/>
</div>
</div>
<div class="form-group">
<div class="col-sm-offset-2 col-sm-10">
<g:actionSubmit class="btn btn-primary" action="update" disabled="${disabled}" value="${message(code: 'default.button.update.label', default: 'Update')}"/>
</div>
</div>
</g:form>
问题 当用户导致验证错误触发服务器端响应时,我遇到了不同返回类型的以下问题。
重定向 - 这会返回Flash消息,但会将gsp重定向到编辑操作,编辑操作会触发初始化程序查询并将所有表单数据替换为原始数据。
示例 -
redirect (controller:"recoveryDetail", action:"edit", params:["division":params.pk_division, "peid":params.pk_peid, "orgkey": params.pk_orgkey, "expenseDate":params.expenseDate.format("MM-dd-yyyy"), "adjustmentNumber":params.adjustmentNumber])
回应 - 所以我认为我只需要使用响应,这导致了以下结果。
网址已更改为http://localhost:8080/pisr/recoveryDetail/update,删除所有参数并返回404页面。
示例
flash.message = "test"
respond view: "edit", model:[recoverDetailInstance:recoverDetailInstance]
return
所以我的问题
如何抛出服务器端验证错误并将其返回到包含用户输入数据的页面?
答案 0 :(得分:0)
您可以将参数添加回重定向或渲染调用。
redirect (..., params: params)
https://grails.github.io/grails-doc/latest/ref/Controllers/redirect.html
另外,我建议使用服务而不是控制器方法。服务已经是交易性的。如果hasErrors
为真,我也会抛出异常。异常消息或对象可以是您的有效负载返回给用户。
答案 1 :(得分:0)
解决方案是使用渲染而不使用像这样的参数。
if(recoveryDetailInstance.hasErrors()) {
render view: "edit", model:[recoveryDetailInstance:recoveryDetailInstance]
return
}