2009-07-17 57 views
2

我用:Db的遷移 - 休眠/ JPA - 就是hbm2ddl - 差異工具

  • EJB3/JPA(休眠)
  • MySQL的5

我必須建立一個系統,以幫助數據庫遷移。試過使用LiquiBase,但它似乎還不夠成熟,但與Hibernate一起使用。

我想什麼做的是:

  • 我有1.0版應用程序在生產
  • 我開發的應用程序的2.0版本和測試
  • 我想更新應用程序數據庫在生產過程中沒有丟失數據

其實我想用新的persistence.xml和生產數據庫能夠生成「三角洲」之間舊數據庫和新數據庫。我希望能夠檢索hbm2ddl處於「更新」模式時執行的SQL代碼。

此SQL代碼將被修改(drop + create = rename等...),以避免丟失可能與hbm2ddl.auto=update發生的數據。

我想這是可能的,因爲Hibernate在部署hbm2ddl.auto=update的新版本時會這樣做。但我希望能夠在與hibernatetool的Ant任務中執行此操作。

我沒有在互聯網上找到關於此的許多信息,所以我想知道是否有人已經在這裏做了類似的事情,可以幫助我。

我已經做了以下內容:

<hibernatetool destdir="${dist}"> 
      <!-- 
      <jdbcconfiguration propertyfile="hibtest.properties"></jdbcconfiguration> 
      --> 
      <jpaconfiguration persistenceunit="server-pu" /> 
      <classpath> 
       <fileset dir="${core.lib.server}" includes="*.jar" /> 
       <fileset dir="${core.lib.runtime}" includes="*.jar" /> 
       <fileset dir="${core.lib.build}" includes="*.jar" /> 
       <pathelement location="${core.class}" /> 
       <pathelement location="${core.etc}" /> 
      </classpath> 
      <hbm2ddl outputfilename="schema-delta.sql" format="true" 
       export="false" update="true" /> 
</hibernatetool> 

我真的不知道該怎麼辦,我是能夠得到創建SQL文件,但我只是想三角洲。 我必須把更新=「真」? 我可以在應用程序服務器之外使用持久性單元的jpaconfig(db設置在Glassfish上的JNDI資源上設置)。 (試過,它只是找不到數據庫)

我也嘗試<jdbcconfiguration propertyfile="hibtest.properties"></jdbcconfiguration>或設置persistence.xml數據庫的屬性,而不是使用JNDI資源。

我得到了以下錯誤:

BUILD FAILED 
/home/slorber/workspace/build/build.xml:899: org.hibernate.exception.JDBCConnectionException: Getting database metadata 
    at org.hibernate.tool.ant.HibernateToolTask.reportException(HibernateToolTask.java:226) 
    at org.hibernate.tool.ant.HibernateToolTask.execute(HibernateToolTask.java:189) 
    at org.apache.tools.ant.UnknownElement.execute(UnknownElement.java:288) 
    at sun.reflect.GeneratedMethodAccessor1.invoke(Unknown Source) 
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25) 
    at java.lang.reflect.Method.invoke(Method.java:597) 
    at org.apache.tools.ant.dispatch.DispatchUtils.execute(DispatchUtils.java:105) 
    at org.apache.tools.ant.Task.perform(Task.java:348) 
    at org.apache.tools.ant.Target.execute(Target.java:357) 
    at org.apache.tools.ant.Target.performTasks(Target.java:385) 
    at org.apache.tools.ant.Project.executeSortedTargets(Project.java:1329) 
    at org.apache.tools.ant.Project.executeTarget(Project.java:1298) 
    at org.apache.tools.ant.helper.DefaultExecutor.executeTargets(DefaultExecutor.java:41) 
    at org.eclipse.ant.internal.ui.antsupport.EclipseDefaultExecutor.executeTargets(EclipseDefaultExecutor.java:32) 
    at org.apache.tools.ant.Project.executeTargets(Project.java:1181) 
    at org.eclipse.ant.internal.ui.antsupport.InternalAntRunner.run(InternalAntRunner.java:423) 
    at org.eclipse.ant.internal.ui.antsupport.InternalAntRunner.main(InternalAntRunner.java:137) 
