NAME
CAF Examples - Guide to the CAF Example scripts
CAF EXAMPLES
Before you Begin
First, make sure that you've read and followed the instructions in the INSTAL.txt file.
Path Conventions
A typical CGI::Application::Framework
project has the following layout:
cgi-bin/
app.cgi # The CGI script
framework/
framework.conf # Global CAF config file
projects/
MyProj/
framework.conf # MyProj config file
common-templates/ # login templates go here
common-modules/
CDBI/
MyProj.pm # The Class::DBI project module
MyProj/
app.pm # The Class::DBI application module
MyProj.pm # The project module
applications/
myapp1/
framework.conf # myapp1 config file
myapp1.pm # An application module
templates/
runmode_one.html # templates for myapp1
runmode_two.html
myapp2/
framework.conf # myapp2 config file
myapp2.pm # An application module
templates/
runmode_one.html # templates for myapp2
runmode_two.html
The Example applications follow this layout closely:
(*) cgi-bin/
app.cgi # The CGI script
(**) framework/
framework.conf # Global CAF config file
(+) projects/
Example/
framework.conf # Exmample config file
common-templates/ # login templates go here
common-modules/
CDBI/
Example.pm # The Class::DBI project module
Example/
example.pm # The Class::DBI application module
Example.pm # The project module
applications/
example_1/
framework.conf # example_1 config file
example_1.pm # The 'example_1' application
templates/
main_display.html # template for myapp1
example_2a/
framework.conf # example_2a config file
example_2a.pm # The 'example_2a' application
templates/
main_display_mutt.html # templates for example_2a
main_display_jeff.html
When we refer to the [cgi-bin]
directory, we are referring to the directory that contains app.cgi
(the [cgi-bin]
directory is indicated above with (*)
). When we refer to the [framework]
directory, we mean the directory marked with (**)
, above. When we refer to the [projects]
directory, we are referring to the projects
directory within the top level framework
directory (as indicated with (+)
, above).
A First Example
Inspect the following files on disk via the command line:
[cgi-bin]/app.cgi
and
[projects]/Example/example_1/example_1.pm
Run the example_1 application through the web, for instance at:
http://localhost/cgi-bin/app.cgi/Example/example_1
Does it give you anything more than a server error? If so, then odds are you set up your installation correctly. If you get an error, happy hunting!
The "app.cgi" file
This is the the only CGI script in the whole entire CGI::Application::Framework
! Through this single script, all your applications are run.
It works by analyzing the PATH_INFO
portion of the URL. In the following example:
http://localhost/cgi-bin/app.cgi/Example/example_1?rm=main_display
The PATH_INFO
is "/Example/example_1".
This is broken up into two parts: project (Example
) and application example_1
.
CGI::Application::Framework
searches for example_1.pm
in the project tree:
[projects]/Example/applications/example_1/example_1.pm
Once it has found your application module, it runs it. Exactly *how* it runs it is controlled by your configuration file.
The programs in the CGI::Application::Framework
follow the general structure of a CGI::Application
program. That means that at any given time, they have a run mode
which is their current state. Each run mode has a subroutine.
In the case of example_1
the only run mode is called main_display
.
When you leave out the run mode, CGI::Application::Framework
's run_app mechanism knows to call the main_display
run mode, because this is set in the configuration file.
Look at the project level framework.conf
in
[projects]/Example/framework.conf
In this file there is a paramter called post_login_rm
:
post_login_rm = main_display
This means that after login, CGI::Application::Framework
will dispatch to the main_display
run mode.
Other applications can have different start run modes. For instance, example_2
's first run mode is main_display_mutt
:
<LocationMatch example_2>
post_login_rm = main_display_mutt
</LocationMatch>
More information on configuration system is available in the CGI::Application::Framework docs.
For more information on CGI::Application, see its documentation at:
http://search.cpan.org/dist/CGI-Application/lib/CGI/Application.pm
Templates
Note that the CAF uses templates for its its presentation (aka "view") layer.
Currently, you have a choice between HTML::Template, HTML::Template::Expr
, Template::Toolkit and Petal
.
See the documentation for these modules at:
http://search.cpan.org/dist/HTML-Template/Template.pm
http://search.cpan.org/dist/Template-Toolkit/lib/Template.pm
http://search.cpan.org/dist/Petal/lib/Petal.pm
The "example_1.pm" file
This file has the heck commented out of it. Read it extensively to start to get an idea of what is possible, and what the efficient idioms are, in the CGI::Application::Framework
.
The Example.pm Base Class
Regarding the line
use base qw ( Example );
When you start developing applications that relate to a different base class, you can change this to
use base qw ( YourProject );
Make sure you create a
[projects]/YourProject/common-modules/YourProject.pm
file and a
[projects]/YourProject/common-templates
directory with the templates that you need to put in there, per what is named in your YourProject.pm
file.
You get to decide what template you need and what their names are, within reasonable limits. For instance, by default, the Framework expects different types of templates to use different extensions:
HTML::Template .html
Template::Toolkit .tmpl
Petal .xhtml
But this is a convention, and configurable in your top-level framework.conf
.
For examples on how to create a project and a set of templates, refer to
[projects]/Example/common-modules/Example.pm
[projects]/Example/common-templates
These are fairly heavily documented internally as well.
Note that, in the setup
subroutine, mode_param
and start_mode
are NOT defined. They normally would be here in a CGI::Application
program, but these are managed by the CGI::Application::Framework
module here.
Configuration
In the main_display
subroutine, the first CAF-oriented thing that you encounter is:
my $config = $self->conf->context;
This is a quick way of getting the current application configuration. $config
is a hash ref. You use it like this:
my $start_mode = $config->{'post_login_rm'};
You can store values in framework.conf
and they will be visible to your application in this hashref.
# in [projects]/Example/applications/example_1/framework.conf
favourite_colour = blue
# in [projects]/Example/applications/example_1/example_1.pm
my $config = $self->conf->context;
my $background = $config->{'favourite_colour'};
The configuration system is very powerful and flexible. To make a configuration value available to all apps in a project, declare it the project level framework.conf
:
[projects]/Example/framework.conf
To make a configuration value available to all apps on the system, declare it the top-level framework.conf
:
[framework]/framework.conf
Additionally, configuration files allow matching based on the current module, the current URL and other runtime data. So for instance, you could put the following in the top level framework.conf
:
<LocationMatch example>
favourite_colour = red
</LocationMatch>
And this would set the favourite_colour
configuration parameter to 'red'
any time an application's URL matched the string 'example'
.
See the CGI::Application::Framework
docs for details.
Session
The next thing unique to CGI::Application::Framework
is the first-class session, $self->session
. This is a "scratchpad" area that persists across all loadings of the application via the web and across all subroutines. The programmer doesn't have to create the session object, as the framework does this for the programmer. The session will come with a few pre-populated fields, e.g. _session_id, _timeout and _cgi_query, as well as whatever is defined by the programmer in the .pm file that defines the project class (e.g. Example.pm
). See that file to get some ideas of what you can do here.
Linking
$self->make_link
is meant to create all <a href=...>
style links that are displayed throughout the application. If you don't use this to link to another run mode within the application and you instead just create the <a href>
yourself then the application will die with a rather strongly-worded error message. More documentation regarding make_link and all its options can be found inside the docs of CGI::Application::Framework
.
Templates
Finally there is the templating code:
return $self->template->fill(\%tmplvars);
In one step, this creates a template, fills it with the paramters in the hash %tmplvars
and then returns the output.
Since we haven't specified a filename, the system will load a template with a filename matching the current runmode (in this case main_display.html
).
As for the type of template used (and other The template options) the template is created according to the rules set out in your application's configuration. In the case of the examples, the template type is set to HTML::Template
by default.
If you want to override this for a particular run mode, you can:
my $template = $self->template('named_config')->load(
file => 'yummy'
type => 'TemplateToolkit'
add_include_path => '.',
);
$template->param('foo' => 'bar');
return $template->output;
This would create a template named yummy.tmpl
in the Template::Toolit format, set some parameters and then return the result.
"app-db.cgi"
This is the same as app.cgi
but with one small difference -- its beginning is:
#!/usr/bin/perl -d:ptkdb
sub BEGIN {
$ENV{DISPLAY} = ":0.0" ;
}
This is meant to activate the Perl TK Debugger. You probably need to specify the hostname or IP of the display that ptkdb will target, and "xhost +" the display as well. ptkdb doesn't work on Apache::Registry
scripts, which is a bit of a drag, so you have to make sure that you run this through CGI. This is an excellent way of finding out what's wrong in your CGI::Application::Framework
applications.
"example_1.pm-small"
This isn't runable, but it is meant to show how small and compact these applications can be without all of the comment verbage in example_1.pm
It is entirely equivalent to example_1.pm
in how it would run, though.
"example_2a.pm" and "example_2b.pm"
This demonstrates how to link between two applications while maintaining the user's session.
"example_3.pm"
This demonstrates how to create a navigation bar of links to multiple applications without having to log in each time.
"example_4.pm"
This demonstrates the component embedding system. It shows you how to embed runmodes within runmodes by using the navigation bar as an example.
"example_5.pm"
This is meant to illustrate a standard CRUD application (Create, Read, Update, Delete) as done in the CAF. All database interaction is mediated via the Class::DBI
-based parallel framework that I've created for CGI::Application::Framework
called CDBI
. The file
[projects]/Example/common-modules/CDBI/Example/example.pm
mirrors the database "example", and highlights some of the relationships within it via its has_many
and has_a
statements. See documentation for this at:
http://search.cpan.org/dist/Class-DBI/lib/Class/DBI.pm
http://search.cpan.org/dist/Class-DBI-mysql/lib/Class/DBI/mysql.pm
http://search.cpan.org/dist/Class-DBI-Pg/Pg.pm
http://search.cpan.org/dist/Class-DBI-SQLite/lib/Class/DBI/SQLite.pm
Note the use of the check_rm
in process_add_user
comes from CGI::Application::Plugin::ValidateRM
. See its documentation, and the documentation for the modules that support it, at:
http://search.cpan.org/dist/CGI-Application-Plugin-ValidateRM/lib/CGI/Application/Plugin/ValidateRM.pm
http://search.cpan.org/dist/Data-FormValidator/lib/Data/FormValidator.pm
The _add_user_profile
subroutine shows a Data::FormValidator
profile that includes some programmatic elements.
AUTHOR
Richard Dice, <rdice-at-pobox.com>
COPYRIGHT & LICENSE
Copyright 2005 Richard Dice, All Rights Reserved.
This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.