/usr/share/perl5/Alzabo/Runtime/Schema.pm is in libalzabo-perl 0.92-3.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 | package Alzabo::Runtime::Schema;
use strict;
use vars qw($VERSION);
use Alzabo::Exceptions ( abbr => [ qw( logic_exception params_exception ) ] );
use Alzabo::Runtime;
use Alzabo::Utils;
use Params::Validate qw( :all );
Params::Validate::validation_options( on_fail => sub { params_exception join '', @_ } );
use base qw(Alzabo::Schema);
$VERSION = 2.0;
1;
sub load_from_file
{
my $class = shift;
my $self = $class->_load_from_file(@_);
$self->prefetch_all_but_blobs;
return $self;
}
sub _schema_file_type
{
return 'runtime';
}
sub user
{
my $self = shift;
return $self->{user};
}
sub password
{
my $self = shift;
return $self->{password};
}
sub host
{
my $self = shift;
return $self->{host};
}
sub port
{
my $self = shift;
return $self->{port};
}
sub referential_integrity
{
my $self = shift;
return defined $self->{maintain_integrity} ? $self->{maintain_integrity} : 0;
}
sub set_db_schema_name
{
my $self = shift;
$self->{db_schema_name} = shift;
}
sub set_user
{
my $self = shift;
$self->{user} = shift;
}
sub set_password
{
my $self = shift;
$self->{password} = shift;
}
sub set_host
{
my $self = shift;
$self->{host} = shift;
}
sub set_port
{
my $self = shift;
$self->{port} = shift;
}
sub set_referential_integrity
{
my $self = shift;
my $val = shift;
$self->{maintain_integrity} = $val if defined $val;
}
sub set_quote_identifiers
{
my $self = shift;
my $val = shift;
$self->{quote_identifiers} = $val if defined $val;
}
sub connect
{
my $self = shift;
my %p;
$p{user} = $self->user if defined $self->user;
$p{password} = $self->password if defined $self->password;
$p{host} = $self->host if defined $self->host;
$p{port} = $self->port if defined $self->port;
$self->driver->connect( %p, @_ );
# $self->set_referential_integrity( ! $self->driver->supports_referential_integrity );
}
sub disconnect
{
my $self = shift;
$self->driver->disconnect;
}
sub one_row
{
# could be replaced with something potentially more efficient
return shift->join(@_)->next;
}
use constant JOIN_SPEC => { join => { type => ARRAYREF | OBJECT,
optional => 1 },
tables => { type => ARRAYREF | OBJECT,
optional => 1 },
select => { type => ARRAYREF | OBJECT,
optional => 1 },
where => { type => ARRAYREF,
optional => 1 },
order_by => { type => ARRAYREF | HASHREF | OBJECT,
optional => 1 },
limit => { type => SCALAR | ARRAYREF,
optional => 1 },
distinct => { type => ARRAYREF | OBJECT,
optional => 1 },
quote_identifiers => { type => BOOLEAN,
optional => 1 },
};
sub join
{
my $self = shift;
my %p = validate( @_, JOIN_SPEC );
$p{join} ||= delete $p{tables};
$p{join} = [ $p{join} ] unless Alzabo::Utils::is_arrayref( $p{join} );
my @tables;
if ( Alzabo::Utils::is_arrayref( $p{join}->[0] ) )
{
# flattens the nested structure and produces a unique set of
# tables
@tables = values %{ { map { $_ => $_ }
grep { Alzabo::Utils::safe_isa( $_, 'Alzabo::Table' ) }
map { @$_ } @{ $p{join} } } };
}
else
{
@tables = grep { Alzabo::Utils::safe_isa($_, 'Alzabo::Table') } @{ $p{join} };
}
if ( $p{distinct} )
{
$p{distinct} =
Alzabo::Utils::is_arrayref( $p{distinct} ) ? $p{distinct} : [ $p{distinct} ];
}
if ( $p{order_by} )
{
$p{order_by} =
Alzabo::Utils::is_arrayref( $p{order_by} )
? $p{order_by}
: $p{order_by}
? [ $p{order_by} ]
: undef;
}
# We go in this order: $p{select}, $p{distinct}, @tables
my @select_tables = ( $p{select} ?
( Alzabo::Utils::is_arrayref( $p{select} ) ?
@{ $p{select} } : $p{select} ) :
$p{distinct} ?
@{ $p{distinct} } :
@tables );
my $sql = Alzabo::Runtime::sqlmaker( $self, \%p );
my @select_cols;
if ( $p{distinct} )
{
my %distinct = map { $_ => 1 } @{ $p{distinct} };
# hack so distinct is not treated as a function, just a
# bareword in the SQL
@select_cols = ( 'DISTINCT',
map { ( $_->primary_key,
$_->prefetch ?
$_->columns( $_->prefetch ) :
() ) }
@{ $p{distinct} }
);
foreach my $t (@select_tables)
{
next if $distinct{$t};
push @select_cols, $t->primary_key;
push @select_cols, $t->columns( $t->prefetch ) if $t->prefetch;
}
if ( $p{order_by} && $sql->distinct_requires_order_by_in_select )
{
my %select_cols = map { $_ => 1 } @select_cols;
push @select_cols, grep { ref } @{ $p{order_by} };
}
@select_tables = ( @{ $p{distinct} }, grep { ! $distinct{$_} } @select_tables );
}
else
{
@select_cols =
( map { ( $_->primary_key,
$_->prefetch ?
$_->columns( $_->prefetch ) :
() ) }
@select_tables );
}
$sql->select(@select_cols);
$self->_join_all_tables( sql => $sql,
join => $p{join} );
Alzabo::Runtime::process_where_clause( $sql, $p{where} ) if exists $p{where};
Alzabo::Runtime::process_order_by_clause( $sql, $p{order_by} )
if $p{order_by};
$sql->limit( ref $p{limit} ? @{ $p{limit} } : $p{limit} ) if $p{limit};
$sql->debug(\*STDERR) if Alzabo::Debug::SQL;
print STDERR Devel::StackTrace->new if Alzabo::Debug::TRACE;
my $statement = $self->driver->statement( sql => $sql->sql,
bind => $sql->bind );
if (@select_tables == 1)
{
return Alzabo::Runtime::RowCursor->new
( statement => $statement,
table => $select_tables[0]->real_table,
);
}
else
{
return Alzabo::Runtime::JoinCursor->new
( statement => $statement,
tables => [ map { $_->real_table } @select_tables ],
);
}
}
sub row_count
{
my $self = shift;
my %p = @_;
return $self->function( select => Alzabo::Runtime::sqlmaker( $self, \%p )->COUNT('*'),
%p,
);
}
sub function
{
my $self = shift;
my %p = @_;
my $sql = $self->_select_sql(%p);
my $method =
Alzabo::Utils::is_arrayref( $p{select} ) && @{ $p{select} } > 1 ? 'rows' : 'column';
$sql->debug(\*STDERR) if Alzabo::Debug::SQL;
print STDERR Devel::StackTrace->new if Alzabo::Debug::TRACE;
return $self->driver->$method( sql => $sql->sql,
bind => $sql->bind );
}
sub select
{
my $self = shift;
my $sql = $self->_select_sql(@_);
$sql->debug(\*STDERR) if Alzabo::Debug::SQL;
print STDERR Devel::StackTrace->new if Alzabo::Debug::TRACE;
return $self->driver->statement( sql => $sql->sql,
bind => $sql->bind );
}
use constant _SELECT_SQL_SPEC => { join => { type => ARRAYREF | OBJECT,
optional => 1 },
tables => { type => ARRAYREF | OBJECT,
optional => 1 },
select => { type => SCALAR | ARRAYREF | OBJECT,
optional => 1 },
where => { type => ARRAYREF,
optional => 1 },
group_by => { type => ARRAYREF | HASHREF | OBJECT,
optional => 1 },
order_by => { type => ARRAYREF | HASHREF | OBJECT,
optional => 1 },
having => { type => ARRAYREF,
optional => 1 },
limit => { type => SCALAR | ARRAYREF,
optional => 1 },
quote_identifiers => { type => BOOLEAN,
optional => 1 },
};
sub _select_sql
{
my $self = shift;
my %p = validate( @_, _SELECT_SQL_SPEC );
$p{join} ||= delete $p{tables};
$p{join} = [ $p{join} ] unless Alzabo::Utils::is_arrayref( $p{join} );
my @tables;
if ( Alzabo::Utils::is_arrayref( $p{join}->[0] ) )
{
# flattens the nested structure and produces a unique set of
# tables
@tables = values %{ { map { $_ => $_ }
grep { Alzabo::Utils::safe_isa( 'Alzabo::Table', $_ ) }
map { @$_ } @{ $p{join} } } };
}
else
{
@tables = grep { Alzabo::Utils::safe_isa( 'Alzabo::Table', $_ ) } @{ $p{join} };
}
my @funcs = Alzabo::Utils::is_arrayref( $p{select} ) ? @{ $p{select} } : $p{select};
my $sql = ( Alzabo::Runtime::sqlmaker( $self, \%p )->
select(@funcs) );
$self->_join_all_tables( sql => $sql,
join => $p{join} );
Alzabo::Runtime::process_where_clause( $sql, $p{where} )
if exists $p{where};
Alzabo::Runtime::process_group_by_clause( $sql, $p{group_by} )
if exists $p{group_by};
Alzabo::Runtime::process_having_clause( $sql, $p{having} )
if exists $p{having};
Alzabo::Runtime::process_order_by_clause( $sql, $p{order_by} )
if exists $p{order_by};
$sql->limit( ref $p{limit} ? @{ $p{limit} } : $p{limit} ) if $p{limit};
return $sql;
}
use constant _JOIN_ALL_TABLES_SPEC => { join => { type => ARRAYREF },
sql => { isa => 'Alzabo::SQLMaker' } };
sub _join_all_tables
{
my $self = shift;
my %p = validate( @_, _JOIN_ALL_TABLES_SPEC );
my @from;
my @joins;
# outer join given as only join
$p{join} = [ $p{join} ] unless ref $p{join}->[0];
# A structure like:
#
# [ [ $t_1 => $t_2 ],
# [ $t_1 => $t_3, $fk ],
# [ left_outer_join => $t_3 => $t_4 ],
# [ left_outer_join => $t_3 => $t_5, undef, [ $where_clause ] ]
#
if ( Alzabo::Utils::is_arrayref( $p{join}->[0] ) )
{
my %map;
my %tables;
foreach my $set ( @{ $p{join} } )
{
# we take some care not to change the contents of $set,
# because the caller may reuse the variable being
# referenced, and changes here could break that.
# XXX - improve
params_exception
'The table map must contain only two tables per array reference'
if @$set > 5;
my @tables;
if ( ! ref $set->[0] )
{
$set->[0] =~ /^(right|left|full)_outer_join$/i
or params_exception "Invalid join type: $set->[0]";
@tables = @$set[1,2];
push @from, [ $1, @tables, @$set[3, 4] ];
}
else
{
@tables = @$set[0,1];
push @from, grep { ! exists $tables{ $_->alias_name } } @tables;
push @joins, [ @tables, $set->[2] ];
}
# Track the tables we've seen
@tables{ $tables[0]->alias_name, $tables[1]->alias_name } = (1, 1);
# Track their relationships
push @{ $map{ $tables[0]->alias_name } }, $tables[1]->alias_name;
push @{ $map{ $tables[1]->alias_name } }, $tables[0]->alias_name;
}
# just get one key to start with
my ($key) = (each %tables)[0];
delete $tables{$key};
my @t = @{ delete $map{$key} };
while (my $t = shift @t)
{
delete $tables{$t};
push @t, @{ delete $map{$t} } if $map{$t};
}
logic_exception
"The specified table parameter does not connect all the tables involved in the join"
if keys %tables;
}
# A structure like:
#
# [ $t_1 => $t_2 => $t_3 => $t_4 ]
#
else
{
for (my $x = 0; $x < @{ $p{join} } - 1; $x++)
{
push @joins, [ $p{join}->[$x], $p{join}->[$x + 1] ];
}
@from = @{ $p{join} };
}
$p{sql}->from(@from);
return unless @joins;
foreach my $join (@joins)
{
$self->_join_two_tables( $p{sql}, @$join );
}
$p{sql}->subgroup_end;
}
sub _join_two_tables
{
my $self = shift;
my ($sql, $table_1, $table_2, $fk) = @_;
my $op = $sql->last_op eq 'and' || $sql->last_op eq 'condition' ? 'and' : 'where';
if ($fk)
{
unless ( $fk->table_from eq $table_1 && $fk->table_to eq $table_2 )
{
if ( $fk->table_from eq $table_2 && $fk->table_to eq $table_1 )
{
$fk = $fk->reverse;
}
else
{
params_exception
( "The foreign key given to join together " .
$table_1->alias_name .
" and " . $table_2->alias_name .
" does not represent a relationship between those two tables" );
}
}
}
else
{
my @fk = $table_1->foreign_keys_by_table($table_2);
logic_exception
( "The " . $table_1->name .
" table has no foreign keys to the " .
$table_2->name . " table" )
unless @fk;
logic_exception
( "The " . $table_1->name .
" table has more than 1 foreign key to the " .
$table_2->name . " table" )
if @fk > 1;
$fk = $fk[0];
}
foreach my $cp ( $fk->column_pair_names )
{
if ( $op eq 'where' )
{
# first time through loop only
$sql->where;
$sql->subgroup_start;
$sql->condition( $table_1->column( $cp->[0] ), '=', $table_2->column( $cp->[1] ) );
}
else
{
$sql->$op( $table_1->column( $cp->[0] ), '=', $table_2->column( $cp->[1] ) );
}
$op = 'and';
}
}
sub prefetch_all
{
my $self = shift;
$_->set_prefetch( $_->columns ) for $self->tables;
}
sub prefetch_all_but_blobs
{
my $self = shift;
$_->set_prefetch( grep { ! $_->is_blob } $_->columns ) for $self->tables;
}
sub prefetch_none
{
my $self = shift;
$_->set_prefetch() for $self->tables;
}
__END__
=head1 NAME
Alzabo::Runtime::Schema - Schema objects
=head1 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;
=head1 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
L<C<Alzabo::Create::Schema>|Alzabo::Create::Schema> object is saved.
=head1 INHERITS FROM
C<Alzabo::Schema>
Note: all relevant documentation from the superclass has been merged into this document.
=head1 METHODS
=head2 load_from_file ( name => $schema_name )
Loads a schema from a file. This is the only constructor for this
class. It returns an C<Alzabo::Runtime::Schema> object. Loaded
objects are cached in memory, so future calls to this method may
return the same object.
Throws: L<C<Alzabo::Exception::Params>|Alzabo::Exceptions>,
L<C<Alzabo::Exception::System>|Alzabo::Exceptions>
=head2 set_user ($user)
Sets the username to use when connecting to the database.
=head2 user
Return the username used by the schema when connecting to the database.
=head2 set_password ($password)
Set the password to use when connecting to the database.
=head2 password
Returns the password used by the schema when connecting to the
database.
=head2 set_host ($host)
Set the host to use when connecting to the database.
=head2 host
Returns the host used by the schema when connecting to the database.
=head2 set_port ($port)
Set the port to use when connecting to the database.
=head2 port
Returns the port used by the schema when connecting to the database.
=head2 set_referential_integrity ($boolean)
Turns referential integrity checking on or off. If it is on, then
when L<C<Alzabo::Runtime::Row>|Alzabo::Runtime::Row> objects are
deleted, updated, or inserted, they will report this activity to any
relevant L<C<Alzabo::Runtime::ForeignKey>|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.
=head2 referential_integrity
Returns a boolean value indicating whether this schema will attempt to
maintain referential integrity.
=head2 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.
=head2 connect (%params)
Calls the L<C<Alzabo::Driver-E<gt>connect>|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 L<C<<
Alzabo::Driver->connect() method >>|Alzabo::Driver/connect> for more
details.
=head2 disconnect
Calls the L<C<< Alzabo::Driver->disconnect()
>>|Alzabo::Driver/disconnect> method for the driver owned by the
schema.
=head2 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
L<C<Alzabo::Runtime::JoinCursor>|Alzabo::Runtime::JoinCursor> for more
information.
This method takes the following parameters:
=over 4
=item * 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 B<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 L<C<< Alzabo::Runtime::JoinCursor->next()
>>|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 L<C<< Alzabo::Runtime::JoinCursor->next()
>>|Alzabo::Runtime::JoinCursor/next> is called, however. It may be
easier to use the L<C<< Alzabo::Runtime::JoinCursor->next_as_hash()
>>|Alzabo::Runtime::JoinCursor/next_as_hash> method in this case.
=item * select => C<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.
=item * distinct => C<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.
B<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.
=item * where (optional)
See the L<documentation on where clauses for the
Alzabo::Runtime::Table class|Alzabo::Runtime::Table/Common
Parameters>.
=item * order_by (optional)
See the L<documentation on order by clauses for the
Alzabo::Runtime::Table class|Alzabo::Runtime::Table/Common
Parameters>.
=item * limit (optional)
See the L<documentation on limit clauses for the
Alzabo::Runtime::Table class|Alzabo::Runtime::Table/Common
Parameters>.
=back
If the "select" parameter specified that more than one table is
desired, then this method will return n
L<JoinCursor|Alzabo::Runtime::JoinCursor> object
representing the results of the join. Otherwise, the method returns
a L<RowCursor|Alzabo::Runtime::RowCursor> object.
Throws: L<C<Alzabo::Exception::Logic>|Alzabo::Exceptions>,
L<C<Alzabo::Exception::Params>|Alzabo::Exceptions>
=head2 one_row
This method takes the exact same parameters as the
L<C<join()>|Alzabo::Runtime::table/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: L<C<Alzabo::Exception::Logic>|Alzabo::Exceptions>,
L<C<Alzabo::Exception::Params>|Alzabo::Exceptions>
=head2 function and select
These two methods differ only in their return values.
They both take the following parameters:
=over 4
=item * select => $function or [ scalars, SQL functions and/or C<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 ...
=item * join
See the L<documentation on the join parameter for the join
method|Alzabo::Runtime::Schema/join E<lt>see belowE<gt>>.
=item * where
See the L<documentation on where clauses for the
Alzabo::Runtime::Table class|Alzabo::Runtime::Table/Common
Parameters>.
=item * order_by
See the L<documentation on order by clauses for the
Alzabo::Runtime::Table class|Alzabo::Runtime::Table/Common
Parameters>.
=item * group_by
See the L<documentation on group by clauses for the
Alzabo::Runtime::Table class|Alzabo::Runtime::Table/Common
Parameters>.
=item * 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.
=item * limit
See the L<documentation on limit clauses for the
Alzabo::Runtime::Table class|Alzabo::Runtime::Table/Common
Parameters>.
=back
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 L<Using SQL
functions|Alzabo::Intro/"Using SQL functions"> for more details.
Throws: L<C<Alzabo::Exception::Logic>|Alzabo::Exceptions>,
L<C<Alzabo::Exception::Params>|Alzabo::Exceptions>
=head3 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.
=head3 select() return values
This method always returns a new
L<C<Alzabo::DriverStatement>|Alzabo::Driver/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 C<next()>, C<next_as_hash()>, etc.
=head2 row_count
This method is simply a shortcut to get the result of COUNT('*') for a
join. It equivalent to calling C<function()> with a "select"
parameter of C<COUNT('*')>.
Throws: L<C<Alzabo::Exception::Logic>|Alzabo::Exceptions>,
L<C<Alzabo::Exception::Params>|Alzabo::Exceptions>
=head2 prefetch_all
This method will set all the tables in the schema to prefetch all
their columns. See the L<lazy column
loading|Alzabo::Runtime::Table/LAZY COLUMN LOADING> section in
L<C<Alzabo::Runtime::Table>|Alzabo::Runtime::Table> for more details.
=head2 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.
=head2 prefetch_none
This method turns of all prefetching.
=head2 name
Returns a string containing the name of the schema.
=head2 tables (@optional_list)
If no arguments are given, this method returns a list of all
L<C<Alzabo::Runtime::Table>|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 L<C<Alzabo::Exception::Params>|Alzabo::Exceptions> exception is
throws if the schema does not contain one or more of the specified
tables.
=head2 table ($name)
Returns an L<C<Alzabo::Runtime::Table>|Alzabo::Runtime::Table> object representing the
specified table.
An L<C<Alzabo::Exception::Params>|Alzabo::Exceptions> exception is
throws if the schema does not contain the table.
=head2 has_table ($name)
Returns a boolean value indicating whether the table exists in the
schema.
=head2 begin_work
Starts a transaction. Calls to this function may be nested and it
will be handled properly.
=head2 rollback
Rollback a transaction.
=head2 commit
Finishes a transaction with a commit. If you make multiple calls to
C<begin_work()>, make sure to call this method the same number of
times.
=head2 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.
=head2 driver
Returns the L<C<Alzabo::Driver>|Alzabo::Driver> object for the schema.
=head2 rules
Returns the L<C<Alzabo::RDBMSRules>|Alzabo::RDBMSRules> object for the
schema.
=head2 sqlmaker
Returns the L<C<Alzabo::SQLMaker>|Alzabo::SQLMaker> object for the
schema.
=head1 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 L<C<alias()>|Alzabo::Runtime::Table/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 C<$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;
=head1 USER AND PASSWORD INFORMATION
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
L<C<set_user()>|Alzabo::Runtime::Schema/set_user ($user)> and
L<C<set_password()>|Alzabo::Runtime::Schema/set_password ($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 C<DBIx::Connect> and
C<AppConfig>, among others.
=head1 AUTHOR
Dave Rolsky, <autarch@urth.org>
=cut
|