Caused by: org.hibernate.exception.JDBCConnectionException: Getting database metadata 
    at org.hibernate.exception.SQLStateConverter.convert(SQLStateConverter.java:97) 
    at org.hibernate.cfg.reveng.dialect.AbstractMetaDataDialect.getMetaData(AbstractMetaDataDialect.java:64) 
    at org.hibernate.cfg.reveng.dialect.AbstractMetaDataDialect.caseForSearch(AbstractMetaDataDialect.java:163) 
    at org.hibernate.cfg.reveng.dialect.JDBCMetaDataDialect.getTables(JDBCMetaDataDialect.java:22) 
    at org.hibernate.cfg.reveng.JDBCReader.processTables(JDBCReader.java:476) 
    at org.hibernate.cfg.reveng.JDBCReader.readDatabaseSchema(JDBCReader.java:74) 
    at org.hibernate.cfg.reveng.JDBCReader.readDatabaseSchema(JDBCReader.java:860) 
    at org.hibernate.cfg.JDBCBinder.readDatabaseSchema(JDBCBinder.java:115) 
    at org.hibernate.cfg.JDBCBinder.readFromDatabase(JDBCBinder.java:88) 
    at org.hibernate.cfg.JDBCMetaDataConfiguration.readFromJDBC(JDBCMetaDataConfiguration.java:42) 
    at org.hibernate.tool.ant.JDBCConfigurationTask.doConfiguration(JDBCConfigurationTask.java:81) 
    at org.hibernate.tool.ant.ConfigurationTask.getConfiguration(ConfigurationTask.java:55) 
    at org.hibernate.tool.ant.HibernateToolTask.getConfiguration(HibernateToolTask.java:302) 
    at org.hibernate.tool.ant.Hbm2DDLExporterTask.createExporter(Hbm2DDLExporterTask.java:51) 
    at org.hibernate.tool.ant.ExporterTask.execute(ExporterTask.java:39) 
    at org.hibernate.tool.ant.HibernateToolTask.execute(HibernateToolTask.java:186) 
    ... 15 more 
Caused by: java.sql.SQLException: No suitable driver found for jdbc:mysql:3306//localhost/db 
    at java.sql.DriverManager.getConnection(DriverManager.java:602) 
    at java.sql.DriverManager.getConnection(DriverManager.java:154) 
    at org.hibernate.connection.DriverManagerConnectionProvider.getConnection(DriverManagerConnectionProvider.java:133) 
    at org.hibernate.cfg.reveng.dialect.AbstractMetaDataDialect.getConnection(AbstractMetaDataDialect.java:122) 
    at org.hibernate.cfg.reveng.dialect.AbstractMetaDataDialect.getMetaData(AbstractMetaDataDialect.java:61) 
    ... 29 more 
--- Nested Exception --- 
org.hibernate.exception.JDBCConnectionException: Getting database metadata 
    at org.hibernate.exception.SQLStateConverter.convert(SQLStateConverter.java:97) 
    at org.hibernate.cfg.reveng.dialect.AbstractMetaDataDialect.getMetaData(AbstractMetaDataDialect.java:64) 
    at org.hibernate.cfg.reveng.dialect.AbstractMetaDataDialect.caseForSearch(AbstractMetaDataDialect.java:163) 
    at org.hibernate.cfg.reveng.dialect.JDBCMetaDataDialect.getTables(JDBCMetaDataDialect.java:22) 
    at org.hibernate.cfg.reveng.JDBCReader.processTables(JDBCReader.java:476) 
    at org.hibernate.cfg.reveng.JDBCReader.readDatabaseSchema(JDBCReader.java:74) 
    at org.hibernate.cfg.reveng.JDBCReader.readDatabaseSchema(JDBCReader.java:860) 
    at org.hibernate.cfg.JDBCBinder.readDatabaseSchema(JDBCBinder.java:115) 
    at org.hibernate.cfg.JDBCBinder.readFromDatabase(JDBCBinder.java:88) 
    at org.hibernate.cfg.JDBCMetaDataConfiguration.readFromJDBC(JDBCMetaDataConfiguration.java:42) 
    at org.hibernate.tool.ant.JDBCConfigurationTask.doConfiguration(JDBCConfigurationTask.java:81) 
    at org.hibernate.tool.ant.ConfigurationTask.getConfiguration(ConfigurationTask.java:55) 
    at org.hibernate.tool.ant.HibernateToolTask.getConfiguration(HibernateToolTask.java:302) 
    at org.hibernate.tool.ant.Hbm2DDLExporterTask.createExporter(Hbm2DDLExporterTask.java:51) 
    at org.hibernate.tool.ant.ExporterTask.execute(ExporterTask.java:39) 
    at org.hibernate.tool.ant.HibernateToolTask.execute(HibernateToolTask.java:186) 
    at org.apache.tools.ant.UnknownElement.execute(UnknownElement.java:288) 
    at sun.reflect.GeneratedMethodAccessor1.invoke(Unknown Source) 
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25) 
    at java.lang.reflect.Method.invoke(Method.java:597) 
    at org.apache.tools.ant.dispatch.DispatchUtils.execute(DispatchUtils.java:105) 
    at org.apache.tools.ant.Task.perform(Task.java:348) 
    at org.apache.tools.ant.Target.execute(Target.java:357) 
    at org.apache.tools.ant.Target.performTasks(Target.java:385) 
    at org.apache.tools.ant.Project.executeSortedTargets(Project.java:1329) 
    at org.apache.tools.ant.Project.executeTarget(Project.java:1298) 
    at org.apache.tools.ant.helper.DefaultExecutor.executeTargets(DefaultExecutor.java:41) 
    at org.eclipse.ant.internal.ui.antsupport.EclipseDefaultExecutor.executeTargets(EclipseDefaultExecutor.java:32) 
    at org.apache.tools.ant.Project.executeTargets(Project.java:1181) 
    at org.eclipse.ant.internal.ui.antsupport.InternalAntRunner.run(InternalAntRunner.java:423) 
    at org.eclipse.ant.internal.ui.antsupport.InternalAntRunner.main(InternalAntRunner.java:137) 
