Our adventure today begins with a seemingly simple task. Upgrading the Grails version for a project. The project was previously on Grails 1.3.7, and we are upgrading/updating/whichever it to 2.0. For a point of reference, here is the setup/environment :
- This a fairly large code base containing many other modules.
- We are using maven
- My IDE of choice is Eclipse Helios (64bit)
- We have a 'local' nexus that proxies stuff for us
Here's an overview of what this blog entails:
- Where are the jars?
- Multiple versions of grails in the same workspace
- Wait, I need to INSTALL grails as well??
- Grails "upgrade" command doesn't work with maven-ized projects
- Trouble re-installing the JAXRS plugin
- Some final thoughts
My first problem will probably be moot for most people now. I could not get the updated jars from our local nexus. It took a little searching and I came across this: Grail 2.0 jars not available through maven central. Which of course explains why our nexus didn't pull it in. The reason this is moot now is that Grails 2.0.1 is available through maven central. So I was able to grab the jars for 2.0.1 fine. But for future reference, if you experience trouble pulling in the jars for new versions of Grails ... check to see if they even exist in your nexus or maven central. If not, you can grab them from Grails' repo: http://repo.grails.org/grails/core.
Issue #2 - Multiple versions of grails in the same workspace.
Since this is a major release of Grails, there are many things that have been changed and therefore our code needs to be changed to work with the Grails. Grails does provide a nice* write-up of how to upgrade your code here: Upgrading From Previous Versions of Grails.
*Since I have only just resolved my issues with getting started, I haven't yet had a chance to follow their write-up, but at a glance, it looks fairly detailed. Whether or not it is detailed enough, I will find out soon.
Because this is potentially going to perturb our code base with all the changes, I am working the upgrade in a branch that contains only this module. Seems like a sound plan, right? At this moment, I have our entire code base open in eclipse and I have this branch open. As soon as I change the POM to reference 2.0.1, and eclipse does its auto build thing to pick up this change ... the bad thing happens. Namely, eclipse exits all by it's lonesome, with no message or anything. Just, one minute it's building ... next minute it's gone.
What happened? Since I'm on a Mac, I opened up the console to take a look at what it says.. and it says a lot:
ar 16 11:31:16 dhc019746 [0x0-0x1f51f5].org.eclipse.eclipse: /Users/frog/Documents/workspaceNew/caffiendFrog-webapp-service-BRANCH/plugins/spring-security-core-18.104.22.168/src/java/grails/plugins/springsecurity/DigestAuthPasswordEncoder.java:88: cannot find symbol Mar 16 11:31:16 dhc019746 [0x0-0x1f51f5].org.eclipse.eclipse: symbol : variable Hex Mar 16 11:31:16 dhc019746 [0x0-0x1f51f5].org.eclipse.eclipse: location: class grails.plugins.springsecurity.DigestAuthPasswordEncoder Mar 16 11:31:16 dhc019746 [0x0-0x1f51f5].org.eclipse.eclipse: return new String(Hex.encode(digest.digest(s.getBytes()))); Mar 16 11:31:16 dhc019746 [0x0-0x1f51f5].org.eclipse.eclipse: ^ Mar 16 11:31:16 dhc019746 [0x0-0x1f51f5].org.eclipse.eclipse: /Users/frog/Documents/workspaceNew/caffiendFrog-webapp-service-BRANCH/plugins/spring-security-core-22.214.171.124/src/java/org/codehaus/groovy/grails/plugins/springsecurity/AbstractFilterInvocationDefinition.java:179: cannot find symbol .... (Lots more errors along the same lines, cannot find symbols) ... Mar 16 11:31:16 dhc019746 [0x0-0x1f51f5].org.eclipse.eclipse: 100 errors Mar 16 11:31:16 dhc019746 [0x0-0x1f51f5].org.eclipse.eclipse: 1 error Mar 16 11:31:16 dhc019746 com.apple.launchd.peruser.503 ([0x0-0x1f51f5].org.eclipse.eclipse): Exited with exit code: 1It took me quite a lot of frustration, many hours, to realize that the problem was that because the two versions of grails is so different, they can not coexist happily, at least not without some tweaking. There is a nice little script available here that does the switching automagically based on what version of grails your project needs. You only need to a little bit of setup, namely setting an environment variable to point to a default version of grails, and to have both versions of grails in the same folder. For example, this is what I did:
- I decided to use 1.3.7 as my default, so I set my $GRAILS_HOME to point to that in my ~/.bashrc file:
- (Listing 2)
- (Listing 3)
frog@hostname:/opt$ ls total 16 drwxr-xr-x 14 frog staff 476 Mar 16 14:37 . drwxrwxr-t 37 root admin 1326 Mar 16 14:20 .. drwxr-xr-x 20 frog admin 680 Nov 2 15:40 grails-1.3.7 drwxr-xr-x@ 21 frog staff 714 Feb 14 16:22 grails-2.0.1 drwx------@ 8 frog staff 272 Mar 16 14:37 grails-docs-1.3.7 drwx------@ 16 frog staff 544 Mar 16 14:36 grails-docs-2.0.1 drwxr-xr-x 3 frog staff 102 Mar 16 14:27 multi-grails
- (Listing 4)
Issue #2a - Wait, I need to INSTALL grails as well??
This is a painful thing for me to admit, and maybe it's obvious in the documentation on grails and I was just trying to do things too quickly and didn't notice. But I completely forgot I had installed grails on my machine. I just assumed it was a plugin like any other. No, no, no. You need to install grails. And by install grails, I mean:
- Download the binary zip of the version you want (http://grails.org/Download)
- C'est Fin.
Issue #3 - Grails "upgrade" command doesn't work with maven-ized projects
Alright, now that I can actually VIEW my code in eclipse, I can start the upgrading process. Grails provides a nice "upgrade" command to make life easier. It is simply "upgrade". To use this command on the command line with the mavenized grails commands:
mvn grails:exec -Dcommand=upgradeSimple, right? Ah, but no. Again, that would be too easy. What you end up with if you try this on the command line is something along the lines of, after lots of downloading of whatever other dependencies your project has:
NOTE: Your application currently expects grails version [2.0.1], this target will upgrade it to Grails 1.3.7 ... WARNING: This target will upgrade an older Grails application to 1.3.7. Are you sure you want to continue? (y, n) y [delete] Deleting directory /Users/frog/Documents/workspaceNew/blog/target/classes [delete] Deleting directory /Users/frog/Documents/workspaceNew/blog/target/resources [delete] Deleting directory /Users/frog/Documents/workspaceNew/blog/target/test-classes [INFO] ------------------------------------------------------------------------ [INFO] BUILD FAILURE [INFO] ------------------------------------------------------------------------ [INFO] Total time: 1:29.344s [INFO] Finished at: Sat Mar 17 19:25:48 EDT 2012 [INFO] Final Memory: 32M/81M [INFO] ------------------------------------------------------------------------ [ERROR] Failed to execute goal org.grails:grails-maven-plugin:1.3.7:exec (default-cli) on project caffiendFrog-webapp-service: Unable to start Grails: java.lang.reflect.InvocationTargetException: : /Users/frog/Documents/workspaceNew/blog/null/src/war not found. -> [Help 1] [ERROR] [ERROR] To see the full stack trace of the errors, re-run Maven with the -e switch. [ERROR] Re-run Maven using the -X switch to enable full debug logging. [ERROR] [ERROR] For more information about the errors and possible solutions, please read the following articles: [ERROR] [Help 1] http://cwiki.apache.org/confluence/display/MAVEN/MojoExecutionException
If you look at the end of line 17 above, you can see it is looking for: /blah/blah/null/src/war. What the heck, why is it looking for null, and where did that even get specified? According to this blog, there is some problem with resolving the environment path and maven and grails. Long story short, it doesn't work. You can't (at this time at least) use the upgrade grails command on a maven-ized project.
So now what? Now you do the upgrade manually. The above mentioned blog entry has instructions as well. Here is what I did, in a little bit more detail. Note that some of the steps I'm not sure are 100% necessary, but I was getting tired of eclipse choking on me.
- Exit eclipse.
- Navigate to the directory of your project
- Using your favorite text editor, manually edit the pom.xml file to change the grails version to 2.0.1. Note that in our case, we have declared a property at the top of the pom.xml that is used in all the grails related dependencies for consistency.
- Using your favorite text editor, manually edit the application.properties to change the app.grails.version to 2.0.1.
- Make a note of all the plugins that are listed in the application.properties file. For example, here are the plugins that our module uses:
- (Listing 7)
plugins.famfamfam=1.0.1 plugins.hibernate=2.0.1 plugins.jaxrs=0.6 plugins.jquery=1.7.1 plugins.jquery-ui=1.8.15 plugins.mail=1.0 plugins.spring-security-core=126.96.36.199 plugins.spring-security-ui=0.2
As I mentioned earlier, it has been a while since we first set this up, so I didn't even remember all the plugins we were using.
- (Listing 8)
mvn grails:exec -Dcommand="uninstall-plugin" -Dargs="famfamfam"
- (Listing 9)
mvn grails:exec -Dcommand="uninstall-plugin" -Dargs="famfamfam"
One of the plugins that we use is jaxrs ...which has a dependency on some restlet.org packages. For some odd reason, when trying to re-install the plugin, I keep getting errors about inability to resolve the dependency. As it turns out, the restlet version that is needed by the plugin is NOT in maven central. To fix this, I added a repository to the pom.xml
Some final thoughts ... for now...
restlet.org restlet.org http://maven.restlet.org
At this point, you should be able to do a mvn clean install on your command line for the project and open both projects up in Eclipse without eclipse having a heart attack and falling over. Keep in mind that your mvn clean install will fail, since you haven't yet changed your code to adhere to the new 2.0.1 ways. This is far as I was able to get to:
[ERROR] Failed to execute goal org.grails:grails-maven-plugin:2.0.1:clean (default-clean) on project eagle-i-webapp-identity-service: Unable to start Grails: java.lang.reflect.InvocationTargetException: Provider for javax.xml.parsers.SAXParserFactory cannot be found -> [Help 1]
Another thing to mention is be wary of messing around with your maven-ized pom.xml file. If I remember correctly, you had to issue a specific command to turn the grails project into a maven-ized one and that the command also created/edited the pom.xml file. One of the motivating factors for us to upgrade is that we have been seeing our CI builds hang for no explicit reason when building this specific project.. Where it hangs varies, but it seems to be hanging while trying to resolve dependencies (another long story, maybe for another time). I tried to do some "cleverness" by cleaning up what I thought were now obsolete/unused dependencies ... which I should have known better than to do.
I hope this has been helpful for someone out there. I tried to document this as best as I could, but it was a long, frustrating process. I may have missed some steps, if I did and it's causing you issues, please let me know and I'll try to help out.