How to run Ruby/Rails with Resque app in a JRuby/Tomcat deploymet with no JRuby installed and using only Java?

1.1k Views Asked by At

I have a JRuby/Rails web application. I use rbenv and JRuby to test locally (using "rails server"), and it works. I also distribute the production version as a WAR file and run in Tomcat, and that machine has no JRuby installation. It works. I now need to be able to use Resque and Redis to process long-running jobs. I installed Resque/redis locally and ran Resque from the command line as

linux> "QUEUE=* bundle exec rake environment resque:work

It works. I created a little worker class (FooClass) which just prints out the parameter, is called by Resque, and I am able to enqueue requests by running

irb(main):041:0>Resque.enqueue(FooWorker, [1,"4"])

in the Rails console, and the Resque eventually processes the request and prints [1,"4"]

I would like to be able to run this Resque Rake task in the Tomcat/Java environment (which does not have JRuby installed), and I have 3 options to run the Resque Rake Task.

  1. Run it within the Tomcat, but in a separate Java thread. I prefer not to do this because I want to be able to kill the worker process and restart it.
  2. Run it from the command line, and have that command called by etc/init.d at startup.
  3. Run it in a separate Tomcat from the web app.

My warbler is configured such that the following files are present in the application WAR file, and get exploded to webapps/WEB-INF out to webapps/abc/WEB-INF when Tomcat starts the app.

Gemfile
Rakefile
web.xml
lib/jruby-core-1.6.4.jar
lib/jruby-rack-1.0.10.jar
lib/jruby-stdlib-1.6.4.jar
lib/tasks/abc.rake
lib/tasks/resque.rake
gems/gems/bundler-1.0.21
gems/gems/warbler-1.3.2
gems/gems/rails-3.0.10
(other gem files in gems/gems)
config/warble.rb

The file config/warble.rb looks like this:

Warble::Config.new do |config|
    config.dirs=%w{app config lib lob vendor tmp}
    config.includes=FileList["./Rakefile"]
    config.webxml.jruby.compat.version = "1.9"      
end

The file lib/tasks/resque.rake has

require "resque/tasks"
task "resque:setup" => :environment
task "resque:work" => :environment

My gemfile includes the following lines:

source 'http://rubygems.org'
gem 'bundler', '1.0.21'
gem 'rails', '3.0.10'
gem 'rake', '0.8.7'
gem 'resque'

Searching on the web yielded the following ways of running Rake tasks (from within WEB-INF/) without using JRuby:

linux>  java -cp lib/jruby-core-1.6.4.jar -S rake

The result was

Unrecognized option: -S
Could not create the Java virtual machine

Just for fun, I tried

linux> java -jar lib/jruby-core-1.6.4.jar -S rake

The result was

 jruby: No such file or directory -- rake (LoadError)

I then tried downloading jruby-complete-1.6.4 to that directory and I ran

linux>  java -jar lib/jruby-complete-1.6.4.jar  -S rake -T
(in /WEB-INF)
rake aborted!
no such file to load -- bundler/setup
/WEB-INF/Rakefile:4:in `(root)'

At this point I am at a complete loss. How can I run my desired Resque Rake task in the Java/Tomcat or just Java environment without installing JRuby on that server?

2

There are 2 best solutions below

0
On

Alternatively you can run Resque with your application in daemon Threads (instead of a separate rake process) using https://github.com/kares/jruby-rack-worker.

Simply configure Warbler (or your Gemfile) to include the jruby-rack-worker gem (alternatively you can download the .jar and tell Warbler it's to be placed in your WEB-INF/lib dir) and setup the Resque in the deployment descriptor. With Warbler create the config/web.xml.erb file and put there the following code :

<!DOCTYPE web-app PUBLIC
  "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
  "http://java.sun.com/dtd/web-app_2_3.dtd">
<web-app>
<% webxml.context_params.each do |k,v| %>
  <context-param>
    <param-name><%= k %></param-name>
    <param-value><%= v %></param-value>
  </context-param>
<% end %>

  <filter>
    <filter-name>RackFilter</filter-name>
    <filter-class>org.jruby.rack.RackFilter</filter-class>
  </filter>
  <filter-mapping>
    <filter-name>RackFilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>

  <listener>
    <listener-class><%= webxml.servlet_context_listener %></listener-class>
  </listener>

<% if webxml.jndi then [webxml.jndi].flatten.each do |jndi| %>
  <resource-ref>
    <res-ref-name><%= jndi %></res-ref-name>
    <res-type>javax.sql.DataSource</res-type>
    <res-auth>Container</res-auth>
  </resource-ref>
<% end; end %>

  <!-- jruby-rack-worker setup using the built-in libraries support : -->

  <context-param>
    <param-name>jruby.worker</param-name>
    <param-value>resque</param-value>
  </context-param>

  <listener>
    <listener-class>org.kares.jruby.rack.WorkerContextListener</listener-class>
  </listener>

</web-app>
0
On

So, the basic thing you need to do for this is to write a small shell script and run it from the WEB-INF directory of the unpacked war file.

  1. Ensure you have this script as well as Rakefile and any other supporting scripts (e.g., db/migrations) included in the war file.
  2. Script should look like the following:
#!/bin/bash
#
# Put in WEB-INF/; call this rake.sh or whatever you prefer.

dir=$(dirname $0)

# Put all jar files in the classpath   
for jar in $dir/*; do
    CLASSPATH="$jar:$CLASSPATH"
done

# You might need to adjust Java memory settings; currently JRuby sets 512m by default
JAVA_OPTS=-Xmx512m

# Set up GEM_HOME and GEM_PATH
GEM_HOME=$dir/gems
GEM_PATH=$GEM_HOME

export CLASSPATH GEM_HOME GEM_PATH

java $JAVA_OPTS org.jruby.Main -S rake $@