NAME
Contize - Help an object be a continuation thingie (suspendable)
SYNOPSIS
#!/usr/bin/perl
package WebGuess;
use strict;
use Contize;
sub new {
my $self = {};
bless $self;
$self = new Contize($self);
return $self;
}
sub setNumber {
my $self = shift;
$self->{number} = int(rand(100)) + 1;
}
sub display {
my ($self, $content) = @_;
print $content;
$self->suspend();
}
sub getNum {
my $self = shift;
$self->display(<<"END");
<form method=POST">
Enter Guess: <input name="num">
<input type=submit value="Guess">
<input type=submit name="done" value="Done">
<br>
</form>
END
return $::q->param('num');
}
# ***********************************
# **** THIS IS THE IMPORTANT BIT ****
# ***********************************
sub run {
my $self = shift;
$self->setNumber();
my $guess;
my $tries = 0;
print
"Hi! I'm thinking of a number from 1 to 100...
can you guess it?<br>\n";
do {
$tries++;
$guess = $self->getNum();
print "It is smaller than $guess.<br>\n" if($guess > $self->{number});
print "It is bigger than $guess.<br>\n" if($guess < $self->{number});
} until ($guess == $self->{number});
print "You got it! My number was in fact $self->{number}.<br>\n";
print "It took you $tries tries.<br>\n";
}
package Main;
use strict;
use CGI;
use CGI::Session;
use Data::Dumper;
# Set up the CGI session and print the header
$::q = new CGI();
my $session = new CGI::Session(undef, $::q, {Directory=>'/tmp'});
print $session->header();
# If there is a guess object in the session use it, otherwise create a new
# WebGuess object and Contize it.
my $g = $session->param('guess') || new WebGuess();
# Fix stuff -- most importantly the Data::Dumper version of the object doesn't
# get recreated correctly (I don't know why)... so to work around it I re-eval
# the thing. And we must reset the callstack and the callcount.
my $VAR1;
eval(Dumper($g));
$g = $VAR1;
$g->resume();
# Add the WebGuess object to the session
$session->param('guess', $g);
# Enter the main loop of the WebGuess object
until($::q->param('done')) {
$g->run();
}
# We won't get here until that exits cleanly (never!)
print "Done.";
DESCRIPTION
Contize is primarily meant to be useful in the context of CGI programming. It effectively alters the programmer's view of what is happening -- changing it from a program which is run and re-run with each input/output into a program which is continuously run, sending output and then pausing for input at certain intervals. Documentation on using Contize for this style of CGI programming can be found elsewhere, the remainder of this documentation will be more directly on Contize (and who knows... maybe there is some other use for Contize of which I haven't thought).
Contize helps an object to be suspendable and resumeable. For this to happen the object must be Contized, which is a lot like being blessed or Memoized. Once an object has been Contized several new methods are provided to it. The two most important methods are suspend and resume.
The suspend method logically replaces the normal return statement. So instead of a method returning its results directly it instead does "$self->suspend(@results)". The suspend method contains an 'exit', so upon suspend the entire process is terminated. In order to succesfully be resumed at a later point, the owner of this object should have an END block which saves the Contized object to long-term storage.
The resume method is called by the program after it has restored the Contized object from long-term storage. This restores the objects internal state so that subsequent calls to its methods will (more or less) pick up right where they left off. So, if you have a CGI::Session object for example, you might have something like this:
my $obj = $session->param('obj') || new Contize(new MyObj);
$obj->resume();
$obj->run();
Fun, eh?
METHODS
- $thingie = new Contize($thingie)
-
Takes a $thingie object and continuizes it... we replace it with ourselves and intercept all method calls.
Note that we take over the following elements of the hash:
- _child - our child object we've overtaken
- _cache - count for how we are doing catch-up wise
- _nocache - a list of methods not to cache
- _callstack - the current call stack (array)
- _callstack_count - the current count of the top callstack item
So you probably should't use these as variables in the real object.
- $thingie->nocache('methodname1', 'methodname2', ...)
-
Turn off caching for the given methods
- $thingie->somemethod(@params) ... aka AUTOLOAD
-
AUTOLOAD actually does the work. We intercept method invocations and usually cache the results. Difficult to explain...
- $thingie->suspend($retval)
-
This replaces the return function in a subroutine and suspends the object. When the object is resumed it will give $retval to the caller.
- $thingie->reset()
-
Reset the thingie so that it will be re-run. This clears the callstack and the callstack_count so that it will begin returning cached results.
- DESTROY
-
Upon destruction we undef our child, thus calling the child's own DESTROY, if such a thing exists. I'm pretty sure this is the proper way to do things, but it might break if their DESTROY does more complicated activities.
BUGS/LIMITATIONS
Contize has quite a bit of overhead for internal caching of method invocations.
There should be a bit more documentation here on how Contize actuall works.
Contize will only work on objects which use a hash as their core thingie.
SEE ALSO
Coro::Cont, http://thelackthereof.org/wiki.pl/Contize
AUTHOR
Brock Wilcox <awwaiid@thelackthereof.org>
L<http://thelackthereof.org/>
COPYRIGHT
Copyright (c) 2004 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.