6

我目前正在开发 Grails 应用程序,并且正在使用 Spring Security 和 UI 插件。我已经创建了用户类和我拥有的另一个区域类之间的关系,如下所示:

用户等级:

class User {

    transient springSecurityService

    String username
    String email
    String password
    boolean enabled
    boolean accountExpired
    boolean accountLocked
    boolean passwordExpired

    static belongsTo = [area:Areas]

        .......
}

区域等级:

class Areas {

    String name

    static hasMany = [users:User]

}

从类中可以看出,一个用户可以链接到一个区域,但一个区域可能有很多用户。这一切都很好,当我引导应用程序时,所有数据都会正确添加。但是,当我尝试使用表单创建新用户时出现以下错误:

object references an unsaved transient instance - save the transient instance before flushing: com.website.Area

下面是我用来保存信息的控制器代码:

def save = {    
        def user = lookupUserClass().newInstance(params)

        if (params.password) {
            String salt = saltSource instanceof NullSaltSource ? null : params.username
            user.password = springSecurityUiService.encodePassword(params.password, salt)
        }

        if (!user.save(flush: true)) {
            render view: 'create', model: [user: user, authorityList: sortedRoles()]
            return
        }

        addRoles(user)
        flash.message = "${message(code: 'default.created.message', args: [message(code: 'user.label', default: 'User'), user.id])}"
        redirect action: edit, id: user.id
    }

以下是 GSP 的示例:

<table>
        <tbody>

            <s2ui:textFieldRow name='username' labelCode='user.username.label' bean="${user}"
                            labelCodeDefault='Username' value="${user?.username}"/>

            <s2ui:passwordFieldRow name='password' labelCode='user.password.label' bean="${user}"
                                labelCodeDefault='Password' value="${user?.password}"/>

            <s2ui:textFieldRow name='email' labelCode='user.email.label' bean="${user}"
                                labelCodeDefault='E-Mail' value="${user?.email}"/>

            <s2ui:textFieldRow readonly='yes' name='area.name' labelCode='user.area.label' bean="${user}"
                                labelCodeDefault='Department' value="${area.name}"/>


            <s2ui:checkboxRow name='enabled' labelCode='user.enabled.label' bean="${user}"
                           labelCodeDefault='Enabled' value="${user?.enabled}"/>

            <s2ui:checkboxRow name='accountExpired' labelCode='user.accountExpired.label' bean="${user}"
                           labelCodeDefault='Account Expired' value="${user?.accountExpired}"/>

            <s2ui:checkboxRow name='accountLocked' labelCode='user.accountLocked.label' bean="${user}"
                           labelCodeDefault='Account Locked' value="${user?.accountLocked}"/>

            <s2ui:checkboxRow name='passwordExpired' labelCode='user.passwordExpired.label' bean="${user}"
                           labelCodeDefault='Password Expired' value="${user?.passwordExpired}"/>
        </tbody>
        </table>

我已经对此进行了研究,我相信这是我试图保存 GORM 对象的方式,我可能应该在尝试保存用户之前保存父(区域)。但是,在创建用户之前该区域将始终存在,因此我不需要再次创建该区域,我该如何处理?我也尝试了“withTransaction”功能但收效甚微

如果可能的话,我真的很感激这方面的一些帮助。

谢谢

编辑 ......

我已在 Controller 中将此问题跟踪到此行:

RegistrationCode registrationCode = springSecurityUiService.register(user, command.password, salt)

这对我来说,这个函数正在为用户对象调用一个保存函数,但是在关系到位的情况下,它需要首先保存区域。有谁知道 SpringSecurityUi 可以帮助我解决这个问题?

谢谢

4

3 回答 3

8

您看到的错误消息

object references an unsaved transient instance - save the transient instance before flushing: com.website.Area

...当您尝试从子(即用户)实体保存不存在的父(即区域)实体时发生。正如您所指出的, RegistrationController.register方法中可能会发生错误,如果那是您保存新用户的地方。

RegistrationCode registrationCode = springSecurityUiService.register(user, command.password, salt)

您只需要在注册调用之前使用逻辑更新RegistrationCode.register方法以将区域分配给用户(假设区域已经存在)- 下面是一个简单的示例。

class RegistrationController ..
 ..
def area = Area.findByName(command.areaName) 
def user = lookupUserClass().newInstance(email: command.email, username: command.username,
                accountLocked: true, enabled: true, area: area)
RegistrationCode registrationCode = springSecurityUiService.register(user, command.password, salt)

几点注意事项:

  • 您正在从 gsp 视图传回“area.name”,因此您需要更新/覆盖 RegisterCommand 以包含区域名称
  • 您使用区域“名称”作为标识符安全吗?在您的类定义中,您没有约束来指示“名称”将是唯一的。也许从您的视图中传回区域id 更安全
  • 理想情况下,您应该使用自定义属性编辑器处理区域查找 - 这是一个示例:Grails 命令对象数据绑定

无论如何,希望有帮助。

于 2012-10-23T09:34:36.730 回答
0
class Areas {
    String name
    static belongsTo = [User] 
    static hasMany = [users:User]
}
于 2014-01-08T10:46:44.680 回答
0

尝试将双向依赖转向另一端

class Areas {

    String name
    static belongsTo = [User] 
    static hasMany = [users:User]
}

不要忘记从用户中删除 belongsTo

于 2012-10-23T09:09:02.083 回答