Warbler, A Little Birdie To Introduce Your Rails App To Java
Posted by Nick Sieger Tue, 04 Sep 2007 02:48:40 GMT
This week I was working on integrating the latest JRuby 1.0.1 and Goldspike 1.3 releases into our environment, when my frustration hit a fever pitch.
See, I had always thought that the .war packaging side of Goldspike was a little clunky and un-ruby-like, but I didn’t see a clear path to fixing it. I had heard little complaints about it here and there: the little configuration DSL didn’t give you enough control or wasn’t documented well enough; the fact that it downloads libraries from the internet during assembly (convenient, but not safe or reproducible for production deployments).
Also, in my own opinion it took the wrong approach to packaging Rails in a .war file. It puts the
Rails application directory structure into the root of the .war file where any web server or Java
application server might mistakenly serve up your code as static content. The Java .war file spec has
this special directory called WEB-INF
expressly for the purpose of hiding that stuff away, so why
not use it?
And then, suddenly Goldspike was packaging up my entire Rails application directory, .svn directories and everything. So I set out to fix this once and for all.
And so I present Warbler. A little bird who chirpily steps up to the task of assembling your Rails application into a Java Web Archive (.war). Here, get it:
gem install warbler
And then, in the top directory of your Rails application,
warble
Those two steps are all it takes to make a .war file, including your application and recent versions of JRuby and Goldspike, that’s deployable to your favorite Java application server.
There are a number of points about Warbler worth mentioning.
Does one thing, well
Warbler only packages, and doesn’t care about anything else, like how to dispatch servlet requests to Rails. This will allow for more runtime servlet binding mechanisms to take advantage of Warbler in the future.
Fast and lightweight
50% less code than the Goldspike packaging plugin, yet does the job quickly and efficiently.
Sane defaults
Warbler only packages code that you need to run the application, omitting database migrations and tests. If your application is self-sufficient (no external dependencies), then the out-of-the-box configuration will probably work for you. Public HTML/images/javascript/stylesheets go in the root of the webapp, where Java webservers expect them to be.
Documented, flexible configuration
Need to customize your configuration? Run warble config
and edit config/warble.rb
. All the
options are there, commented and documented.
Need to change out the bundled JRuby/Goldspike versions? warble pluginize
makes a copy of Warbler
in the vendor/plugins
area of your application, allowing you to change the .jar files in the
vendor/plugins/warbler-0.9/lib
directory. Warbler then makes his nest in your project’s list of
rake tasks (as rake -T | grep war
shows)
rake war # Create trunk.war
rake war:app # Copy all application files into the .war
rake war:clean # Clean up the .war file and the staging area
rake war:gems # Unpack all gems into WEB-INF/gems
rake war:jar # Run the jar command to create the .war
rake war:java_libs # Copy all java libraries into the .war
rake war:public # Copy all public HTML files to the root of the .war
rake war:webxml # Generate a web.xml file for the webapp
Warbler even omits himself in the .war file produced when running in plugin mode, since you won’t need him at runtime. It’s the little details that matter.
Give him a try and let me know if it makes your life deploying Rails applications to JRuby on Java appservers easier!
Very nice! However, when when my app boots, it says ” Failed to load Rails: Could not find RubyGem jruby-openssl (>= 0.0.0)”
I tried just a plain-Jane test app, but it can’t find my database module “postgres”.
I think this is because these need extra JRuby-specific gems. How would I load them using warbler?
I can get both of these running using plain JRuby very easily by just using JRuby’s gem installation function. Do I need to copy folders from my local JRuby installation or is there some way to automate it with warbler?
If you have custom gems (including
jruby-openssl
in this case), you’ll have to tell warbler about them.Set up a
config/warble.rb file
withwarble config
, and edit it. There’s a spot to add gems by name there. As long as they’re present in the JRuby installation where you run warbler (or rake) from, they’ll get copied into your .war file.Nice job Nick!
First, with the 0.9 version, I received a
undefined method `exitstatus' for nil:NilClass
error. I think this was due to my (old)rake
version. Upgrading warbler to 0.9.1 withgem update warbler --include-dependencies
solved my problem!Stéphane
When I create a war file and try to run in tomcat, on startup i get a no class found for org/apache/commons/pool/PoolableObjectFactory for SEVERE: Error configuring application listener of class org.jruby.webapp.RailsContextListener. I looked in the WEB-INF\lib directory and see commons-pool-1.3.jar. Does anyone know how to solve this. Thanks
Jeff
@Jeff: Interesting, I haven’t seen that issue yet. Can you send more details to me in an email?
Problem revised: It was not due to rake. The command to unpack the gems did not succeed on my system. But after calling
warble
several times, the specs of the gems were present in the working directory and warble presumed all gems were correctly unpacked and build the war file... and making me think, my updating of rake had “solved” the problem.@Nick: I fixed it my sources. It uses
Gem::Installer
directly to unpack the gem. Should I send you the snippet?When I try installing the gem I get a buffer error: “ERROR: While executing gem ... (Zlib::BufError)”
Anyone else getting this or have any suggestions. I was excited to see this tool after running into no luck with any of the deployment tutorials.
@Stephane: please send it, I’d be happy to incorporate it.
@Joslyn: this is during install? Can you try downloading the gem from the downloads page and run gem install directly on the .gem file? And is this with Ruby or JRuby?
I went ahead and completely reinstalled the latest ruby on my windows box and immediately tried installing warbler both remotely and from a local .gem.
Both sill had the buffer error. I was able to install other gems (such as rails). Any other information you need?
WinXP Pro SP2 Ruby 1.8.5-24
Update: Wasn’t pointing to my JRuby path to install the gem. I feel like an idiot.. :)
I’ll keep you posted on my results. Our state agency is currently a Java shop and it would be great to present them the slight possibility of deploying rails apps on our WebSphere servers. I’d much rather do development using Rails :)
One thing you should mention is that warbler should be installed from JRuby and not regular ruby.
One other thing -- I don’t know if this is a configuration setting or something w/ Jruby itself. When I use ActiveRecord-jdbc to connect to a JNDI datasource, on every page access it gives me a warning that it is closing a connection for me and shaming me for not doing it myself (not to mention throwing a stacktrace).
I’m deploying via JBoss, if that matters.
Here’s an example log message:
@Nick: It´s in your mailbox!
@Jonathan: I installed warbler into my regular ruby installation. Works (now) without problems.