NAME
OP - Compact prototyping of schema-backed object classes
NEW NAME
Oct 19 2009 - Please be advised that this distribution will be moving to a new name, Devel::Ladybug. The OP distribution will be removed from CPAN shortly. Sincere apologies for any inconvenience this may cause.
SYNOPSIS
use strict;
use warnings;
use OP qw| :all |;
create "YourApp::YourClass" => { };
See PROTOTYPE COMPONENTS in OP::Class for detailed examples.
A cheat sheet, ex/cheat.html
, is included with this distribution.
DESCRIPTION
OP is a Perl 5 framework for prototyping schema-backed object classes.
Using OP's create()
function, the developer asserts rules for object classes. OP's purpose is to automatically derive a database schema, handle object-relational mapping, and provide input validation for classes created in this manner.
OP works with MySQL/InnoDB, PostgreSQL, SQLite, and YAML flatfile. If the backing store type for a class is not specified, OP will try to automatically determine an appropriate type for the local system. If memcached is available, OP will use it in conjunction with the permanent backing store.
VERSION
This documentation is for version 0.320 of OP.
EXPORT TAGS
All exports are optional. Specify a tag or symbol by name to import it into your caller's namespace.
use OP qw| :all |;
:all
This imports each of the symbols listed below.
:create
This imports the
create
andsubtype
class prototyping functions; see OP::Class, OP::Subtype, and examples in this document.:bool
This imports
true
andfalse
boolean constants.:yield
This imports the
yield
,emit
, andbreak
functions for array collectors; see OP::Array.
FRAMEWORK ASSUMPTIONS
When using OP, as with any framework, a number of things "just happen" by design. Trying to go against the flow of any of these base assumptions is not recommended.
Configuration
See CONFIGURATION AND ENVIRONMENT in this document.
Classes Make Tables
OP derives database schemas from the assertions contained in object classes, and creates the tables that it needs.
Default Base Attributes
Database-backed OP objects always have "id", "name", "ctime", and "mtime".
id
=> OP::IDid
is the primary key at the database table level. The gets used in database table indexes, and should generally not be altered once assigned.Base64-encoded Globally Unique IDs are used by default, though it is possible to assert any scalar OP object class for the
id
column. See OP::ID, OP::Serial.name
=> OP::Namename
is a secondary human-readable key.The assertions for the name attribute may be changed to suit a class's requirements, and the name value for any object may be freely changed.
For more information on named objects, see OP::Name.
ctime
=> OP::DateTimectime
is an object's creation timestamp. OP sets this when saving an object for the first time.This is not the same as, and should not be confused with, the
st_ctime
filesystem attribute returned by thefstat
system call, which represents inode change time for files. If this ends up being too confusing or offensive, OP may use a name other thanctime
for creation time in a future version. It is currently being left alone.For more information on timestamps, see OP::DateTime.
mtime
=> OP::DateTimemtime
is the Unix timestamp representing an object's last modified time. OP updates this each time an object is saved.Again, this is an object attribute, and is unrelated to the
st_mtime
filesystem attribute returned by thefstat
system call. OP may use a different name in a future version.
undef
Requires Assertion
Undefined values in objects translate to NULL in the database, and OP does not permit this to happen by default.
Instance variables may not be undef, (and the corresponding table column may not be NULL), unless the instance variable was explicitly asserted as optional in the class prototype. To do so, provide "optional" as an assertion argument, as in the following example:
create "YourApp::Example" => {
### Do not permit NULL:
someMandatoryDate => OP::DateTime->assert,
### Permit NULL:
someOptionalDate => OP::DateTime->assert(
subtype(
optional => true,
)
),
# ...
};
Namespace Matters
OP's core packages live under the OP:: namespace. Your classes should live in their own top-level namespace, e.g. "YourApp::". This will translate (in lower case) to the name of the app's database. The database name may be overridden by implementing class method databaseName
.
Namespace elements beyond the top-level translate to lower case table names. In cases of nested namespaces, Perl's "::" delineator is swapped out for an underscore (_). The table name may be overriden by implementing class method tableName
.
create "YourApp::Example::Foo" => {
# overrides default value of "yourapp"
databaseName => sub {
my $class = shift;
return "some_legacy_db";
},
# overrides default value of "example_foo"
tableName => sub {
my $class = shift;
return "some_legacy_table";
},
# ...
};
OBJECT TYPES
OP object types are used when asserting attributes within a class, and are also suitable for instantiation or subclassing in a self-standing manner.
The usage of these types is not mandatory outside the context of creating a new class-- OP always returns attributes from the database in object form, but these object types are not a replacement for Perl's native data types in general usage, unless the developer wishes them to be.
These modes of usage are shown below, and covered in greater detail in specific object class docs.
DECLARING AS SUBCLASS
By default, a superclass of OP::Node is used for new classes. This may be overridden using __BASE__:
use OP qw| :all |;
create "YourApp::Example" => {
__BASE__ => "OP::Hash",
# ...
};
ASSERTING AS ATTRIBUTES
When defining the allowed instance variables for a class, the assert()
method is used:
#
# File: Example.pm
#
use OP qw| :all |;
create "YourApp::Example" => {
someString => OP::Str->assert,
someInt => OP::Int->assert,
};
INSTANTIATING AS OBJECTS
When instantiating, the class method new()
is used, typically with a prototype object for its argument.
#
# File: somecaller.pl
#
use strict;
use warnings;
use YourApp::Example;
my $example = YourApp::Example->new(
name => "Hello",
someString => "foo",
someInt => 12345,
);
$example->save;
$example->print;
IN METHODS
Constructors and setter methods accept both native Perl 5 data types and their OP object class equivalents. The setters will automatically handle any necessary conversion, or throw an exception if the received arg doesn't quack like a duck.
To wit, native types are OK for constructors:
my $example = YourApp::Example->new(
someString => "foo",
someInt => 123,
);
#
# someStr became a string object:
#
say $example->someString->class;
# "OP::Str"
say $example->someString->size;
# "3"
say $example->someString;
# "foo"
#
# someInt became an integer object:
#
say $example->someInt->class;
# "OP::Int"
say $example->someInt->sqrt;
# 11.0905365064094
say $example->someInt;
# 123
Native types are OK for setters:
$example->setSomeInt(456);
say $example->someInt->class;
# "OP::Int"
CORE OBJECT TYPES
The basic types listed here may be instantiated as objects, and asserted as inline attributes.
OP::Array - List
OP::Bool - Overloaded boolean
OP::DateTime - Overloaded time object
OP::Double - Overloaded double-precision number
OP::ExtID - Overloaded foreign key
OP::Float - Overloaded floating point number
OP::Hash - Hashtable
OP::ID - Overloaded GUID primary key
OP::Int - Overloaded integer
OP::Name - Unique secondary key
OP::Num - Overloaded number
OP::Rule - Regex reference (qr/ /)
OP::Serial - Auto-incrementing primary key
OP::Str - Overloaded unicode string
OP::TimeSpan - Overloaded time range object
CONSTANTS & ENUMERATIONS
OP::Constants - "dot rc" values as constants
OP::Enum - C-style enumerated types as constants
ABSTRACT CLASSES & MIX-INS
OP::Class - Abstract "Class" class
OP::Class::Dumper - Introspection mix-in
OP::Object - Abstract object class
OP::Persistence - Storage and retrieval mix-in
OP::Persistence::Bulk - Deferred fast bulk table writes
OP::Persistence::Generic - Base for vendor-specific DBI modules
OP::Persistence::MySQL - MySQL/InnoDB runtime overrides
OP::Persistence::PostgreSQL - PostgreSQL runtime overrides
OP::Persistence::SQLite - SQLite runtime overrides
OP::Node - Abstract stored object class
OP::Type - Instance variable typing
OP::Scalar - Base class for scalar values
OP::Subtype - Instance variable subtyping
HELPER MODULES
OP::Utility - System functions required globally by OP
OP::Exceptions - Errors thrown by OP
TOOLS
opconf
- Generate an .oprc on the local machineoped
- Edit OP objects using VIM and YAMLopid
- Dump OP objects to STDOUT in various formats
CONFIGURATION AND ENVIRONMENT
OP and your DBA
If using MySQL or PostgreSQL, your app's database and the "op" database should exist with the proper access prior to use - see OP::Persistence::MySQL, OP::Persistence::PostgreSQL.
OP_HOME and .oprc
OP looks for its config file, .oprc
, under $ENV{OP_HOME}. OP_HOME defaults to the current user's home directory.
To generate a first-time config for the local machine, copy the .oprc (included with this distribution as oprc-dist
) to the proper location, or run opconf
(also included with this distribution) as the user who will be running OP.
See OP::Constants for information regarding customizing and extending the local rc file.
OP and mod_perl
OP-based classes used in a mod_perl app should be preloaded by a startup script. OP_HOME must be set in the script's BEGIN block.
For example, in a file startup.pl
:
use strict;
use warnings;
BEGIN {
$ENV{OP_HOME} = '/home/user/op'; # Directory with the .oprc
}
use YourApp::Component;
use YourApp::OtherComponent;
1;
And in your httpd.conf
:
PerlRequire /path/to/your/startup.pl
SEE ALSO
OP::Class, ex/cheat.html
OP is on GitHub: http://github.com/aayars/op
POSTAMBLE
OP could be an acronym for Objective Perl, or Object Persistence, or Overpowered, or all of those things, or none of them, or something completely different. You are encouraged to come up with your own meaning, but please be kind. I'll admit the currently chosen name isn't great. It's an unregistered namespace, it conveys little meaning, and will probably offend the sensibilities of anyone who works with optree packages. The project's name may change eventually, but not today.
OP was specifically written as an object persistence layer. Its purpose is focused to the task of letting developers save and retrieve objects to/from a backing store without incurring a lot of legwork, and its usage reflects this.
Thanks to all who have provided feedback and testing. My aim is to make this software as good and as useful as possible. I welcome suggestions, contributions, and input.
AUTHOR
Alex Ayars <pause@nodekit.org>
COPYRIGHT
File: OP.pm
Copyright (c) 2009 TiVo Inc.
All rights reserved. This program and the accompanying materials
are made available under the terms of the Common Public License v1.0
which accompanies this distribution, and is available at
http://opensource.org/licenses/cpl1.0.txt