NAME¶
Alzabo::Runtime::Schema - Schema objects
SYNOPSIS¶
  use Alzabo::Runtime::Schema qw(some_schema);
  my $schema = Alzabo::Runtime::Schema->load_from_file( name => 'foo' );
  $schema->set_user( $username );
  $schema->set_password( $password );
  $schema->connect;
DESCRIPTION¶
Objects in this class represent schemas, and can be used to retrieve data from
  that schema.
This object can only be loaded from a file. The file is created whenever a
  corresponding "Alzabo::Create::Schema" object is saved.
INHERITS FROM¶
"Alzabo::Schema"
Note: all relevant documentation from the superclass has been merged into this
  document.
METHODS¶
load_from_file ( name => $schema_name )¶
Loads a schema from a file. This is the only constructor for this class. It
  returns an "Alzabo::Runtime::Schema" object. Loaded objects are
  cached in memory, so future calls to this method may return the same object.
Throws: "Alzabo::Exception::Params",
  "Alzabo::Exception::System"
set_user ($user)¶
Sets the username to use when connecting to the database.
user¶
Return the username used by the schema when connecting to the database.
set_password ($password)¶
Set the password to use when connecting to the database.
password¶
Returns the password used by the schema when connecting to the database.
set_host ($host)¶
Set the host to use when connecting to the database.
host¶
Returns the host used by the schema when connecting to the database.
set_port ($port)¶
Set the port to use when connecting to the database.
port¶
Returns the port used by the schema when connecting to the database.
set_referential_integrity ($boolean)¶
Turns referential integrity checking on or off. If it is on, then when
  "Alzabo::Runtime::Row" objects are deleted, updated, or inserted,
  they will report this activity to any relevant
  "Alzabo::Runtime::ForeignKey" objects for the row, so that the
  foreign key objects can take appropriate action.
This defaults to false. If your RDBMS supports foreign key constraints, these
  should be used instead of Alzabo's built-in referential integrity checking, as
  they will be much faster.
referential_integrity¶
Returns a boolean value indicating whether this schema will attempt to maintain
  referential integrity.
set_quote_identifiers ($boolean)¶
If this is true, then all SQL constructed for this schema will have quoted
  identifiers (like `Table`.`column` in MySQL).
This defaults to false. Turning this on adds some overhead to all SQL
  generation.
connect (%params)¶
Calls the "Alzabo::Driver->connect" method for the driver owned by
  the schema. The username, password, host, and port set for the schema will be
  passed to the driver, as will any additional parameters given to this method.
  See the "Alzabo::Driver->connect() method" for more details.
disconnect¶
Calls the "Alzabo::Driver->disconnect()" method for the driver
  owned by the schema.
join¶
Joins are done by taking the tables provided in order, and finding a relation
  between them. If any given table pair has more than one relation, then this
  method will fail. The relations, along with the values given in the optional
  where clause will then be used to generate the necessary SQL. See
  "Alzabo::Runtime::JoinCursor" for more information.
