package org.jruby.jubilee; import io.netty.buffer.ByteBuf; import io.netty.buffer.Unpooled; import org.jruby.Ruby; import org.jruby.RubyArray; import org.jruby.RubyClass; import org.jruby.RubyFixnum; import org.jruby.javasupport.JavaEmbedUtils; import org.jruby.jubilee.impl.RackEnvironment; import org.jruby.jubilee.impl.RubyIORackInput; import org.jruby.jubilee.impl.RubyNullIO; import org.jruby.runtime.ThreadContext; import org.jruby.runtime.builtin.IRubyObject; import org.vertx.java.core.Handler; import org.vertx.java.core.Vertx; import org.vertx.java.core.VoidHandler; import org.vertx.java.core.buffer.Buffer; import org.vertx.java.core.http.HttpServerRequest; import org.vertx.java.core.impl.DefaultVertx; import java.io.IOException; import java.util.concurrent.atomic.AtomicBoolean; /** * Created with IntelliJ IDEA. * User: isaiah * Date: 11/29/12 * Time: 5:40 PM */ public class RackApplication { private IRubyObject app; private boolean ssl; private Ruby runtime; private DefaultVertx vertx; private RubyClass rackIOInputClass; private RubyClass httpServerResponseClass; private RubyArray rackVersion; private RubyNullIO nullio; private RackEnvironment rackEnv; public RackApplication(Vertx vertx, ThreadContext context, IRubyObject app, boolean ssl) throws IOException { this.app = app; this.ssl = ssl; this.vertx = (DefaultVertx) vertx; this.runtime = context.runtime; this.rackVersion = RubyArray.newArrayLight(runtime, RubyFixnum.one(runtime), RubyFixnum.four(runtime)); // Memorize the ruby classes this.rackIOInputClass = (RubyClass) runtime.getClassFromPath("Jubilee::IORackInput"); this.httpServerResponseClass = (RubyClass) runtime.getClassFromPath("Jubilee::HttpServerResponse"); this.nullio = new RubyNullIO(runtime, (RubyClass) runtime.getClassFromPath("Jubilee::NullIO")); this.rackEnv = new RackEnvironment(runtime); } public void call(final HttpServerRequest request) { // String te = request.headers().get(Const.Vertx.TRANSFER_ENCODING); // String contentLength; final RackInput input; // This should be handled by Netty (expose a contentLength methods via HttpRequest, // it set the empty content flag when passing the header // if ((te != null && te.equals("chunked")) || // ((contentLength = request.headers().get(Const.Vertx.CONTENT_LENGTH)) != null && !contentLength.equals("0"))) { final ByteBuf bodyBuf = Unpooled.buffer(0, Integer.MAX_VALUE); final AtomicBoolean eof = new AtomicBoolean(false); input = new RubyIORackInput(runtime, rackIOInputClass, request, bodyBuf, eof); request.dataHandler(new Handler() { @Override public void handle(Buffer buffer) { bodyBuf.writeBytes(buffer.getByteBuf()); } }); request.endHandler(new VoidHandler() { @Override protected void handle() { eof.set(true); } }); // } else { // input = nullio; // } Runnable task = new Runnable() { @Override public void run() { // This is a different context, do NOT replace runtime.getCurrentContext() // IRubyObject result = app.callMethod(runtime.getCurrentContext(), "call", env.getEnv()); try { IRubyObject result = app.callMethod(runtime.getCurrentContext(), "call", rackEnv.getEnv(request, input, ssl)); RackResponse response = (RackResponse) JavaEmbedUtils.rubyToJava(runtime, result, RackResponse.class); RubyHttpServerResponse resp = new RubyHttpServerResponse(runtime, httpServerResponseClass, request.response()); response.respond(resp); } catch (IOException e) { // noop } } }; vertx.startInBackground(task, false); } }