Caused by: java.sql.SQLException: No suitable driver found for jdbc:mysql:3306//localhost/db 
    at java.sql.DriverManager.getConnection(DriverManager.java:602) 
    at java.sql.DriverManager.getConnection(DriverManager.java:154) 
    at org.hibernate.connection.DriverManagerConnectionProvider.getConnection(DriverManagerConnectionProvider.java:133) 
    at org.hibernate.cfg.reveng.dialect.AbstractMetaDataDialect.getConnection(AbstractMetaDataDialect.java:122) 
    at org.hibernate.cfg.reveng.dialect.AbstractMetaDataDialect.getMetaData(AbstractMetaDataDialect.java:61) 
    ... 29 more 

我使用MySQL和我的JDBC屬性是:

hibernate.dialect=org.hibernate.dialect.MySQLInnoDBDialect 
hibernate.connection.driver_class=com.mysql.jdbc.Driver 
hibernate.connection.url=jdbc:mysql:3306//localhost/db 
hibernate.connection.username=root 
hibernate.connection.password=root 
hibernate.connection.show_sql=true 

我的JDBC MySQL驅動是${core.lib.server}: mysql-connector-java-5.1.7-bin.jar。 (我也嘗試把類路徑<fileset dir="/home/slorber/workspace/core/lib/server" includes="*.jar" />。)

我也將它添加到Ant(Eclipse插件)的運行配置類路徑。

所以我的問題是:

  • 難道我做錯了什麼事做我想要什麼?
  • 這是你將使用Hibernate框架進行數據庫遷移的方式嗎? (如果不是在手工製作的SQL文件中編寫所有數據庫更改,您還會做什麼?)
+0

有趣,我最閱讀的線程之一,而不是任何+1:D – 2011-11-17 00:25:00

回答

2

Hbm2ddl確實有一個您可以自己調用的API。如果現有的ant任務沒有達到你想要的,你總是可以自己進行hbm2ddl調用,甚至可以通過hibernate傳遞一個重載的jdbc驅動程序來記錄執行語句,而不是實際執行它們或者重載hbm2ddl execute方法。

我也會補充一點,你應該看看使用liquibase和手動創建XML文件的任何差異。您意識到您需要驗證生成的sql是否正確,在開發時一次只創建一個步驟就很容易,然後您知道它是正確的。

我們使用hibernate,而不是使用liquibase hibernate集成,我們更新了hibernate映射,運行我們的集成測試以查看它們失敗,創建必要的liquibase變更集(添加表,添加列,重命名列等)。 ),然後重新運行測試並看到它們通過。逐步建立更新日誌可以和正常的開發節奏一起工作,而不需要區分除了完整性檢查之外的任何事情。