This method takes the following parameters:
  - •
 
  - join => <see below>
    
    This parameter can either be a simple array reference of tables or an array
      reference of array references. In the latter case, each array reference
      should contain two tables. These array references can also include an
      optional modifier specifying a type of join for the two tables, like
      'left_outer_join', an optional foreign key object which will be used to
      join the two tables, and an optional where clause used to restrict the
      join.
    
    If a simple array reference is given, then the order of these tables is
      significant when there are more than 2 tables. Alzabo expects to find
      relationships between tables 1 & 2, 2 & 3, 3 & 4, etc.
    
    For example, given:
    
    
  join => [ $table_A, $table_B, $table_C ]
    
    
    Alzabo would expect that table A has a relationship to table B, which in
      turn has a relationship to table C. If you simply provide a simple array
      reference, you cannot include any outer joins, and every element of the
      array reference must be a table object.
    
    If you need to specify a more complicated set of relationships, this can be
      done with a slightly more complicated data structure, which looks like
      this:
    
      join => [ [ $table_A, $table_B ],
            [ $table_A, $table_C ],
            [ $table_C, $table_D ],
            [ $table_C, $table_E ] ]
    
    
    This is fairly self explanatory. Alzabo will expect to find a relationship
      between each pair of tables. This allows for the construction of
      arbitrarily complex join clauses.
    
    For even more complex needs, there are more options:
    
      join => [ [ left_outer_join => $table_A, $table_B ],
            [ $table_A, $table_C, $foreign_key ],
            [ right_outer_join => $table_C, $table_D, $foreign_key ] ]
    
    
    In this example, we are specifying two types of outer joins, and in two of
      the three cases, specifying which foreign key should be used to join the
      two tables.
    
    It should be noted that if you want to join two tables that have more than
      one foreign key between them, you must provide a foreign key object
      when using them as part of your query.
    
    The way an outer join is interpreted is that this:
    
      [ left_outer_join => $table_A, $table_B ]
    
    
    is interepreted to mean
    
      SELECT ... FROM table_A LEFT OUTER JOIN table_B ON ...
    
    
    Table order is relevant for right and left outer joins, obviously.
    
    However, for regular (inner) joins, table order is not important.
    
    It is also possible to apply restrictions to an outer join, for example:
    
      join => [ [ left_outer_join => $table_A, $table_B,
              # outer join restriction
              [ [ $table_B->column('size') > 2 ],
                'and',
                [ $table_B->column('name'), '!=', 'Foo' ] ],
            ] ]
    
    
    This corresponds to this SQL;
    
      SELECT ... FROM table_A
  LEFT OUTER JOIN table_B ON ...
              AND (table_B.size > 2 AND table_B.name != 'Foo')
    
    
    These restrictions are only allowed when performing an outer join, since
      there is no point in using them for regular inner joins. An inner join
      restriction has the same effect when included in the "WHERE"
      clause.
    
    If the more multiple array reference of specifying tables is used and no
      "select" parameter is provided, then the order of the rows
      returned from calling "Alzabo::Runtime::JoinCursor->next()"
      is not guaranteed. In other words, the array that the cursor returns will
      contain a row from each table involved in the join, but the which row
      belongs to which table cannot be determined except by examining the
      objects. The order will be the same every time
      "Alzabo::Runtime::JoinCursor->next()" is called, however. It
      may be easier to use the
      "Alzabo::Runtime::JoinCursor->next_as_hash()" method in this
      case. 
  - •
 
  - select => "Alzabo::Runtime::Table" object or objects
      (optional)
    
    This parameter specifies from which tables you would like rows returned. If
      this parameter is not given, then the "distinct" or
      "join" parameter will be used instead, with the
      "distinct" parameter taking precedence.
    
    This can be either a single table or an array reference of table
    objects.
 
  - •
 
  - distinct => "Alzabo::Runtime::Table" object or objects
      (optional)
    
    If this parameter is given, it indicates that results from the join should
      never contain repeated rows.
    
    This can be used in place of the "select" parameter to indicate
      from which tables you want rows returned. The "select"
      parameter, if given, supercedes this parameter.
    
    For some databases (notably Postgres), if you want to do a "SELECT
      DISTINCT" query then all of the columns mentioned in your "ORDER
      BY" clause must also be in your SELECT clause. Alzabo will make sure
      this is the case, but it may cause more rows to be returned than you
      expected, though this depends on the query.
    
     NOTE: The adding of columns to the SELECT clause from the ORDER BY
      clause is considered experimental, because it can change the expected
      results in some cases.
 
  - •
 
  - where (optional)
    
    See the documentation on where clauses for the Alzabo::Runtime::Table
    class.
 
  - •
 
  - order_by (optional)
    
    See the documentation on order by clauses for the Alzabo::Runtime::Table
      class.
 
  - •
 
  - limit (optional)
    
    See the documentation on limit clauses for the Alzabo::Runtime::Table
    class.
 
If the "select" parameter specified that more than one table is
  desired, then this method will return n JoinCursor object representing the
  results of the join. Otherwise, the method returns a RowCursor object.
Throws: "Alzabo::Exception::Logic",
  "Alzabo::Exception::Params"
one_row¶
This method takes the exact same parameters as the "join()" method but
  instead of returning a cursor, it returns a single array of row objects. These
  will be the rows representing the first row (a set of one or more table's
  primary keys) that is returned by the database.
Throws: "Alzabo::Exception::Logic",
  "Alzabo::Exception::Params"
function and select¶
These two methods differ only in their return values.
They both take the following parameters:
  - •
 
  - select => $function or [ scalars, SQL functions and/or
      "Alzabo::Column" objects ]
    
    If you pass an array reference for this parameter, it may contain scalars,
      SQL functions, or column objects. For example:
    
    
  $schema->function( select =>
                     [ 1,
                       $foo->column('name'),
                       LENGTH( $foo->column('name') ) ],
                     join => [ $foo, $bar_table ],
                   );
    
    
    This is equivalent to the following SQL:
    
      SELECT 1, foo.name, LENGTH( foo.name )
    FROM foo, bar
   WHERE ...
    
   
  - •
 
  - join
    
    See the documentation on the join parameter for the join method.
 
  - •
 
  - where
    
    See the documentation on where clauses for the Alzabo::Runtime::Table
    class.
 
  - •
 
  - order_by
    
    See the documentation on order by clauses for the Alzabo::Runtime::Table
      class.
 
  - •
 
  - group_by
    
    See the documentation on group by clauses for the Alzabo::Runtime::Table
      class.
 
  - •
 
  - having
    
    This parameter is specified in the same way as the "where"
      parameter, but is used to generate a "HAVING" clause. It only
      allowed when you also specify a "group_by" parameter.
 
  - •
 
  - limit
    
    See the documentation on limit clauses for the Alzabo::Runtime::Table
    class.
 
These methods are used to call arbitrary SQL functions such as 'AVG' or 'MAX',
  and to select data from individual columns. The function (or functions) should
  be the return values from the functions exported by the SQLMaker subclass that
  you are using. Please see Using SQL functions for more details.
