NAME
Continuity - Abstract away statelessness of HTTP using continuations, for stateful Web applications
SYNOPSIS
#!/usr/bin/perl
use strict;
use Continuity;
my $server = new Continuity;
$server->loop;
sub main {
my $request = shift;
$request->print("Your name: <form><input type=text name=name></form>");
$request->next; # this waits for the form to be submitted!
my $name = $request->param('name');
$request->print("Hello $name!");
}
DESCRIPTION
This is BETA software, and feedback/code is welcomed.
Continuity is a library to simplify web applications. Each session is written and runs as a persistant application, and is able to request additional input at any time without exiting. This is significantly different from the traditional CGI model of web applications in which a program is restarted for each new request.
The program is passed a $request variable which holds the request (including any form data) sent from the browser. In concept, this is a lot like a $cgi
object from CGI.pm with one very very significant difference. At any point in the code you can call $request->next. Your program will then suspend, waiting for the next request in the session. Since the program doesn't actually halt, all state is preserved, including lexicals -- getting input from the browser is then similar to doing $line=<>
in a command-line application.
GETTING STARTED
First, check out the small demo applications in the eg/ directory of the distribution. Sample code there rages from simple counters to more complex multi-user ajax applications.
Declare all your globals, then declare and create your server. Parameters to the server will determine how sessions are tracked, what ports it listens on, what will be served as static content, and things of that nature. Then call the loop
method of the server, which will get things going (and never exits).
use Continuity;
my $server = Continuity->new( port => 8080 );
$server->loop;
Continuity must have a starting point for creating a new instance of your application. The default is to \&::main
, which is passed the $request
handle. See the Continuity::Request documentation for details on the methods available from the $request
object beyond this introduction.
sub main {
my $request = shift;
# ...
}
Outputting to the client (that is, sending text to the browser) is done by calling the $request->print(...)
method, rather than the plain print
used in CGI.pm applications.
$request->print("Hello, guvne'<br>");
$request->print("'ow ya been?");
HTTP query parameters (both GET and POST) are also gotten through the $request
handle, by calling $p = $request->param('p')
.
# If they go to http://webapp/?x=7
my $input = $request->param('x');
# now $input is 7
Once you have output your HTML, call $request->next
to wait for the next response from the client browser. While waiting other sessions will handle other requests, allowing the single process to handle many simultaneous sessions.
$request->print("Name: <form><input type=text name=n></form>");
$request->next; # <-- this is where we suspend execution
my $name = $request->param('n'); # <-- start here once they submit
Anything declared lexically (using my) inside of main
is private to the session, and anything you make global is available to all sessions. When main
returns the session is terminated, so that another request from the same client will get a new session. Only one continuation is ever executing at a given time, so there is no immediate need to worry about locking shared global variables when modifying them.
ADVANCED USAGE
Merely using the above code can completely change the way you think about web application infrastructure. But why stop there? Here are a few more things to ponder.
Since Continuity is based on Coro, we also get to use Coro::Event. This means that you can set timers to wake a continuation up after a while, or you can have inner-continuation signaling by watch-events on shared variables.
For AJAX applications, we've found it handy to give each user multiple sessions. In the chat-ajax-push demo each user gets a session for sending messages, and a session for receiving them. The receiving session uses a long-running request (aka COMET) and watches the globally shared chat message log. When a new message is put into the log, it pushes to all of the ajax listeners.
Don't forget about those pretty little lexicals you have at your disposal. Taking a hint from the Seaside folks, instead of regular links you could have callbacks that trigger a anonymous subs. Your code could easily look like:
my $x;
$link1 = gen_link('This is a link to stuff', sub { $x = 7 });
$link2 = gen_link('This is another link', sub { $x = 42 });
$request->print($link1, $link2);
$request->next;
process_links($request);
# Now use $x
To scale a Continuity-based application beyond a single process you need to investigate the keywords "session affinity". The Seaside folks have a few articles on various experiments they've done for scaling, see the wiki for links and ideas. Note, however, that premature optimization is evil. We shouldn't even be talking about this.
EXTENDING AND CUSTOMIZING
This library is designed to be extensible but have good defaults. There are two important components which you can extend or replace.
The Adaptor, such as the default Continuity::Adapt::HttpDaemon, actually makes the HTTP connections with the client web broswer. If you want to use FastCGI or even a non-HTTP protocol, then you will use or create an adaptor.
The Mapper, such as the default Continuity::Mapper, identifies incoming requests from The Adaptor and maps them to instances of your program. In other words, Mappers keep track of sessions, figuring out which requests belong to which session. The default mapper can identify sessions based on any combination of cookie, ip address, and URL path. Override The Mapper to create alternative session identification and management.
METHODS
$server = Continuity->new(...)
The Continuity
object wires together an adapter and a mapper. Creating the Continuity
object gives you the defaults wired together, or if user-supplied instances are provided, it wires those together.
Arguments:
callback
-- coderef of the main application to run persistantly for each unique visitor -- defaults to\&::main
adapter
-- defaults to an instance ofContinuity::Adapt::HttpDaemon
mapper
-- defaults to an instance ofContinuity::Mapper
docroot
-- defaults to.
staticp
-- defaults tosub { $_[0]->url =~ m/\.(jpg|jpeg|gif|png|css|ico|js)$/ }
, used to indicate whether any request is for static contentdebug
-- defaults to4
at the moment ;)
Arguments passed to the default adaptor:
port
-- the port on which to listenno_content_type
-- defaults to 0, set to 1 to disable theContent-Type: text/html
header and similar headers
Arguments passed to the default mapper:
query_session
-- set to the name of a query variable for session tracking (defaults to undef)assign_session_id
-- coderef of routine to custom generate session id numbers (defaults to a simple random string generator)ip_session
-- set to true to enable ip-addresses for session tracking (defaults to false)path_session
-- set to true to use URL path for session tracking (defaults to false)implicit_first_next
-- set to false to get an empty first request to the main callback (defaults to true)
$server->loop()
Calls Coro::Event::loop and sets up session reaping. This never returns!
Internal Structure
For the curious or the brave, here is an ASCII diagram of how the pieces fit:
+---------+ +---------+ +--------+
| Browser | <--> | Adaptor | --> | Mapper |
+---------+ +---------+ +--------+
^ |
| |
+---------------------+ |
| +-------------------+---------+----------+
| | | |
| V V V
| +---------+ +---------+ +---------+
| | Session | | Session | | Session |
| | Request | | Request | | Request |
| | Queue | | Queue | | Queue |
| | | | | | | | | |
| | V | | V | | V |
| +---------+ +---------+ +---------+
| | | |
| V V V
| +-----+ +------+ +-----+ +------+ +-----+ +------+
| | Cur |<->| Your | | Cur |<->| Your | | Cur |<->| Your |
| | Req | | Code | | Req | | Code | | Req | | Code |
| +-----+ +------+ +-----+ +------+ +-----+ +------+
| | | |
| V V V
+-----+--------------------+--------------------+
** "Cur Req" == "Current Request"
Basically, the Adaptor accepts requests from the browser, hands them off to the Mapper, which then queues them into the correct session queue (or creates a new queue).
When Your Code calls "$request->next" the Current Request overwrites itself with the next item in the queue (or waits until there is one).
Most of the time you will have pretty empty queues -- they are mostly there for safety, in case you have a lot of incoming requests and running sessions.
For further internal development documentation, please see the wiki or email me.
SEE ALSO
See the Wiki for development information, more waxing philosophic, and links to similar technologies such as http://seaside.st/.
Website/Wiki: http://continuity.tlt42.org/
Continuity::Request, Continuity::Mapper, Continuity::Adapt::HttpDaemon, Coro
AUTHOR
Brock Wilcox <awwaiid@thelackthereof.org> - http://thelackthereof.org/
Scott Walters <scott@slowass.net> - http://slowass.net/
Special thanks to Marc Lehmann for creating (and maintaining) Coro
COPYRIGHT
Copyright (c) 2004-2007 Brock Wilcox <awwaiid@thelackthereof.org>. All
rights reserved. This program is free software; you can redistribute it
and/or modify it under the same terms as Perl itself.