Liquibase Community Forum
2010-11-15 12:37:43 UTC
A new topic, 'ClassNotFoundException from Ant task 'updateDatabase' ', has been made on a board you are watching.
You can see it at
http://liquibase.org/forum/index.php?topic=785.new#new
The text of the topic is shown below:
ClassNotFoundException from Ant task 'updateDatabase' if its classpathref differs from classpath used to load the Ant tasks:
Hello Nathan,
if I want to load my own database implementation (which extends H2Database) from Ant task 'updateDatabase' I get a ClassNotFoundException.
Running the Debugger I see that my own database implementation is loaded and my constructor is properly executed,
but then there is a ClassCastException when my class is cast to class 'liquibase.database.Database'.
The following patch solves the problem:
Index: src/main/java/liquibase/integration/ant/BaseLiquibaseTask.java
===================================================================
--- src/main/java/liquibase/integration/ant/BaseLiquibaseTask.java (revision 1837)
+++ src/main/java/liquibase/integration/ant/BaseLiquibaseTask.java (working copy)
@@ -191,7 +191,7 @@
URLClassLoader loader = AccessController.doPrivileged(new PrivilegedAction<URLClassLoader>() {
public URLClassLoader run() {
- return new URLClassLoader(taskClassPath.toArray(new URL[taskClassPath.size()]));
+ return new URLClassLoader(taskClassPath.toArray(new URL[taskClassPath.size()]), Database.class.getClassLoader());
}
});
I assume that the
try ... catch (ClassCastException e) { //fails in Ant in particular
which follows seven lines below can be omitted
if the patch is applied.
Some background info:
---------------------
With
<taskdef resource="liquibasetasks.properties" classpathref="classpathWithLiquibase"/>
I load all the Ant task definitions.
The 'classpathWithLiquibase' does not contain my own database implementation.
If I call the Ant task 'updateDatabase' I set the classpathref to an extended classpath,
which also contains my own database implementation.
But in spite of setting the classpathref correctly I get the ClassNotFoundException which is really a ClassCastException thrown before.
If I load the Ant task definition with the extended classpath (which also contains my own database implementation)
there is no problem - I don't get the ClassNotFoundException.
I assume that without the patch the following takes place:
When the URLClassLoader is loading my own database implementation it must load the base class 'Database' first.
If it loads this class by itself, this class may be incompatible to the already loaded Database class, because that class is loaded by another ClassLoader.
(Still I am using RC6. I have seen RC7 only now, sorry. But I think there is no change which concerns the patch above.)
Best regards,
Markus Müller
Unsubscribe to new topics from this board by clicking here: http://liquibase.org/forum/index.php?action=notifyboard;board=1.0
Regards,
The Liquibase Community Forum Team.
You can see it at
http://liquibase.org/forum/index.php?topic=785.new#new
The text of the topic is shown below:
ClassNotFoundException from Ant task 'updateDatabase' if its classpathref differs from classpath used to load the Ant tasks:
Hello Nathan,
if I want to load my own database implementation (which extends H2Database) from Ant task 'updateDatabase' I get a ClassNotFoundException.
Running the Debugger I see that my own database implementation is loaded and my constructor is properly executed,
but then there is a ClassCastException when my class is cast to class 'liquibase.database.Database'.
The following patch solves the problem:
Index: src/main/java/liquibase/integration/ant/BaseLiquibaseTask.java
===================================================================
--- src/main/java/liquibase/integration/ant/BaseLiquibaseTask.java (revision 1837)
+++ src/main/java/liquibase/integration/ant/BaseLiquibaseTask.java (working copy)
@@ -191,7 +191,7 @@
URLClassLoader loader = AccessController.doPrivileged(new PrivilegedAction<URLClassLoader>() {
public URLClassLoader run() {
- return new URLClassLoader(taskClassPath.toArray(new URL[taskClassPath.size()]));
+ return new URLClassLoader(taskClassPath.toArray(new URL[taskClassPath.size()]), Database.class.getClassLoader());
}
});
I assume that the
try ... catch (ClassCastException e) { //fails in Ant in particular
which follows seven lines below can be omitted
if the patch is applied.
Some background info:
---------------------
With
<taskdef resource="liquibasetasks.properties" classpathref="classpathWithLiquibase"/>
I load all the Ant task definitions.
The 'classpathWithLiquibase' does not contain my own database implementation.
If I call the Ant task 'updateDatabase' I set the classpathref to an extended classpath,
which also contains my own database implementation.
But in spite of setting the classpathref correctly I get the ClassNotFoundException which is really a ClassCastException thrown before.
If I load the Ant task definition with the extended classpath (which also contains my own database implementation)
there is no problem - I don't get the ClassNotFoundException.
I assume that without the patch the following takes place:
When the URLClassLoader is loading my own database implementation it must load the base class 'Database' first.
If it loads this class by itself, this class may be incompatible to the already loaded Database class, because that class is loaded by another ClassLoader.
(Still I am using RC6. I have seen RC7 only now, sorry. But I think there is no change which concerns the patch above.)
Best regards,
Markus Müller
Unsubscribe to new topics from this board by clicking here: http://liquibase.org/forum/index.php?action=notifyboard;board=1.0
Regards,
The Liquibase Community Forum Team.