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 - 动态引用默认数据源的主要内容,如果未能解决你的问题,请参考以下文章
ClassNotFoundException: SimpleGrantedAuthority - Grails 2.4.2 和 Spring Security