NAME
SQL::Catalog - test, label, store, search and retrieve SQL queries
SYNOPSIS
shell% cd sql_lair/city,date/weather/1/
shell% cat concrete.sql
select city, date from weather where temp_lo < 20 and temp_hi > 40 LIMIT 10
shell% sql_test concrete.sql
shell% cat testexec.out # see results of prepare, execute on this
shell% cat abstract.sql
select city, date from weather where temp_lo < ? and temp_hi > ?
shell% sql_test abstract.sql 55 # send in placeholder value
shell% cat testexec.out # to see results... looks good
shell% sql_register abstract.sql basic_weather "basic weather query"
[hi_and_low] inserted as
[select city from weather where temp_lo < ? and temp_hi > ?]
... then in a Perl program (e.g. test.pl in this distribution)
my $dbh = SQL::Catalog->db_handle; # optional - get the handle as you please
my $sql = SQL::Catalog->lookup('hi_and_low');
my $sth = $dbh->prepare($sql);
$sth->execute(55);
my $rows = $sth->rows;
DESCRIPTION
Over time, it has become obvious that a few things about SQL queries are necessary. And before this module, time-consuming:
database independence
You may at some time to be forced to deploy an application which has to work on more than one database. Prior to SQL::Catalog, there were two choices - DBIx::AnyDBD and DBIx::Recordset. With SQL::Catalog, you simply store the SQL for a particular label in each database and you are done.
labelled queries
A large, well-scaled business database application has several layers with simple well-defined tasks. The layer just above the database does database things. It inserts. It retrieves. It updates. etc, etc. Call this the database application layer. Just above the database application layer is the business object layer. These are conceptual entities whose data structures are program data structures. For permanent stores, they make simple, technology-agnostic requests of the database application layer, which then takes the business data and stores it as database data. Then above this we have the application layer. And this layer makes use of business objects, ldap objects, web objects, what have you, to string together a complete application.
queryable queries
That's right, you want to be able to query on the queries themselves. It makes it easy to do a study on just what queries are doing what.
separation of concerns
By now, everyone has heard that phrase: "my templating module is the best because it allows the HTML designer to work separately from the Perl programmer." Well, given that databases are another foreign technology to Perl proper, it only makes sense that the same ability that is afforded to HTML designers be afforded to SQL programmers.
centralization of queries
This makes it easy for someone to see how you did something so they can imitate.
SQL::Catalog addresses all of these issues.
COMMON STEPS TO USAGE
Develop your concrete query in a db shell
The first step to developing a database query is to play around at the db shell. In this case, you normally don't have any placeheld values. You just keep mucking with the query until it gives you what you want.
When you finally get what you want, save it in a file, say concrete.sql
for example. Here is a concrete query:
select city, date from weather where temp_hi > 20
Abstract your query with placeholders
Now it's time to make your query more abstract. So we do the following:
select city, date from weather where temp_hi > ?
and save in a different file, say abstract.sql
.
Now let's test this query also, being sure to pass in data for the placeholder fields:
sql_test abstract.sql 34
Certain drivers are not very good with their error messages in response to queries sent in without placeholder bindings, so take care here.
And let's cat testexec.out to see the results.
Register your query (store in the sql_category table)
sql_register abstract.sql city_date_via_temp_hi
and the system tells you
[city_date_via_temp_hi] saved as
[select city, date from weather where temp_hi > ?]
Use your query from DBI:
use SQL::Catalog;
my $dbh = SQL::Catalog->db_handle; # or however you get your DBI handles
my $SQL = SQL::Catalog->lookup('city_date_via_temp_hi') or die "not found";
my $sth = $dbh->prepare($SQL, $cgi->param('degrees'));
.... etc
INSTALLATION
See the README in the home directory of the distribution.
What SQL::Catalog does
It stores each query in a database table. I could have gone for something more normalized and exquisite in database design but wanted to maintain database independence without requiring extra tools for schema creation and database use.
Right now we have schema creation and SQL code which works for Informix and Postgresql and welcome more.
The queries are stored in these tables (this file is db-creation/postgresql.renderer
):
CREATE TABLE sql_catalog (
label varchar(80) ,
cmd varchar(40) ,
phold int4 ,
author varchar(40) ,
query varchar(65536) ,
comments varchar(1600) ,
PRIMARY KEY (label)
);
CREATE TABLE sql_catalog_ft (
label_ft varchar(80) ,
tbl varchar(255) ,
col varchar(255) ,
PRIMARY KEY (label_ft)
);
And here is the result of ONE sql_register:
mydb=# select * from sql_catalog_ft;
label_ft | tbl | col
----------+---------+---------
basic_weather | weather | city
basic_weather | weather | date
basic_weather | weather | temp_lo
basic_weather | weather | temp_hi
(4 rows)
mydb=# select * from sql_catalog;
label | cmd | phold | author | query | comments
-------+--------+-------+----------+------------------------------------------------------------------------------+----------
basic_weather | SELECT | 1 | metaperl | select city, date, temp_lo, temp_hi from weather where temp_lo < ? LIMIT 40
| ahah
(1 row)
NOTES
Read the README for thorough install instructions for various databases.
Do NOT end your SQL statements for testing within this framework with a semicolon.
It is entirely feasible (and oh so cool), to have a "query server". Ie, a cheap Linux box running MySQL which has no table but sql_catalog on it. And all it does is serve the queries. Then your actual "data database" can be on a completely different machine. The idea is that SQL::Catalog connects to the table sql_catalog based on its
DSN
value (see README) while your data database connects based on a different DSN.When dropping these tables, you will also have to drop one index
AUTHOR
T. M. Brannon, <tbone@cpan.org>
Substantial contribution (and ass-kicking) by Jonathan Leffler.
SEE ALSO
There are several related modules on CPAN. Each do some of what SQL::Catalog does.
Class::Phrasebook::SQL stores a "phrasebook" of SQL in XML files. Allows for retrieval of queries via a convenient API. The querying of queries that SQL::Catalog supports can be done using an XML processor along with SQL::Statement.
DBIx::SearchProfiles. Does query labeling and also has some convenience functions for query retrieval. It does not store the SQL in a database or make it searchable by table, column, or number of placeholders. Your standard Perl data munging techniques would be the way to do statistical analysis of your queries.
Queries stored in Perl modules
A different approach is suggested using Perl modules. Interesting idea.
http://perlmonks.org/index.pl?node_id=96268&lastnode_id=96273
"Leashing DBI"
Various issues in building applications on top of DBI.
http://perlmonks.org/index.pl?node=Leashing%20DBI&lastnode_id=96268