resin tutorial Custom Protocol Handling

This tutorial shows the usage of the Resin server architecture to handle a custom protocol. Resin handles the TCP connections, multi-threading, and the request object pooling. The application implements a class that reads from a stream and writes to a stream.

Professor Trelawny once got a student to make a Magic8Ball, used for prophecy. Originally it was used with a simple web interface. Now Trelawny wants to provide a protocol server on the Hogwart's public web server. The protocol is at the same level as or , it sit's directly on top of TCP/IP.

The orignal Magic8Ball class, it knows nothing about the code that uses it. The implementation of that Resin uses to create a Request object. The implementation of interface that represents a single request. protocol-specific parsing class to parse the ReadStream and determine commands. protocol-specific abstract base class for commands. protocol-specific Command to set the prophet to use for the magic 8 ball. protocol-specific Command to ask the magic 8 ball for a prophecy. Ant build file

Implementation of a protocol at the level that is at normally requires a large amount of bug-prone code, such as code for handling tcp/ip connections, multi-threading, and request object reuse. Resin provides a simple way to handle protocols without requiring the implementation of that code.

Applications take advantage of Resin's highly tested and configurable server architecture and only need to implement some simple objects. Two classes are required, one which extends the abstract class and one which implements the interface

.

A custom class derived from is a factory that produces custom Request objects.

The Protocol passes a object to the Request object it creates. The Request object uses the to obtain a read stream and a write stream.

The example in this tutorial is simple, it's main purpose is to override the createRequest method to return an instance of the request handling object.

... /** * Create a Magic8BallRequest object for the new thread. */ public ServerRequest createRequest(Connection conn) { return new Magic8BallRequest(this, conn); } ...

It also stores and returns the protocol name.

private String _protocolName = "magic8ball"; ... /** * Return the protocol name. */ public String getProtocolName() { return _protocolName; } /** * Set the protocol name. */ public void setProtocolName(String name) { _protocolName = name; }

is the interface used for implementing the class that handles a request. This is where the bulk of the work is done.

The method handleRequest() is called for each request. Implementations should not assume that a new request object is created for each new connection from a client. Resin may reuse a request object once it has finished handling a request. It is guaranteed, however, that the handleRequest() method of a given instance will only be handling a single request at a time.

This means that any member variables must be initialized at the beginning of the code in handleRequest().

The first step in handleRequest() is usually to obtain a and a from the Connection that was stored with the constructor. These streams are used to read raw data from the client and write raw data to the client.

This tutorial then goes on to initialize a parser and perform appropriate commands, depending on what is submitted by the client. The readStream is used to determine what to do, and the writeStream is used to send the response. The implementation here depends on the protocol that is being supported.

... /** * Handle a new connection. The controlling Server may call * handleRequest again after the connection completes, so the * implementation must initialize any variables for each connection. */ public boolean handleRequest() throws IOException { ReadStream readStream = _conn.getReadStream(); WriteStream writeStream = _conn.getWriteStream(); try { _parser.init(readStream); ... if (error != null) { writeStream.print("ERROR: "); writeStream.println(_parser.getError()); break; } else if (result != null) { writeStream.print("RESULT: "); writeStream.println(result); } ... } catch (Throwable e) { log.log(Level.WARNING, e.toString(), e); } return false; }

The protocol is handled at the server level in the Resin Environment hierarchy. This means that the code has to be available to the appropriate classloader. A jar with your code in it can be placed in $RESIN_HOME/lib/, for example $RESIN_HOME/lib/magic8ball.jar.

The use of the Protocol class is configured with and .

<server> ... <!-- The magic8ball port --> <port id='' host='*' port='8888'> <protocol resin:type="example.Magic8BallProtocol"/> </port>

This tutorial has to be deployed by hand in your installation of Resin.

The filesystem location of this tutorial is at the top of this page.

<server> ... <!-- The magic8ball port --> <port id='' host='*' port='8888'> <protocol resin:type="example.Magic8BallProtocol"/> </port>

To run the demo, you can find the build/magic8ball.jar file and copy it to $RESIN_HOME/lib/.

An ant build file is provided. Calling "ant deploy" will build the jar and deploy it in $RESIN_HOME/lib/magic8ball.jar

Start Resin, or if it is already running wait for it to notice the new library and restart. The default Resin configuration will notice the new jar file (eventually) and restart itself. If you are impatient you can stop and start the server yourself.

Starting Resin on Tue, 23 Sep 2003 15:40:16 -0500 (GMT-05:00) [15:40:17.920] Loaded Socket JNI library. [15:40:38.871] http listening to *:8080 [15:40:59.831] magic8ball listening to *:8888 ...

The simplest way to test a protocol like the is to use telnet.

$ Trying 127.0.0.1... Connected to localhost. Escape character is '^]'. RESULT: trelawney says "The future for you is very dire" RESULT: trelawney says "You must not leave the room" RESULT: prophet set to `classic' RESULT: classic says "It is certain" RESULT: prophet set to `kingwu' RESULT: kingwu says "THE CREATIVE works sublime success, Furthering through perseverance." RESULT: prophet set to `xxxxx' RESULT: xxxxx says "a false prophet is never wise" ERROR: Unknown command `QUIT' Connection closed by foreign host.

The and classes are of course Resin specific. The and are as well.

If portability is a concern, implement a class that is similar in functionality to . Code that class to get passed a and a . A bare-bones Resin-specific Protocol and ServerRequest can be used as a thin wrapper to your custom class, and the Resin's ReadStream and WriteStream can be passed as the InputStream and OutputStream. When porting to another application server (if they provide this kind of functionality) you can code a thin-wrapper specific to that app-server that uses your custom Request class.