NAME
Mojolicious::Plugin::Tables -- Quickstart and grow a Tables-Maintenance application
SYNOPSIS
# in a new Mojolicious app..
$app->plugin( Tables => {connect_info=>['dbi:Pg:dbname="mydatabase"', 'uid', 'pwd']} );
# when ready to grow..
$app->plugin( Tables => {model_class => 'StuffDB'} );
DESCRIPTION
Mojolicious::Plugin::Tables is a Mojolicious plugin which helps you grow a high-featured web app by starting with basic table / relationship maintenance and growing by overriding default behaviours.
By supplying 'connect_info' (DBI-standard connect parameters) you get a presentable database maintenance web app, featuring full server-side support for paged lists of large tables using 'Datatables' (datatables.net), plus ajax handling for browsing 1-many relationships, plus JQueryUI-styled select lists for editing many-to-1 picklists.
By supplying your own Model Class you can override most of the default behaviours and build an enterprise-ready rdbms-based web app.
By supplying your own templates for context-specific calls you can start to give your app a truly specialised look and feel.
STATUS
This is an early release. Guides and more override-hooks coming Real Soon Now.
GROWTH PATH
Ground Zero Startup
tables dbi-connect-params..
The 'tables' script is supplied with this distribution. Give it standard DBI-compatible parameters on the commandline and it will run a minimal web-server at localhost:3000 to allow maintenance on the named database.
Day One
In your Mojolicious 'startup'..
$self->plugin(Tables => { connect_info => [ connect-param, connect-param, .. ] };
Add this line to a new Mojolicious app then run it using any Mojo-based server (morbo, prefork, hypnotoad..) to achieve exactly the same functionality as the 'tables' script.
Day Two
# templates/:table/{view,edit,dml,page,view}.html.ep
# e.g.
# templates/artist/view.html.ep
<h1>Artist: <%= $row %></h1>
For any :table in your database, create override-templates as required. e.g. The above code will give a very different look-and-feel when viewing a single Artist, but all other pages are unchanged. For better examples and to see which stash-variables are available, start by copying the distribution templates from ../Plugin/Tables/resources/templates into your own template area.
Infinity and Beyond
$self->plugin(Tables => { model_class => 'MyDB' } );
Where model_class implements the specification given below. This lets you start to customise and grow your web app.
Customising model_class
Prepare your own model_class to override the default database settings which "Tables" has determined from the database. This class (and its per-table descendants) can be within or without your Mojolicious application.
model_class PARENT
Your model_class must inherit from Mojolicious::Plugin::Tables::Model. So, e.g.
package StuffDB;
use strict;
use warnings;
use base qw/Mojolicious::Plugin::Tables::Model/;
...
model_class METHODS
Your model_class optionally overrides any of the following methods.
connect_info
A sub returning an arrayref supplying your locally declared DBI parameters. e.g.
sub connect_info { [ 'dbi:Pg:dbname="stuff"', '', '' ] }
which works for a Postgres database called 'stuff' accepting IDENT credentials.
glossary
A sub returning a hashref which maps "syllables" to displaynames. "Syllables" are all the abbreviated words that are separated by underscores in your database table and column names. Any syllable by default is made into a nice word by simply init-capping it, so the glossary only needs to supply non-obvious displaynames. The mapping 'id=>Identifier' is built-in. So for example with table or column names such as "stock", "active_stock", "stock_id", "stock_name", "dyngrp_id", "mkt_ldr", "ccld_ts" we would supply just this
sub glossary { +{
ccld => 'Cancelled',
dyngrp => 'Dynamic Group',
mkt => 'Market',
ldr => 'Leader',
ts => 'Timestamp',
} }
.. and we will see labels "Stock", "Active Stock", "Stock Identifier", "Stock Name", "Dynamic Group Identifier", "Market Leader", "Cancelled Timestamp" in the generated application.
input_attrs
A sub returning a hashref giving appropriate html5 form-input tag attributes for any fieldname. By default these attributes are derived depending on field type and database length. But these can be overriden here, e.g.
sub input_attrs { +{
var_pct => { min=>-100, max=>100 },
vol_pct => { min=>-100, max=>500 },
name => { size=>80 },
picture => { size=>80, type=>'url' },
email => { size=>40, type=>'email' },
email_verified => { type=>'checkbox' },
ts => { step=>1 },
} }
rel_name_map
A sub returning a hashref which maps default generated relationship names to more appropriate choices. More detail in DBIx::Class::Schema::Loader. e.g.
sub rel_name_map { +{
AssetRange => { range_age => 'range' },
Asset => { symbol => 'lse_security' },
Dyngrp => { dyngrps => 'subgroups' },
Trader => { authority_granters => 'grants_to',
authority_traders => 'audit_trail' },
} }
ResultClass Methods
After creating a model_class as described above you will automatically be able to introduce 'ResultClass' classes for any of the tables in your database. Place these directly 'under' your model_class, e.g. if StuffDB is your model_class and you want to introduce a nice stringify rule for the table 'asset', then you can create the class StuffDB::Asset and give it just the stringify method. e.g. from http://octalfutures.com :
package StuffDB::Asset;
use strict;
use warnings;
sub stringify {
my $self = shift;
my $latest_str = '';
if (my $latest = $self->latest) {
$latest_str = " $latest";
if (my $var_pct = $self->var_pct) {
$latest_str .= sprintf ' (%+.2f%%)', $var_pct
}
}
return sprintf '%s (%s)%s', $self->name, $self->dataset_code, $latest_str
}
1;
Any of these ResultClasses will inherit all methods of DBIx::Class::Row. In addition these methods inherit all methods of Mojolicious::Plugin::Tables::Model::Row, namely:
stringify
It's recommended to implement this. The stringification logic for this table. The default implementation tries to use any columns such as 'cd' or 'description', and falls back to joining the primary keys.
present
Generate a presentation of a row->column value, formatted depending on type. Args: column_name, a hash-ref containing schema info about that column, and a hashref containing context info (currently just 'foredit=>1').
options
Given: column_name, a hash-ref containing schema info about that column, a hash-ref containing info about the parent table, the full DBIX::Class schema, and a hash-ref containing schema information about all tables..
Generate the full pick-list that lets the fk $column pick from its parent, in a structure suitable for the Mojolicious 'select_field' tag. The default version simply lists all choices (limited by safety row-count of 200) but inherited versions are expected to do more context-sensitive filtering. Works as a class method to support the 'add' context.
nuke
Perform full depth-wise destruction of a database record. The default implementation runs an alghorithm to delete all child records and finally delete $self. Override this to prohibit (by dieing) or perform additional work.
all the rest
Of course, all the methods described at DBIx::Class::Row can be overriden here.
CAVEAT
We use dynamically-generated DBIx::Class classes. This technique does not scale well for very large numbers of tables. Previous (private) incarnations of this Framework used specially prepared high-performance versions of Class::DBI::Loader to speed this up. So that speeding-up at start-time is a TODO for this DBIx::Class-based release.
SOURCE and ISSUES REPOSITORY
Open-Sourced at Github: https://github.com/frank-carnovale/Mojolicious-Plugin-Tables. Please use the Issues register there.
COPYRIGHT AND LICENSE
Copyright (C) 2016, Frank Carnovale <frankc@cpan.org>
This program is free software, you can redistribute it and/or modify it under the terms of the Artistic License version 2.0.
SEE ALSO
Mojolicious, DBIx::Class::Schema::Loader, Mojolicious::Guides, http://mojolicious.org.