NAME

DBIx::VersionedSubs - all your code are belong into the DB

SYNOPSIS

package My::App;
use strict;
use base 'DBIx::VersionedSubs';

package main;
use strict;

My::App->startup($dsn);
while (my $request = Some::Server->get_request) {
    My::App->update_code; # update code from the DB
    My::App->handle_request($request);
}

ABSTRACT

This module implements a minimal driver to load your application code from a database into a namespace and to update that code whenever the database changes.

TABLES USED

This module uses two tables in the database, code_live and code_history. The code_live table stores the current version of the code and is used to initialize the namespace. The code_history table stores all modifications to the code_live table, that is, insertions, deletions and modifications of rows in it. It is used to determine if the code has changed and what changes to apply to the namespace to bring it up to par with the database version.

The two tables are presumed to have a layout like the following:

create table code_live (
    name varchar(256) not null primary key,
    code varchar(65536) not null
);

create table code_history (
    version integer primary key not null,
    timestamp varchar(15) not null,
    name varchar(256) not null,
    action varchar(1) not null, -- IUD, redundant with old_* and new_*
    old_code varchar(65536) not null,
    new_code varchar(65536) not null
);

Additional columns are ignored by this code. It is likely prudent to create an index on code_history.version as that will be used for (descending) ordering of rows.

CLASS METHODS

Package->setup

Sets up the class data defaults:

code_source  => {}
code_live    => 'code_live',
code_history => 'code_history',
code_version => 0,
verbose      => 0,

code_source contains the Perl source code for all loaded functions. code_live and code_history are the names of the two tables in which the live code and the history of changes to the live code are stored. code_version is the version of the code when it was last loaded from the database.

The verbose setting determines if progress gets output to STDERR. Likely, this package variable will get dropped in favour of a method to output (or discard) the progress.

Package->connect DSN,User,Pass,Options

Connects to the database with the credentials given.

If called in void context, stores the DBI handle in the dbh accessor, otherwise returns the DBI handle.

Package->create_sub NAME, CODE

Creates a subroutine in the Package namespace.

If you want a code block to be run automatically when loaded from the database, you can name it BEGIN. The loader code basically uses

*{"$package\::$name"} = eval "sub { $code }"

so you cannot stuff attributes and other whatnot into the name of your subroutine, not that you should.

One name is special cased - BEGIN will be immediately executed instead of installed. This is most likely what you expect. As the code elements are loaded by init_code in alphabetical order on the name, your Aardvark and AUTOLOAD subroutines will still be loaded before your BEGIN block runs.

Also, names like main::foo or Other::Package::foo are possible but get stuffed below $package. The practice doesn't get saner there.

Package->live_code_version

Returns the version number of the live code in the database.

This is done with a SELECT max(version) FROM ... query, so this might scale badly on MySQL which (I hear) is bad with queries even against indexed tables. If this becomes a problem, changing the layout to a single-row table which stores the live version number is the best approach.

Package->init_code

Adds / overwrites subroutines/methods in the Package namespace from the database.

Package->update_code

Updates the namespace from the database by loading all changes.

Note that if you have/use closures or iterators, these will behave weird if you redefine a subroutine that was previously closed over.

Package->add_code_history Name,Old,New,Action

Inserts a new row in the code history table.

This would be done with triggers on a real database, but my development target includes MySQL 3 and 4.

Package->update_sub name,code

Updates the code for the subroutine Package::$name with the code given.

Note that the update only happens in the database, so the change will only take place on the next roundtrip / code refresh.

This cannot override subroutines that don't exist in the database.

Package->insert_sub name,code

Inserts the code for the subroutine Package::$name.

Note that the insert only happens in the database, so the change will only take place on the next roundtrip / code refresh.

This can also be used to override methods / subroutines that are defined elsewhere in the Package:: namespace.

Package->redefine_sub name,code

Inserts or updates the code for the subroutine Package::$name.

Note that the change only happens in the database, so the change will only take place on the next roundtrip / code refresh.

This can be used to override methods / subroutines that are defined in the database, elsewhere in the Package:: namespace or not at all.

Package->delete_sub name,code

Deletes the code for the subroutine Package::$name.

Note that the update only happens in the database, so the change will only take place on the next roundtrip / code refresh.

If you delete the row of a subroutine that overrides a subroutine declared elsewhere (for example in Perl code), the non-database Perl code will not become visible to the Perl interpreter until the next call to Package->init_code, that is, likely until the next process restart. This will lead to very weird behaviour, so don't do that.

Package->startup(DBIargs)

Shorthand method to initialize a package from a database connection.

If Package->dbh already returns a true value, no new connection is made.

This method is equivalent to:

if (! Package->dbh) {
    Package->connect(@_);
};
Package->setup;
Package->init_code;

TODO

  • Implement closures (marked via a bare block)

  • Find a saner way instead of ->setup and %default_values for configuring the initial class values while still preventing hashref usage across packages. The "classic" approach of using Class::Data::Inheritable means that there is the risk of sharing the code_source reference across namespaces which is wrong. Maybe the accessor should simply be smart and depend on the namespace it was called with instead of a stock accessor (slated for v0.02)

  • Discuss whether it's sane to store all your code with your data in the database. It works well for http://perlmonks.org/ and the Everything Engine.

AUTHOR

Max Maischein, <corion@cpan.org>

CREDITS

Tye McQueen for suggesting the module name

SEE ALSO

The Everything Engine, http://everydevel.com/

AUTHOR

This module is licensed under the same terms as Perl itself.

ALTERNATIVE NAMES

DBIx::Seven::Days, Nothing::Driver, Corion's::Code::From::Outer::Space