Grails 2.4.2 - 动态引用默认数据源

Posted

技术标签:

【中文标题】Grails 2.4.2 - 动态引用默认数据源【英文标题】:Grails 2.4.2 - Dynamically referencing default datasource 【发布时间】:2014-08-23 23:43:30 【问题描述】:

这个问题已经部分回答了here,但是动态引用默认数据源仍然存在问题。

我正在开发一个内部应用程序,它允许开发人员修改我们的多租户应用程序之一的配置设置,并将这些设置从开发人员推送到测试、暂存和生产。 其中每一个都有自己的数据源,并且 Grails 应用程序将安装在每个开发人员的计算机上。

本地数据源会是默认的,然后dataSource_testing、dataSource_staging等会引用相应的环境。

我可以通过以下方式动态引用远程数据源:

def setting = Setting."$params.environmnet".get(id)

但是,如果 params.environment 引用默认数据源,则此代码不再有效。 在设置域类的文档中,它指出:

如果域类使用默认数据源和一个或多个其他数据源,请使用特殊名称“DEFAULT”来指示默认数据源

见the documentation

这看起来在定义您的域类适用于哪些数据源时有效,但无法引用您的域类,例如:def setting = Setting.DEFAULT.get(id)。 您会收到一条错误消息:

没有这样的属性:类的默认值...

我真的真的不想重新设计应用程序以便不真正使用默认数据源,然后创建一个新的dataSource_local 数据源。如果我能避免这种情况,那就太好了,因为这意味着更新仅适用于本地数据源的代码堆栈。

所以我的问题...有没有办法动态引用默认数据源?

【问题讨论】:

【参考方案1】:

目前不支持,但您可以通过一些技巧来添加缺少的元方法 (getDEFAULT)。

如果您认为这是一项重要功能,您可以在 Grails Jira 上提出功能请求问题。通过添加一个单独的方法来为给定的数据源查找所谓的 GormStaticApi 实例来支持静态访问 (CompileStatic) 可能是有意义的。请向 Grails Jira 添加功能请求,以便解释您的用例。

现有实现跳过默认数据源:

Hibernate4 的逻辑: https://github.com/grails/grails-data-mapping/blob/f9da9fe/grails-datastore-gorm-hibernate4/src/main/groovy/org/codehaus/groovy/grails/orm/hibernate/cfg/HibernateUtils.groovy#L122-L125 和休眠3: https://github.com/grails/grails-data-mapping/blob/f9da9fe/grails-datastore-gorm-hibernate/src/main/groovy/org/codehaus/groovy/grails/orm/hibernate/cfg/HibernateUtils.groovy#L121-L124

解决方法是将以下这个类保存在 grails-app/conf/WorkaroundsBootStrap 中:

import org.codehaus.groovy.grails.commons.DomainClassArtefactHandler
import org.codehaus.groovy.grails.commons.GrailsApplication
import org.codehaus.groovy.grails.commons.GrailsClass
import org.codehaus.groovy.grails.commons.GrailsDomainClassProperty
import org.codehaus.groovy.grails.orm.hibernate.cfg.GrailsHibernateUtil
import org.codehaus.groovy.grails.orm.hibernate.cfg.HibernateUtils

class WorkaroundsBootStrap 
    GrailsApplication grailsApplication
    def dataSource
    def transactionManager
    def hibernateDatastore

    def init =  servletContext ->
        def datasourceName = GrailsDomainClassProperty.DEFAULT_DATA_SOURCE
        for(GrailsClass grailsClass in grailsApplication.getArtefacts(DomainClassArtefactHandler.TYPE)) 
            def dc = grailsClass
            if (GrailsHibernateUtil.isMappedWithHibernate(dc) && GrailsHibernateUtil.usesDatasource(dc, datasourceName)) 
                HibernateUtils.registerNamespaceMethods dc, hibernateDatastore, datasourceName, transactionManager, grailsApplication
            
        
    

【讨论】:

感谢您的回答 - 不幸的是,对我来说这有点晚了,我刚刚重写了必要的代码。但是,我肯定会在未来使用它。我认为一个优雅的整体特性是能够将数据源作为参数传递给 GORM 方法。

以上是关于Grails 2.4.2 - 动态引用默认数据源的主要内容,如果未能解决你的问题,请参考以下文章

GRAILS / GORM:动态多重连接

在 grails 中,我如何获得对所有当前会话的引用?

ClassNotFoundException: SimpleGrantedAuthority - Grails 2.4.2 和 Spring Security

动态连接到 grails 中的数据库

Grails 应用程序似乎持有对陈旧数据库连接的引用

Grails MongoDB 插件:嵌入式集合与引用