Throws: "Alzabo::Exception::Logic",
  "Alzabo::Exception::Params"
function() return values
The return value of this method is highly context sensitive.
If you only requested a single element in your "select" parameter,
  such as "DISTINCT(foo)", then it returns the first value in scalar
  context and all the values as an array in list context.
If you requested multiple functions such as "AVG(foo), MAX(foo)", then
  it returns a single array reference, the first row of values, in scalar
  context and a list of array references in list context.
select() return values
This method always returns a new "Alzabo::DriverStatement" object
  containing the results of the query. This object has an interface very similar
  to the Alzabo cursor interface, and has methods such as "next()",
  "next_as_hash()", etc.
row_count¶
This method is simply a shortcut to get the result of COUNT('*') for a join. It
  equivalent to calling "function()" with a "select"
  parameter of "COUNT('*')".
Throws: "Alzabo::Exception::Logic",
  "Alzabo::Exception::Params"
prefetch_all¶
This method will set all the tables in the schema to prefetch all their columns.
  See the lazy column loading section in "Alzabo::Runtime::Table" for
  more details.
prefetch_all_but_blobs¶
This method will set all the tables in the schema to prefetch all their
  non-blob-type columns.
This method is called as soon as a schema is loaded.
prefetch_none¶
This method turns of all prefetching.
name¶
Returns a string containing the name of the schema.
tables (@optional_list)¶
If no arguments are given, this method returns a list of all
  "Alzabo::Runtime::Table" objects in the schema, or in a scalar
  context the number of such tables. If one or more arguments are given, returns
  a list of table objects with those names, in the same order given (or the
  number of such tables in a scalar context, but this isn't terribly useful).
An "Alzabo::Exception::Params" exception is throws if the schema does
  not contain one or more of the specified tables.
table ($name)¶
Returns an "Alzabo::Runtime::Table" object representing the specified
  table.
An "Alzabo::Exception::Params" exception is throws if the schema does
  not contain the table.
has_table ($name)¶
Returns a boolean value indicating whether the table exists in the schema.
begin_work¶
Starts a transaction. Calls to this function may be nested and it will be
  handled properly.
rollback¶
Rollback a transaction.
commit¶
Finishes a transaction with a commit. If you make multiple calls to
  "begin_work()", make sure to call this method the same number of
  times.
run_in_transaction ( sub { code... } )¶
This method takes a subroutine reference and wraps it in a transaction.
It will preserve the context of the caller and returns whatever the wrapped code
  would have returned.
driver¶
Returns the "Alzabo::Driver" object for the schema.
rules¶
Returns the "Alzabo::RDBMSRules" object for the schema.
sqlmaker¶
Returns the "Alzabo::SQLMaker" object for the schema.
JOINING A TABLE MORE THAN ONCE¶
It is possible to join to the same table more than once in a query. Table
  objects contain an "alias()" method that, when called, returns an
  object that can be used in the same query as the original table object, but
  which will be treated as a separate table. This faciliaties queries similar to
  the following SQL::
  SELECT ... FROM Foo AS F1, Foo as F2, Bar AS B ...
The object returned from the table functions more or less exactly like a table
  object. When using this table to set where clause or order by (or any other)
  conditions, it is important that the column objects for these conditions be
  retrieved from the alias object.
For example:
 my $foo_alias = $foo->alias;
 my $cursor = $schema->join( select => $foo,
                             join   => [ $foo, $bar, $foo_alias ],
                             where  => [ [ $bar->column('baz'), '=', 10 ],
                                         [ $foo_alias->column('quux'), '=', 100 ] ],
                             order_by => $foo_alias->column('briz') );
If we were to use the $foo object to retrieve the 'quux' and 'briz' columns then
  the join would simply not work as expected.
It is also possible to use multiple aliases of the same table in a join, so that
  this will work properly:
 my $foo_alias1 = $foo->alias;
 my $foo_alias2 = $foo->alias;
This information is never saved to disk. This means that if you're operating in
  an environment where the schema object is reloaded from disk every time it is
  used, such as a CGI program spanning multiple requests, then you will have to
  make a new connection every time. In a persistent environment, this is not a
  problem. For example, in a mod_perl environment, you could load the schema and
  call the "set_user()" and "set_password()" methods in the
  server startup file. Then all the mod_perl children will inherit the schema
  with the user and password already set. Otherwise you will have to provide it
  for each request.
You may ask why you have to go to all this trouble to deal with the user and
  password information. The basic reason was that I did not feel I could come up
  with a solution to this problem that was secure, easy to configure and use,
  and cross-platform compatible. Rather, I think it is best to let each user
  decide on a security practice with which they feel comfortable.
In addition, there are a number of modules aimed at helping store and use this
  sort of information on CPAN, including "DBIx::Connect" and
  "AppConfig", among others.
AUTHOR¶
Dave Rolsky, <autarch@urth.org>