| 1 |
package Class::Tangram; |
| 2 |
|
| 3 |
# Copyright (c) 2001, 2002 Sam Vilain. All right reserved. This file |
| 4 |
# is licensed under the terms of the Perl Artistic license. |
| 5 |
|
| 6 |
=head1 NAME |
| 7 |
|
| 8 |
Class::Tangram - create constructors, accessor and update methods for |
| 9 |
objects from a Tangram-compatible object specification. |
| 10 |
|
| 11 |
=head1 SYNOPSIS |
| 12 |
|
| 13 |
package MyObject; |
| 14 |
|
| 15 |
use base qw(Class::Tangram); |
| 16 |
|
| 17 |
our $fields => { int => [ qw(foo bar) ], |
| 18 |
string => [ qw(baz quux) ] }; |
| 19 |
|
| 20 |
package main; |
| 21 |
|
| 22 |
my $object = MyObject->new(foo => 2, baz => "hello"); |
| 23 |
|
| 24 |
print $object->baz(); # prints "hello" |
| 25 |
|
| 26 |
$object->set_quux("Something"); |
| 27 |
|
| 28 |
$object->set_foo("Something"); # dies - not an integer |
| 29 |
|
| 30 |
=head1 DESCRIPTION |
| 31 |
|
| 32 |
Class::Tangram is a tool for defining objects attributes. Simply |
| 33 |
define your object's fields/attributes using the same syntax |
| 34 |
introduced in _A Guided Tour of Tangram_, and you get objects that |
| 35 |
work As You'd Expect(tm). |
| 36 |
|
| 37 |
Class::Tangram has no dependancy upon Tangram, and vice versa. |
| 38 |
Neither requires anything special of your objects, nor do they insert |
| 39 |
any special fields into your objects. This is a very important |
| 40 |
feature with innumerable benefits, and few (if any) other object |
| 41 |
persistence tools have this feature. |
| 42 |
|
| 43 |
So, fluff aside, let's run through how you use Class::Tangram to make |
| 44 |
objects. |
| 45 |
|
| 46 |
First, you decide upon the attributes your object is going to have. |
| 47 |
You might do this using UML, or you might pick an existing database |
| 48 |
table and declare each column to be an attribute (you can leave out |
| 49 |
"id"; that one is implicit). |
| 50 |
|
| 51 |
Your object should use Class::Tangram as a base class; |
| 52 |
|
| 53 |
use base qw(Class::Tangram) |
| 54 |
|
| 55 |
or for older versions of perl: |
| 56 |
|
| 57 |
use Class::Tangram; |
| 58 |
use vars qw(@ISA); |
| 59 |
@ISA = qw(Class::Tangram) |
| 60 |
|
| 61 |
You should then define a C<$fields> variable in the scope of the |
| 62 |
package, that is a B<hash> from attribute B<types> (see |
| 63 |
L<Tangram::Type>) to either an B<array> of B<attribute names>, or |
| 64 |
another B<hash> from B<attribute names> to B<options hashes> (or |
| 65 |
C<undef>). The layout of this structure coincides with the C<fields> |
| 66 |
portion of a tangram schema (see L<Tangram::Schema>). Note: the term |
| 67 |
`schema' is used frequently to refer to the C<$fields> structure. |
| 68 |
|
| 69 |
For example, |
| 70 |
|
| 71 |
package MyObject; |
| 72 |
use base qw(Class::Tangram); |
| 73 |
|
| 74 |
our $fields = { |
| 75 |
int => { |
| 76 |
juiciness => undef, |
| 77 |
segments => { |
| 78 |
# this code reference is called when this |
| 79 |
# attribute is set, to check the value is |
| 80 |
# OK |
| 81 |
check_func => sub { |
| 82 |
die "too many segments" |
| 83 |
if (${(shift)} > 30); |
| 84 |
}, |
| 85 |
# the default for this attribute. |
| 86 |
init_default => 7, |
| 87 |
}, |
| 88 |
}, |
| 89 |
ref => { |
| 90 |
grower => undef, |
| 91 |
}, |
| 92 |
|
| 93 |
# 'required' attributes - insist that these fields are |
| 94 |
# set, both with constructor and set()/set_X methods |
| 95 |
string => { |
| 96 |
# true: 'type' must have non-empty value (for |
| 97 |
# strings) or be logically true (for other types) |
| 98 |
type => { required => 1 }, |
| 99 |
|
| 100 |
# false: 'tag' must be defined but may be empty |
| 101 |
tag => { required => '' }, |
| 102 |
}, |
| 103 |
|
| 104 |
# fields allowed by Class::Tangram but not ever |
| 105 |
# stored by Tangram - no type checking by default |
| 106 |
transient => [ qw(_tangible) ], |
| 107 |
}; |
| 108 |
|
| 109 |
It is of critical importance to your sanity that you understand how |
| 110 |
anonymous hashes and anonymous arrays work in Perl. Some additional |
| 111 |
features are used above that have not yet been introduced, but you |
| 112 |
should be able to look at the above data structure and see that it |
| 113 |
satisfies the conditions stated in the paragraph before it. If it is |
| 114 |
hazy, I recommend reading L<perlref> or L<perlreftut>. |
| 115 |
|
| 116 |
When the schema for the object is first imported (see L<Schema |
| 117 |
import>), Class::Tangram defines accessor functions for each of the |
| 118 |
attributes defined in the schema. These accessor functions are then |
| 119 |
available as C<$object-E<gt>function> on created objects. By virtue |
| 120 |
of inheritance, various other methods are available. |
| 121 |
|
| 122 |
From Class::Tangram 1.13 onwards, no use of perl's C<AUTOLOAD> |
| 123 |
functionality is used. |
| 124 |
|
| 125 |
=cut |
| 126 |
|
| 127 |
use strict; |
| 128 |
use Carp qw(croak cluck); |
| 129 |
|
| 130 |
use vars qw($VERSION %defaults); |
| 131 |
|
| 132 |
use Set::Object; |
| 133 |
|
| 134 |
use Pod::Constants -trim => 1, |
| 135 |
'MODULE RELEASE' => sub { ($VERSION) = /(\d+\.\d+)/ }, |
| 136 |
'Default Type Checking' => sub { %defaults = eval; }; |
| 137 |
|
| 138 |
# $types{$class}->{$attribute} is the tangram type of each attribute |
| 139 |
my (%types); |
| 140 |
|
| 141 |
# $attribute_options{$class}->{$attribute} is the hash passed to tangram |
| 142 |
# for the given attribute |
| 143 |
my (%attribute_options); |
| 144 |
|
| 145 |
# $check{$class}->{$attribute}->($value) is a function that will die |
| 146 |
# if $value is not alright, see check_X functions |
| 147 |
my (%check); |
| 148 |
|
| 149 |
# Destructors for each attribute. They are called as |
| 150 |
# $cleaners{$class}->{$attribute}->($self, $attribute); |
| 151 |
my (%cleaners); |
| 152 |
|
| 153 |
# init_default values for each attribute. These could be hash refs, |
| 154 |
# array refs, code refs, or simple scalars. They will be stored as |
| 155 |
# $init_defaults{$class}->{$attribute} |
| 156 |
my (%init_defaults); |
| 157 |
|
| 158 |
# $required_attributes{$class}->{$attribute} records which attributes |
| 159 |
# are required... used only by new() at present. |
| 160 |
my (%required_attributes); |
| 161 |
|
| 162 |
# if a class is abstract, complain if one is constructed. |
| 163 |
my (%abstract); |
| 164 |
|
| 165 |
# records the inheritances of classes. |
| 166 |
my (%bases); |
| 167 |
|
| 168 |
=head1 METHODS |
| 169 |
|
| 170 |
The following methods are available for all Class::Tangram objects |
| 171 |
|
| 172 |
=head2 Constructor |
| 173 |
|
| 174 |
A Constructor is a method that returns a new instance of an object. |
| 175 |
|
| 176 |
=over 4 |
| 177 |
|
| 178 |
=item Class-E<gt>new (attribute1 =E<gt> value, attribute2 =E<gt> value) |
| 179 |
|
| 180 |
Sets up a new object of type C<Class>, with attributes set to the |
| 181 |
values supplied. |
| 182 |
|
| 183 |
Can also be used as an object method (normal use is as a "class |
| 184 |
method"), in which case it returns a B<copy> of the object, without |
| 185 |
any deep copying. |
| 186 |
|
| 187 |
=cut |
| 188 |
|
| 189 |
sub new ($@) |
| 190 |
{ |
| 191 |
my $invocant = shift; |
| 192 |
my $class = ref $invocant || $invocant; |
| 193 |
|
| 194 |
(my @values, @_) = @_; |
| 195 |
|
| 196 |
# Setup the object |
| 197 |
my $self = { }; |
| 198 |
bless $self, $class; |
| 199 |
|
| 200 |
# auto-load schema as necessary |
| 201 |
exists $types{$class} or import_schema($class); |
| 202 |
|
| 203 |
croak "Attempt to instantiate an abstract type" |
| 204 |
if ($abstract{$class}); |
| 205 |
|
| 206 |
if (ref $invocant) |
| 207 |
{ |
| 208 |
# The copy constructor; this could be better :) |
| 209 |
# this has the side effect of much auto-vivification. |
| 210 |
%$self = %$invocant; |
| 211 |
$self->set (@values); # override with @values |
| 212 |
} |
| 213 |
else |
| 214 |
{ |
| 215 |
$self->set (@values); # start with @values |
| 216 |
} |
| 217 |
|
| 218 |
$self->_fill_init_default(); |
| 219 |
$self->_check_required(); |
| 220 |
|
| 221 |
return $self; |
| 222 |
|
| 223 |
} |
| 224 |
|
| 225 |
sub _fill_init_default { |
| 226 |
my $self = shift; |
| 227 |
my $class = ref $self or die "_fill_init_default usage error"; |
| 228 |
|
| 229 |
# fill in fields that have defaults |
| 230 |
while ( my ($attribute, $default) = |
| 231 |
each %{$init_defaults{$class}} ) { |
| 232 |
|
| 233 |
next if (exists $self->{$attribute}); |
| 234 |
|
| 235 |
my $setter = "set_$attribute"; |
| 236 |
if (ref $default eq "CODE") { |
| 237 |
# sub { }, attribute gets return value |
| 238 |
$self->$setter( $default->($self) ); |
| 239 |
|
| 240 |
} elsif (ref $default eq "HASH") { |
| 241 |
# hash ref, copy hash |
| 242 |
$self->$setter( { %{ $default } } ); |
| 243 |
|
| 244 |
} elsif (ref $default eq "ARRAY") { |
| 245 |
# array ref, copy array |
| 246 |
$self->$setter( [ @{ $default } ] ); |
| 247 |
|
| 248 |
} else { |
| 249 |
# something else, an object or a scalar |
| 250 |
$self->$setter($default); |
| 251 |
} |
| 252 |
} |
| 253 |
} |
| 254 |
|
| 255 |
sub _check_required { |
| 256 |
my $self = shift; |
| 257 |
my $class = ref $self; |
| 258 |
|
| 259 |
# make sure field is not undef if 'required' option is set |
| 260 |
if (my $required = $required_attributes{$class}) { |
| 261 |
|
| 262 |
# find the immediate caller outside of this package |
| 263 |
my $i = 0; |
| 264 |
$i++ while UNIVERSAL::isa($self, scalar(caller($i))||";->"); |
| 265 |
|
| 266 |
# give Tangram some lenience - it is exempt from the effects |
| 267 |
# of the "required" option |
| 268 |
unless ( caller($i) =~ m/^Tangram::/ ) { |
| 269 |
my @missing; |
| 270 |
while ( my ($attribute, $value) = each %$required ) { |
| 271 |
push(@missing, $attribute) |
| 272 |
if ! exists $self->{$attribute}; |
| 273 |
} |
| 274 |
croak("object missing required attribute(s): " |
| 275 |
.join(', ',@missing).'.') if @missing; |
| 276 |
} |
| 277 |
} |
| 278 |
} |
| 279 |
|
| 280 |
=back |
| 281 |
|
| 282 |
=head2 Accessing & Setting Attributes |
| 283 |
|
| 284 |
=over |
| 285 |
|
| 286 |
=item $instance->set(attribute => $value, ...) |
| 287 |
|
| 288 |
Sets the attributes of the given instance to the given values. croaks |
| 289 |
if there is a problem with the values. |
| 290 |
|
| 291 |
This function simply calls C<$instance-E<gt>set_attribute($value)> for |
| 292 |
each of the C<attribute =E<gt> $value> pairs passed to it. |
| 293 |
|
| 294 |
=cut |
| 295 |
|
| 296 |
sub set { |
| 297 |
my $self = shift; |
| 298 |
|
| 299 |
# yes, this is a lot to do. yes, it's slow. But I'm fairly |
| 300 |
# certain that this could be handled efficiently if it were to be |
| 301 |
# moved inside the Perl interpreter or an XS module |
| 302 |
UNIVERSAL::isa($self, "Class::Tangram") or croak "type mismatch"; |
| 303 |
my $class = ref $self; |
| 304 |
exists $check{$class} or import_schema($class); |
| 305 |
croak "set must be called with an even number of arguments" |
| 306 |
if (scalar(@_) & 1); |
| 307 |
|
| 308 |
while (my ($name, $value) = splice @_, 0, 2) { |
| 309 |
|
| 310 |
my $setter = "set_".$name; |
| 311 |
|
| 312 |
croak "attempt to set an illegal field $name in a $class" |
| 313 |
unless $self->can($setter); |
| 314 |
|
| 315 |
$self->$setter($value); |
| 316 |
} |
| 317 |
} |
| 318 |
|
| 319 |
=item $instance->get("attribute") |
| 320 |
|
| 321 |
Gets the value of C<$attribute>. This simply calls |
| 322 |
C<$instance-E<gt>get_attribute>. If multiple attributes are listed, |
| 323 |
then a list of the attribute values is returned in order. Note that |
| 324 |
you get back the results of the scalar context C<get_attribute> call |
| 325 |
in this case. |
| 326 |
|
| 327 |
=cut |
| 328 |
|
| 329 |
sub get { |
| 330 |
my $self = shift; |
| 331 |
croak "get what?" unless @_; |
| 332 |
UNIVERSAL::isa($self, "Class::Tangram") or croak "type mismatch"; |
| 333 |
|
| 334 |
my $class = ref $self; |
| 335 |
exists $check{$class} or import_schema($class); |
| 336 |
|
| 337 |
my $multiget = (scalar(@_) != 1); |
| 338 |
|
| 339 |
my @return; |
| 340 |
while ( my $field = shift ) { |
| 341 |
my $getter = "get_".$field; |
| 342 |
croak "attempt to read an illegal field $field in a $class" |
| 343 |
unless $self->can($getter); |
| 344 |
|
| 345 |
if ( $multiget ) { |
| 346 |
push @return, scalar($self->$getter()); |
| 347 |
} else { |
| 348 |
return $self->$getter(); |
| 349 |
} |
| 350 |
} |
| 351 |
|
| 352 |
return @return; |
| 353 |
} |
| 354 |
|
| 355 |
=item $instance->attribute($value) |
| 356 |
|
| 357 |
If C<$value> is not given, then |
| 358 |
this is equivalent to C<$instance-E<gt>get_attribute> |
| 359 |
|
| 360 |
If C<$value> is given, then this is equivalent to |
| 361 |
C<$instance-E<gt>set_attribute($value)>. This usage issues a warning |
| 362 |
if warnings are on; you should change your code to use the |
| 363 |
set_attribute syntax for better readability. OO veterans will tell |
| 364 |
you that for maintainability object method names should always be a |
| 365 |
verb. |
| 366 |
|
| 367 |
=item $instance->get_attribute |
| 368 |
|
| 369 |
=item $instance->set_attribute($value) |
| 370 |
|
| 371 |
The normative way of getting and setting attributes. If you wish to |
| 372 |
override the behaviour of an object when getting or setting an |
| 373 |
attribute, override these functions. They will be called when you use |
| 374 |
C<$instance-E<gt>attribute>, C<$instance-E<gt>get()>, constructors, |
| 375 |
etc. |
| 376 |
|
| 377 |
=item $instance->attribute_includes(@objects) |
| 378 |
|
| 379 |
=item $instance->attribute_insert(@objects) |
| 380 |
|
| 381 |
=item $instance->attribute_size |
| 382 |
|
| 383 |
=item $instance->attribute_clear |
| 384 |
|
| 385 |
=item $instance->attribute_remove(@objects) |
| 386 |
|
| 387 |
This suite of functions applies to attributes that are sets (C<iset> |
| 388 |
or C<set>). It could in theory also apply generally to all |
| 389 |
collections - ie also arrays (C<iarray> or C<array>), and hashes |
| 390 |
(C<hash>, C<ihash>). This will be implemented subject to user demand. |
| 391 |
|
| 392 |
=back |
| 393 |
|
| 394 |
B<Note:> The above functions can be overridden, but they may not be |
| 395 |
called with the C<$self-E<gt>SUPER::> superclass chaining method. |
| 396 |
This is because they are not defined within the scope of |
| 397 |
Class::Tangram, only your package. |
| 398 |
|
| 399 |
=cut |
| 400 |
|
| 401 |
=head1 ATTRIBUTE TYPE CHECKING |
| 402 |
|
| 403 |
Class::Tangram provides type checking of attributes when attributes |
| 404 |
are set - either using the default C<set_attribute> functions, or |
| 405 |
created via the C<new> constructor. |
| 406 |
|
| 407 |
The checking has default behaviour for each type of attribute (see |
| 408 |
L<Default Type Checking>), and can be extended arbitrarily via a |
| 409 |
per-attribute C<check_func>, described below. Critical attributes can |
| 410 |
be marked as such with the C<required> flag. |
| 411 |
|
| 412 |
The specification of this type checking is placed in the class schema, |
| 413 |
in the per-attribute B<options hash>. This is a Class::Tangram |
| 414 |
extension to the Tangram schema structure. |
| 415 |
|
| 416 |
=over |
| 417 |
|
| 418 |
=item check_func |
| 419 |
|
| 420 |
A function that is called with a B<reference> to the new value in |
| 421 |
C<$_[0]>. It should call C<die()> if the value is bad. Note that |
| 422 |
this check_func will never be passed an undefined value; this is |
| 423 |
covered by the "required" option, below. |
| 424 |
|
| 425 |
In the example schema (above), the attribute C<segments> has a |
| 426 |
C<check_func> that prevents setting the value to anything greater than |
| 427 |
30. Note that it does not prevent you from setting the value to |
| 428 |
something that is not an integer; if you define a C<check_func>, it |
| 429 |
replaces the default. |
| 430 |
|
| 431 |
=item required |
| 432 |
|
| 433 |
If this option is set to a true value, then the attribute must be set |
| 434 |
to a true value to pass type checking. For string attributes, this |
| 435 |
means that the string must be defined and non-empty (so "0" is true). |
| 436 |
For other attribute types, the normal Perl definition of logical truth |
| 437 |
is used. |
| 438 |
|
| 439 |
If the required option is defined but logically false, (ie "" or 0), |
| 440 |
then the attribute must also be defined, but may be set to a logically |
| 441 |
false value. |
| 442 |
|
| 443 |
If the required option is undefined, then the attribute may be set to |
| 444 |
an undefined value. |
| 445 |
|
| 446 |
For integration with tangram, the C<new()> function has a special |
| 447 |
hack; if it is being invoked from within Tangram, then the required |
| 448 |
test is skipped. |
| 449 |
|
| 450 |
=back |
| 451 |
|
| 452 |
=head2 Other per-attribute options |
| 453 |
|
| 454 |
Any of the following options may be inserted into the per-attribute |
| 455 |
B<options hash>: |
| 456 |
|
| 457 |
=over |
| 458 |
|
| 459 |
=item init_default |
| 460 |
|
| 461 |
This value specifies the default value of the attribute when |
| 462 |
it is created with C<new()>. It is a scalar value, it is |
| 463 |
copied to the fresh object. If it is a code reference, that |
| 464 |
code reference is called and its return value inserted into |
| 465 |
the attribute. If it is an ARRAY or HASH reference, then |
| 466 |
that array or hash is COPIED into the attribute. |
| 467 |
|
| 468 |
=item destroy_func |
| 469 |
|
| 470 |
If anything special needs to happen to this attribute before the |
| 471 |
object is destroyed (or when someone calls |
| 472 |
C<$object-E<gt>clear_refs()>), then define this. It is called as |
| 473 |
C<$sub-E<gt>($object, "attribute")>. |
| 474 |
|
| 475 |
=back |
| 476 |
|
| 477 |
=head2 Default Type Checking |
| 478 |
|
| 479 |
# The following list is eval'ed from this documentation |
| 480 |
# when Class::Tangram loads, and used as default attribute |
| 481 |
# options for the specified types. So, eg, the default |
| 482 |
# "init_default" for "set" types is a subroutine that |
| 483 |
# returns a new Set::Object container. |
| 484 |
|
| 485 |
# "parse" is special - it is passed the options hash given |
| 486 |
# by the user and should return (\&check_func, |
| 487 |
# \&destroy_func). This is how the magical string type |
| 488 |
# checking is performed - see the entry for parse_string(), |
| 489 |
# below. |
| 490 |
|
| 491 |
int => { check_func => \&check_int }, |
| 492 |
real => { check_func => \&check_real }, |
| 493 |
string => { parse => \&parse_string }, |
| 494 |
ref => { check_func => \&check_obj, |
| 495 |
destroy_func => \&destroy_ref }, |
| 496 |
array => { check_func => \&check_array, |
| 497 |
destroy_func => \&destroy_array }, |
| 498 |
iarray => { check_func => \&check_array, |
| 499 |
destroy_func => \&destroy_array }, |
| 500 |
flat_array => { check_func => \&check_flat_array }, |
| 501 |
set => { check_func => \&check_set, |
| 502 |
destroy_func => \&destroy_set, |
| 503 |
init_default => sub { Set::Object->new() } }, |
| 504 |
iset => { check_func => \&check_set, |
| 505 |
destroy_func => \&destroy_set, |
| 506 |
init_default => sub { Set::Object->new() } }, |
| 507 |
dmdatetime => { check_func => \&check_dmdatetime }, |
| 508 |
rawdatetime => { check_func => \&check_rawdatetime }, |
| 509 |
rawdate => { check_func => \&check_rawdate }, |
| 510 |
rawtime => { check_func => \&check_rawtime }, |
| 511 |
flat_hash => { check_func => \&check_flat_hash }, |
| 512 |
transient => { check_func => \&check_nothing }, |
| 513 |
hash => { check_func => \&check_hash, |
| 514 |
destroy_func => \&destroy_hash, |
| 515 |
get_func => \&get_hash }, |
| 516 |
perl_dump => { check_func => \&check_nothing } |
| 517 |
|
| 518 |
=over |
| 519 |
|
| 520 |
=item check_X (\$value) |
| 521 |
|
| 522 |
This series of internal functions are built-in C<check_func> functions |
| 523 |
defined for all of the standard Tangram attribute types. |
| 524 |
|
| 525 |
=over |
| 526 |
|
| 527 |
=item check_string |
| 528 |
|
| 529 |
checks that the supplied value is less than 255 characters long. |
| 530 |
|
| 531 |
=cut |
| 532 |
|
| 533 |
sub check_string { |
| 534 |
croak "string too long" |
| 535 |
if (length ${$_[0]} > 255); |
| 536 |
} |
| 537 |
|
| 538 |
=item check_int |
| 539 |
|
| 540 |
checks that the value is a (possibly signed) integer |
| 541 |
|
| 542 |
=cut |
| 543 |
|
| 544 |
my $int_re = qr/^-?\d+$/; |
| 545 |
sub check_int { |
| 546 |
croak "not an integer" |
| 547 |
if (${$_[0]} !~ m/$int_re/o); |
| 548 |
} |
| 549 |
|
| 550 |
=item check_real |
| 551 |
|
| 552 |
checks that the value is a real number, by stringifying it and |
| 553 |
matching it against (C<m/^-?\d*(\.\d*)?(e-?\d*)?$/>). Inefficient? |
| 554 |
Yes. Patches welcome. |
| 555 |
|
| 556 |
=cut |
| 557 |
|
| 558 |
my $real_re = qr/^-?\d*(\.\d*)?(e-?\d*)?$/; |
| 559 |
sub check_real { |
| 560 |
croak "not a real number" |
| 561 |
if (${$_[0]} !~ m/$real_re/o); |
| 562 |
} |
| 563 |
|
| 564 |
=item check_obj |
| 565 |
|
| 566 |
checks that the supplied variable is a reference to a blessed object |
| 567 |
|
| 568 |
=cut |
| 569 |
|
| 570 |
# this pattern matches a regular reference |
| 571 |
my $obj_re = qr/^(?:HASH|ARRAY|SCALAR)?$/; |
| 572 |
sub check_obj { |
| 573 |
croak "not an object reference" |
| 574 |
if ((ref ${$_[0]}) =~ m/$obj_re/o); |
| 575 |
} |
| 576 |
|
| 577 |
=item check_flat_array |
| 578 |
|
| 579 |
checks that $value is a ref ARRAY and that all elements are unblessed |
| 580 |
scalars. Does NOT currently check that all values are of the correct |
| 581 |
type (int vs real vs string, etc) |
| 582 |
|
| 583 |
=cut |
| 584 |
|
| 585 |
sub check_flat_array { |
| 586 |
croak "not a flat array" |
| 587 |
if (ref ${$_[0]} ne "ARRAY"); |
| 588 |
croak "flat array may not contain references" |
| 589 |
if (map { (ref $_ ? "1" : ()) } @{$_[0]}); |
| 590 |
} |
| 591 |
|
| 592 |
=item check_array |
| 593 |
|
| 594 |
checks that $value is a ref ARRAY, and that each element in the array |
| 595 |
is a reference to a blessed object. |
| 596 |
|
| 597 |
=cut |
| 598 |
|
| 599 |
sub check_array { |
| 600 |
croak "array attribute not passed an array ref" |
| 601 |
if (ref ${$_[0]} ne "ARRAY"); |
| 602 |
for my $a (@{${$_[0]}}) { |
| 603 |
croak "member in array not an object reference" |
| 604 |
if ((ref $a) =~ m/$obj_re/o); |
| 605 |
} |
| 606 |
} |
| 607 |
|
| 608 |
=item check_set |
| 609 |
|
| 610 |
checks that $value->isa("Set::Object") |
| 611 |
|
| 612 |
=cut |
| 613 |
|
| 614 |
sub check_set { |
| 615 |
croak "set type not passed a Set::Object" |
| 616 |
unless (UNIVERSAL::isa(${$_[0]}, "Set::Object")); |
| 617 |
} |
| 618 |
|
| 619 |
=item check_rawdate |
| 620 |
|
| 621 |
checks that $value is of the form YYYY-MM-DD, or YYYYMMDD, or YYMMDD. |
| 622 |
|
| 623 |
=cut |
| 624 |
|
| 625 |
# YYYY-MM-DD HH:MM:SS |
| 626 |
my $rawdate_re = qr/^(?: \d{4}-\d{2}-\d{2} |
| 627 |
| (?:\d\d){3,4} |
| 628 |
)$/x; |
| 629 |
sub check_rawdate { |
| 630 |
croak "invalid SQL rawdate" |
| 631 |
unless (${$_[0]} =~ m/$rawdate_re/o); |
| 632 |
} |
| 633 |
|
| 634 |
=item check_rawtime |
| 635 |
|
| 636 |
checks that $value is of the form HH:MM(:SS)? |
| 637 |
|
| 638 |
=cut |
| 639 |
|
| 640 |
# YYYY-MM-DD HH:MM:SS |
| 641 |
my $rawtime_re = qr/^\d{1,2}:\d{2}(?::\d{2})?$/; |
| 642 |
sub check_rawtime { |
| 643 |
croak "invalid SQL rawtime" |
| 644 |
unless (${$_[0]} =~ m/$rawtime_re/o); |
| 645 |
} |
| 646 |
|
| 647 |
=item check_rawdatetime |
| 648 |
|
| 649 |
checks that $value is of the form YYYY-MM-DD HH:MM(:SS)? (the time |
| 650 |
and/or the date can be missing), or a string of numbers between 6 and |
| 651 |
14 numbers long. |
| 652 |
|
| 653 |
=cut |
| 654 |
|
| 655 |
my $rawdatetime_re = qr/^(?: |
| 656 |
# YYYY-MM-DD HH:MM:SS |
| 657 |
(?: (?:\d{4}-\d{2}-\d{2}\s+)? |
| 658 |
\d{1,2}:\d{2}(?::\d{2})? |
| 659 |
| \d{4}-\d{2}-\d{2} |
| 660 |
) |
| 661 |
| # YYMMDD, etc |
| 662 |
(?:\d\d){3,7} |
| 663 |
)$/x; |
| 664 |
sub check_rawdatetime { |
| 665 |
croak "invalid SQL rawdatetime dude" |
| 666 |
unless (${$_[0]} =~ m/$rawdatetime_re/o); |
| 667 |
} |
| 668 |
|
| 669 |
=item check_dmdatetime |
| 670 |
|
| 671 |
checks that $value is of the form YYYYMMDDHH:MM:SS, or those allowed |
| 672 |
for rawdatetime. |
| 673 |
|
| 674 |
=cut |
| 675 |
|
| 676 |
sub check_dmdatetime { |
| 677 |
croak "invalid SQL rawdatetime dude" |
| 678 |
unless (${$_[0]} =~ m/^\d{10}:\d\d:\d\d$|$rawdatetime_re/o); |
| 679 |
} |
| 680 |
|
| 681 |
=item check_flat_hash |
| 682 |
|
| 683 |
checks that $value is a ref HASH and all values are scalars. Does NOT |
| 684 |
currently check that all values are of the correct type (int vs real |
| 685 |
vs string, etc) |
| 686 |
|
| 687 |
=cut |
| 688 |
|
| 689 |
sub check_flat_hash { |
| 690 |
croak "not a hash" |
| 691 |
unless (ref ${$_[0]} eq "HASH"); |
| 692 |
while (my ($k, $v) = each %${$_[0]}) { |
| 693 |
croak "hash not flat" |
| 694 |
if (ref $k or ref $v); |
| 695 |
} |
| 696 |
} |
| 697 |
|
| 698 |
=item check_hash |
| 699 |
|
| 700 |
checks that $value is a ref HASH, that every key in the hash is a |
| 701 |
scalar, and that every value is a blessed object. |
| 702 |
|
| 703 |
=cut |
| 704 |
|
| 705 |
sub check_hash { |
| 706 |
croak "not a hash" |
| 707 |
unless (ref ${$_[0]} eq "HASH"); |
| 708 |
while (my ($k, $v) = each %${$_[0]}) { |
| 709 |
croak "hash key not flat" |
| 710 |
if (ref $k); |
| 711 |
croak "hash value not an object" |
| 712 |
if (ref $v !~ m/$obj_re/); |
| 713 |
} |
| 714 |
} |
| 715 |
|
| 716 |
=item check_nothing |
| 717 |
|
| 718 |
checks whether Australians like sport |
| 719 |
|
| 720 |
=cut |
| 721 |
|
| 722 |
sub check_nothing { } |
| 723 |
|
| 724 |
=back |
| 725 |
|
| 726 |
=item destroy_X ($instance, $attr) |
| 727 |
|
| 728 |
Similar story with the check_X series of functions, these are called |
| 729 |
during object destruction on every attribute that has a reference that |
| 730 |
might need breaking. Note: B<these functions all assume that |
| 731 |
attributes belonging to an object that is being destroyed may be |
| 732 |
destroyed also>. In other words, do not allow distinct objects to |
| 733 |
share Set::Object containers or hash references in their attributes, |
| 734 |
otherwise when one gets destroyed the others will lose their data. |
| 735 |
|
| 736 |
Available functions: |
| 737 |
|
| 738 |
=over |
| 739 |
|
| 740 |
=item destroy_array |
| 741 |
|
| 742 |
empties an array |
| 743 |
|
| 744 |
=cut |
| 745 |
|
| 746 |
sub destroy_array { |
| 747 |
my $self = shift; |
| 748 |
my $attr = shift; |
| 749 |
my $t = tied $self->{$attr}; |
| 750 |
@{$self->{$attr}} = () |
| 751 |
unless (defined $t and $t =~ m,Tangram::CollOnDemand,); |
| 752 |
delete $self->{$attr}; |
| 753 |
} |
| 754 |
|
| 755 |
=item destroy_set |
| 756 |
|
| 757 |
Calls Set::Object::clear to clear the set |
| 758 |
|
| 759 |
=cut |
| 760 |
|
| 761 |
sub destroy_set { |
| 762 |
my $self = shift; |
| 763 |
my $attr = shift; |
| 764 |
|
| 765 |
my $t = tied $self->{$attr}; |
| 766 |
return if (defined $t and $t =~ m,Tangram::CollOnDemand,); |
| 767 |
if (ref $self->{$attr} eq "Set::Object") { |
| 768 |
$self->{$attr}->clear; |
| 769 |
} |
| 770 |
delete $self->{$attr}; |
| 771 |
} |
| 772 |
|
| 773 |
=item destroy_hash |
| 774 |
|
| 775 |
empties a hash |
| 776 |
|
| 777 |
=cut |
| 778 |
|
| 779 |
sub destroy_hash { |
| 780 |
my $self = shift; |
| 781 |
my $attr = shift; |
| 782 |
my $t = tied $self->{$attr}; |
| 783 |
%{$self->{$attr}} = () |
| 784 |
unless (defined $t and $t =~ m,Tangram::CollOnDemand,); |
| 785 |
delete $self->{$attr}; |
| 786 |
} |
| 787 |
|
| 788 |
=item destroy_ref |
| 789 |
|
| 790 |
destroys a reference. |
| 791 |
|
| 792 |
=cut |
| 793 |
|
| 794 |
sub destroy_ref { |
| 795 |
my $self = shift; |
| 796 |
delete $self->{shift}; |
| 797 |
} |
| 798 |
|
| 799 |
=back |
| 800 |
|
| 801 |
=item parse_X ($attribute, { schema option }) |
| 802 |
|
| 803 |
Parses the schema option field, and returns one or two closures that |
| 804 |
act as a check_X and a destroy_X function for the attribute. |
| 805 |
|
| 806 |
This is currently a very ugly hack, parsing the SQL type definition of |
| 807 |
an object. But it was bloody handy in my case for hacking this in |
| 808 |
quickly. This is probably unmanagably unportable across databases; |
| 809 |
but send me bug reports on it anyway, and I'll try and make the |
| 810 |
parsers work for as many databases as possible. |
| 811 |
|
| 812 |
This perhaps should be replaced by primitives that go the other way, |
| 813 |
building the SQL type definition from a more abstract definition of |
| 814 |
the type. |
| 815 |
|
| 816 |
Available functions: |
| 817 |
|
| 818 |
=over |
| 819 |
|
| 820 |
=item parse_string |
| 821 |
|
| 822 |
parses SQL types of: |
| 823 |
|
| 824 |
=over |
| 825 |
|
| 826 |
=cut |
| 827 |
|
| 828 |
use vars qw($quoted_part $sql_list); |
| 829 |
|
| 830 |
$quoted_part = qr/(?: \"([^\"]+)\" | \'([^\']+)\' )/x; |
| 831 |
$sql_list = qr/\(\s* |
| 832 |
( |
| 833 |
$quoted_part |
| 834 |
(?:\s*,\s* $quoted_part )* |
| 835 |
) \s*\)/x; |
| 836 |
|
| 837 |
sub parse_string { |
| 838 |
|
| 839 |
my $attribute = shift; |
| 840 |
my $option = shift; |
| 841 |
|
| 842 |
# simple case; return the check_string function. We don't |
| 843 |
# need a destructor for a string so don't return one. |
| 844 |
if (!$option->{sql}) { |
| 845 |
return \&check_string; |
| 846 |
} |
| 847 |
|
| 848 |
=item CHAR(N), VARCHAR(N) |
| 849 |
|
| 850 |
closure checks length of string is less than N characters |
| 851 |
|
| 852 |
=cut |
| 853 |
|
| 854 |
if ($option->{sql} =~ m/^\s*(?:var)?char\s*\(\s*(\d+)\s*\)/ix) { |
| 855 |
my $max_length = $1; |
| 856 |
return sub { |
| 857 |
die "string too long for $attribute" |
| 858 |
if (length ${$_[0]} > $max_length); |
| 859 |
}; |
| 860 |
|
| 861 |
=item TINYBLOB, BLOB, LONGBLOB |
| 862 |
|
| 863 |
checks max. length of string to be 255, 65535 or 16777215 chars |
| 864 |
respectively. Also works with "TEXT" instead of "BLOB" |
| 865 |
|
| 866 |
=cut |
| 867 |
|
| 868 |
} elsif ($option->{sql} =~ m/^\s*(tiny|long|medium)? |
| 869 |
(?:blob|text)/ix) { |
| 870 |
my $max_length = ($1 ? ($1 eq "tiny"?255:2**24 - 1) |
| 871 |
: 2**16 - 1); |
| 872 |
return sub { |
| 873 |
die "string too long for $attribute" |
| 874 |
if (${$_[0]} and length ${$_[0]} > $max_length); |
| 875 |
}; |
| 876 |
|
| 877 |
=item SET("members", "of", "set") |
| 878 |
|
| 879 |
checks that the value passed is valid as a SQL set type, and that all |
| 880 |
of the passed values are allowed to be a member of that set. |
| 881 |
|
| 882 |
=cut |
| 883 |
|
| 884 |
} elsif (my ($members) = $option->{sql} =~ |
| 885 |
m/^\s*set\s*$sql_list/oi) { |
| 886 |
|
| 887 |
my %members; |
| 888 |
$members{lc($1 || $2)} = 1 |
| 889 |
while ( $members =~ m/\G[,\s]*$quoted_part/cog ); |
| 890 |
|
| 891 |
return sub { |
| 892 |
for my $x (split /\s*,\s*/, ${$_[0]}) { |
| 893 |
croak ("SQL set badly formed or invalid member $x " |
| 894 |
." (SET" . join(",", keys %members). ")") |
| 895 |
if (not exists $members{lc($x)}); |
| 896 |
} |
| 897 |
}; |
| 898 |
|
| 899 |
=item ENUM("possible", "values") |
| 900 |
|
| 901 |
checks that the value passed is one of the allowed values. |
| 902 |
|
| 903 |
=cut |
| 904 |
|
| 905 |
} elsif (my ($values) = $option->{sql} =~ |
| 906 |
m/^\s*enum\s*$sql_list/oi ) { |
| 907 |
|
| 908 |
my %values; |
| 909 |
$values{lc($1 || $2)} = 1 |
| 910 |
while ( $values =~ m/\G[,\s]*$quoted_part/gc); |
| 911 |
|
| 912 |
return sub { |
| 913 |
croak ("invalid enum value ${$_[0]} must be (" |
| 914 |
. join(",", keys %values). ")") |
| 915 |
if (not exists $values{lc(${$_[0]})}); |
| 916 |
} |
| 917 |
|
| 918 |
|
| 919 |
} else { |
| 920 |
die ("Please build support for your string SQL type in " |
| 921 |
."Class::Tangram (".$option->{sql}.")"); |
| 922 |
} |
| 923 |
} |
| 924 |
|
| 925 |
=back |
| 926 |
|
| 927 |
=back |
| 928 |
|
| 929 |
=back |
| 930 |
|
| 931 |
=head2 Quick Object Dumping and Destruction |
| 932 |
|
| 933 |
=over |
| 934 |
|
| 935 |
=item $instance->quickdump |
| 936 |
|
| 937 |
Quickly show the blessed hash of an object, without descending into |
| 938 |
it. Primarily useful when you have a large interconnected graph of |
| 939 |
objects so don't want to use the B<x> command within the debugger. |
| 940 |
It also doesn't have the side effect of auto-vivifying members. |
| 941 |
|
| 942 |
This function returns a string, suitable for print()ing. It does not |
| 943 |
currently escape unprintable characters. |
| 944 |
|
| 945 |
=cut |
| 946 |
|
| 947 |
sub quickdump($) { |
| 948 |
my $self = shift; |
| 949 |
|
| 950 |
my $r = "REF ". (ref $self). "\n"; |
| 951 |
for my $k (sort keys %$self) { |
| 952 |
$r .= (" $k => " |
| 953 |
. ( |
| 954 |
tied $self->{$k} |
| 955 |
|| ( ref $self->{$k} |
| 956 |
? $self->{$k} |
| 957 |
: "'".$self->{$k}."'" ) |
| 958 |
) |
| 959 |
. "\n"); |
| 960 |
} |
| 961 |
return $r; |
| 962 |
} |
| 963 |
|
| 964 |
|
| 965 |
=item $instance->DESTROY |
| 966 |
|
| 967 |
This function ensures that all of your attributes have their |
| 968 |
destructors called. It calls the destroy_X function for attributes |
| 969 |
that have it defined, if that attribute exists in the instance that we |
| 970 |
are destroying. It calls the destroy_X functions as destroy_X($self, |
| 971 |
$k) |
| 972 |
|
| 973 |
=cut |
| 974 |
|
| 975 |
sub DESTROY($) { |
| 976 |
my $self = shift; |
| 977 |
|
| 978 |
my $class = ref $self; |
| 979 |
|
| 980 |
# if no cleaners are known for this class, it hasn't been imported |
| 981 |
# yet. Don't call import_schema, that would be a bad idea in a |
| 982 |
# destructor. |
| 983 |
exists $cleaners{$class} or return; |
| 984 |
|
| 985 |
# for every attribute that is defined, and has a cleaner function, |
| 986 |
# call the cleaner function. |
| 987 |
for my $k (keys %$self) { |
| 988 |
if (defined $cleaners{$class}->{$k} and exists $self->{$k}) { |
| 989 |
$cleaners{$class}->{$k}->($self, $k); |
| 990 |
} |
| 991 |
} |
| 992 |
$self->{_DESTROYED} = 1; |
| 993 |
} |
| 994 |
|
| 995 |
=item $instance->clear_refs |
| 996 |
|
| 997 |
This clears all references from this object, ie exactly what DESTROY |
| 998 |
normally does, but calling an object's destructor method directly is |
| 999 |
bad form. Also, this function has no qualms with loading the class' |
| 1000 |
schema with import_schema() as needed. |
| 1001 |
|
| 1002 |
This is useful for breaking circular references, if you know you are |
| 1003 |
no longer going to be using an object then you can call this method, |
| 1004 |
which in many cases will end up cleaning up most of the objects you |
| 1005 |
want to get rid of. |
| 1006 |
|
| 1007 |
However, it still won't do anything about Tangram's internal reference |
| 1008 |
to the object, which must still be explicitly unlinked with the |
| 1009 |
Tangram::Storage->unload method. |
| 1010 |
|
| 1011 |
=cut |
| 1012 |
|
| 1013 |
sub clear_refs($) { |
| 1014 |
my $self = shift; |
| 1015 |
my $class = ref $self; |
| 1016 |
|
| 1017 |
exists $cleaners{$class} or import_schema($class); |
| 1018 |
|
| 1019 |
# break all ref's, sets, arrays |
| 1020 |
for my $k (keys %$self) { |
| 1021 |
if (defined $cleaners{$class}->{$k} and exists $self->{$k}) { |
| 1022 |
$cleaners{$class}->{$k}->($self, $k); |
| 1023 |
} |
| 1024 |
} |
| 1025 |
$self->{_NOREFS} = 1; |
| 1026 |
} |
| 1027 |
|
| 1028 |
=back |
| 1029 |
|
| 1030 |
=head1 FUNCTIONS |
| 1031 |
|
| 1032 |
The following functions are not intended to be called as object |
| 1033 |
methods. |
| 1034 |
|
| 1035 |
=head2 Schema Import |
| 1036 |
|
| 1037 |
our $fields = { int => [ qw(foo bar) ], |
| 1038 |
string => [ qw(baz quux) ] }; |
| 1039 |
|
| 1040 |
# Version 1.115 and below compatibility: |
| 1041 |
our $schema = { |
| 1042 |
fields => { int => [ qw(foo bar) ], |
| 1043 |
string => [ qw(baz quux) ] } |
| 1044 |
}; |
| 1045 |
|
| 1046 |
=over |
| 1047 |
|
| 1048 |
=item Class::Tangram::import_schema($class) |
| 1049 |
|
| 1050 |
Parses a tangram object field list, in C<${"${class}::fields"}> (or |
| 1051 |
C<${"${class}::schema"}-E<gt>{fields}> to the internal type information |
| 1052 |
hashes. It will also define all of the attribute accessor and update |
| 1053 |
methods in the C<$class> package. |
| 1054 |
|
| 1055 |
Note that calling this function twice for the same class is not |
| 1056 |
tested and may produce arbitrary results. Patches welcome. |
| 1057 |
|
| 1058 |
=cut |
| 1059 |
|
| 1060 |
sub import_schema($) { # Damn this function is long |
| 1061 |
my $class = shift; |
| 1062 |
|
| 1063 |
eval { |
| 1064 |
my ($fields, $bases, $abstract); |
| 1065 |
{ |
| 1066 |
|
| 1067 |
# Here, we go hunting around for their defined schema and |
| 1068 |
# options |
| 1069 |
no strict 'refs'; |
| 1070 |
local $^W=0; |
| 1071 |
eval { |
| 1072 |
$fields = (${"${class}::fields"} || |
| 1073 |
${"${class}::schema"}->{fields}); |
| 1074 |
$abstract = (${"${class}::abstract"} || |
| 1075 |
${"${class}::schema"}->{abstract}); |
| 1076 |
$bases = ${"${class}::schema"}->{bases}; |
| 1077 |
}; |
| 1078 |
if ( my @stack = @{"${class}::ISA"}) { |
| 1079 |
# clean "bases" information from @ISA |
| 1080 |
my %seen = map { $_ => 1 } $class, __PACKAGE__; |
| 1081 |
#$bases = []; # will anything break without this? it's needed for recording inheritances later on |
| 1082 |
while ( my $super = pop @stack ) { |
| 1083 |
if ( defined ${"${super}::schema"} |
| 1084 |
or defined ${"${super}::fields"} ) { |
| 1085 |
push @$bases, $super; |
| 1086 |
} else { |
| 1087 |
push @stack, grep { !$seen{$_}++ } |
| 1088 |
@{"${super}::ISA"}; |
| 1089 |
} |
| 1090 |
} |
| 1091 |
if ( !$fields and !@$bases ) { |
| 1092 |
die ("No schema and no Class::Tangram " |
| 1093 |
."superclass for $class; define " |
| 1094 |
."${class}::fields!"); |
| 1095 |
} |
| 1096 |
} |
| 1097 |
} |
| 1098 |
|
| 1099 |
# if this is an abstract type, do not allow it to be |
| 1100 |
# instantiated |
| 1101 |
if ($abstract) { |
| 1102 |
$abstract{$class} = 1; |
| 1103 |
} |
| 1104 |
|
| 1105 |
# If there are any base classes, import them first so that the |
| 1106 |
# check, cleaners and init_defaults can be inherited |
| 1107 |
if (defined $bases) { |
| 1108 |
(ref $bases eq "ARRAY") |
| 1109 |
or die "bases not an array ref for $class"; |
| 1110 |
|
| 1111 |
# record bases of current class for later retrieval via Run-time type information |
| 1112 |
$bases{$class} = $bases; |
| 1113 |
|
| 1114 |
# Note that the order of your bases is significant, that |
| 1115 |
# is if you are using multiple iheritance then the later |
| 1116 |
# classes override the earlier ones. |
| 1117 |
for my $super ( @$bases ) { |
| 1118 |
import_schema $super unless (exists $check{$super}); |
| 1119 |
|
| 1120 |
# copy each of the per-class configuration hashes to |
| 1121 |
# this class as defaults. |
| 1122 |
my ($k, $v); |
| 1123 |
|
| 1124 |
# FIXME - this repetition of code is getting silly :) |
| 1125 |
$types{$class}->{$k} = $v |
| 1126 |
while (($k, $v) = each %{ $types{$super} } ); |
| 1127 |
$check{$class}->{$k} = $v |
| 1128 |
while (($k, $v) = each %{ $check{$super} } ); |
| 1129 |
$cleaners{$class}->{$k} = $v |
| 1130 |
while (($k, $v) = each %{ $cleaners{$super} } ); |
| 1131 |
$attribute_options{$class}->{$k} = $v |
| 1132 |
while (($k, $v) = each %{ $attribute_options{$super} } ); |
| 1133 |
$init_defaults{$class}->{$k} = $v |
| 1134 |
while (($k, $v) = each %{ $init_defaults{$super} } ); |
| 1135 |
$required_attributes{$class}->{$k} = $v |
| 1136 |
while (($k, $v) = each %{ $required_attributes{$super} } ); |
| 1137 |
} |
| 1138 |
} |
| 1139 |
|
| 1140 |
# iterate over each of the *types* of fields (string, int, ref, etc.) |
| 1141 |
while (my ($type, $v) = each %$fields) { |
| 1142 |
if (ref $v eq "ARRAY") { |
| 1143 |
$v = { map { $_, undef } @$v }; |
| 1144 |
} |
| 1145 |
my $def = $defaults{$type}; |
| 1146 |
|
| 1147 |
# iterate each of the *attributes* of a particular type |
| 1148 |
while (my ($attribute, $options) = each %$v) { |
| 1149 |
|
| 1150 |
# this is what we are finding out about each attribute |
| 1151 |
# $type is already set |
| 1152 |
my ($default, $check_func, $required, $cleaner); |
| 1153 |
# set defaults from what they give |
| 1154 |
$options ||= {}; |
| 1155 |
if (ref $options eq "HASH" or |
| 1156 |
UNIVERSAL::isa($options, 'Tangram::Type')) { |
| 1157 |
($check_func, $default, $required, $cleaner) |
| 1158 |
= @{$options}{qw(check_func init_default |
| 1159 |
required destroy_func)}; |
| 1160 |
} |
| 1161 |
|
| 1162 |
# Fill their settings with info from defaults |
| 1163 |
if (ref $def eq "HASH") { |
| 1164 |
|
| 1165 |
# try to magically parse their options |
| 1166 |
if ( $def->{parse} and !($check_func and $cleaner) ) { |
| 1167 |
my @a = $def->{parse}->($attribute, $options); |
| 1168 |
$check_func ||= $a[0]; |
| 1169 |
$cleaner ||= $a[1]; |
| 1170 |
} |
| 1171 |
|
| 1172 |
# fall back to defaults for this class |
| 1173 |
$check_func ||= $def->{check_func}; |
| 1174 |
$cleaner ||= $def->{destroy_func}; |
| 1175 |
$default = $def->{init_default} unless defined $default; |
| 1176 |
} |
| 1177 |
|
| 1178 |
# everything must be checked! |
| 1179 |
die "No check function for ${class}\->$attribute (type $type)" |
| 1180 |
unless (ref $check_func eq "CODE"); |
| 1181 |
|
| 1182 |
$types{$class}->{$attribute} = $type; |
| 1183 |
$check{$class}->{$attribute} = $check_func; |
| 1184 |
{ |
| 1185 |
no strict "refs"; |
| 1186 |
local ($^W) = 0; |
| 1187 |
|
| 1188 |
# build an appropriate "get_attribute" method, and |
| 1189 |
# define other per-type methods |
| 1190 |
my ($get_closure, $set_closure); |
| 1191 |
|
| 1192 |
# implement with closures for speed |
| 1193 |
if ( $type =~ m/i?set/ ) { |
| 1194 |
|
| 1195 |
# GET_$attribute (Set::Object) |
| 1196 |
$get_closure = sub { |
| 1197 |
my $self = shift; |
| 1198 |
if ( !defined $self->{$attribute} ) { |
| 1199 |
$self->{$attribute} = Set::Object->new(); |
| 1200 |
} |
| 1201 |
my $set = $self->{$attribute}; |
| 1202 |
( wantarray ? $set->members : $set ) |
| 1203 |
}; |
| 1204 |
|
| 1205 |
# and add a whole load of other functions too |
| 1206 |
for my $set_method (qw(includes insert size clear |
| 1207 |
remove)) { |
| 1208 |
|
| 1209 |
# ${attribute}_includes, etc |
| 1210 |
my $set_method_closure = sub { |
| 1211 |
my $self = shift; |
| 1212 |
$self->{$attribute} = Set::Object->new() |
| 1213 |
unless defined $self->{$attribute}; |
| 1214 |
return $self->{$attribute}->$set_method(@_); |
| 1215 |
}; |
| 1216 |
*{$class."::${attribute}_$set_method"} = |
| 1217 |
$set_method_closure unless |
| 1218 |
(defined &{$class."::${attribute}_$set_method"}); |
| 1219 |
} |
| 1220 |
|
| 1221 |
} elsif ( $type =~ m/i?array/ ) { |
| 1222 |
|
| 1223 |
# GET_$attribute (array) |
| 1224 |
# allow array slices, and return whole array |
| 1225 |
# in list context |
| 1226 |
$get_closure = sub { |
| 1227 |
my $array = ($_[0]->{$attribute} ||= []); |
| 1228 |
shift; |
| 1229 |
if ( @_ ) { |
| 1230 |
@{$array}[@_]; |
| 1231 |
} else { |
| 1232 |
( wantarray ? @{ $array } : $array ) |
| 1233 |
} |
| 1234 |
}; |
| 1235 |
|
| 1236 |
} elsif ( $type =~ m/i?hash/ ) { |
| 1237 |
# GET_$attribute (hash) |
| 1238 |
# allow hash slices, and return whole hash in |
| 1239 |
# list context |
| 1240 |
$get_closure = sub { |
| 1241 |
my $hash = ($_[0]->{$attribute} ||= {}); |
| 1242 |
shift; |
| 1243 |
if ( @_ ) { |
| 1244 |
@{$hash}{@_} |
| 1245 |
} else { |
| 1246 |
( wantarray ? %{ $hash } : $hash ); |
| 1247 |
} |
| 1248 |
}; |
| 1249 |
} else { |
| 1250 |
# GET_$attribute (scalar) |
| 1251 |
# return value only |
| 1252 |
$get_closure = sub { $_[0]->{$attribute}; }; |
| 1253 |
} |
| 1254 |
|
| 1255 |
*{$class."::get_$attribute"} = $get_closure |
| 1256 |
unless (defined &{$class."::get_$attribute"}); |
| 1257 |
|
| 1258 |
# SET_$attribute (all) |
| 1259 |
my $checkit = \$check{$class}->{$attribute}; |
| 1260 |
|
| 1261 |
# required hack for strings - duplicate the code |
| 1262 |
# to avoid the following string comparison for |
| 1263 |
# every set |
| 1264 |
if ( $type eq "string" ) { |
| 1265 |
$set_closure = sub { |
| 1266 |
my $self = shift; |
| 1267 |
my $value = shift; |
| 1268 |
eval { |
| 1269 |
if ( defined $value and length $value ) { |
| 1270 |
${$checkit}->(\$value); |
| 1271 |
} elsif ( $required ) { |
| 1272 |
die "value is required" |
| 1273 |
} elsif ( defined $required ) { |
| 1274 |
die "value must be defined" |
| 1275 |
unless defined $value; |
| 1276 |
} |
| 1277 |
}; |
| 1278 |
$@ && croak("value failed type check - ${class}->" |
| 1279 |
."set_$attribute('$value') ($@)"); |
| 1280 |
$self->{$attribute} = $value; |
| 1281 |
}; |
| 1282 |
} else { |
| 1283 |
$set_closure = sub { |
| 1284 |
my $self = shift; |
| 1285 |
my $value = shift; |
| 1286 |
eval { |
| 1287 |
if ( $value ) { |
| 1288 |
${$checkit}->(\$value); |
| 1289 |
} elsif ( $required ) { |
| 1290 |
die "value is required" |
| 1291 |
} elsif ( defined $required ) { |
| 1292 |
die "value must be defined" |
| 1293 |
unless defined $value; |
| 1294 |
} |
| 1295 |
}; |
| 1296 |
$@ && croak("value failed type check - ${class}->" |
| 1297 |
."set_$attribute('$value') ($@)"); |
| 1298 |
$self->{$attribute} = $value; |
| 1299 |
}; |
| 1300 |
} |
| 1301 |
|
| 1302 |
# now export them into the caller's namespace |
| 1303 |
my ($getter, $setter) |
| 1304 |
= ("get_$attribute", "set_$attribute"); |
| 1305 |
*{$class."::$getter"} = $get_closure |
| 1306 |
unless defined &{$class."::$getter"}; |
| 1307 |
*{$class."::$setter"} = $set_closure |
| 1308 |
unless defined &{$class."::$setter"}; |
| 1309 |
|
| 1310 |
*{$class."::$attribute"} = sub { |
| 1311 |
my $self = shift; |
| 1312 |
if ( @_ ) { |
| 1313 |
warn("The OO Police say change your call " |
| 1314 |
."to ->set_$attribute") if ($^W); |
| 1315 |
#goto $set_closure; # NO! BAD!! :-) |
| 1316 |
return $self->$setter(@_); |
| 1317 |
} else { |
| 1318 |
return $self->$getter(@_); |
| 1319 |
#goto $get_closure; |
| 1320 |
} |
| 1321 |
} unless defined &{$class."::$attribute"}; |
| 1322 |
} |
| 1323 |
|
| 1324 |
$cleaners{$class}->{$attribute} = $cleaner |
| 1325 |
if (defined $cleaner); |
| 1326 |
$init_defaults{$class}->{$attribute} = $default |
| 1327 |
if (defined $default); |
| 1328 |
$required_attributes{$class}->{$attribute} = $required |
| 1329 |
if (defined $required); |
| 1330 |
$attribute_options{$class}->{$attribute} = |
| 1331 |
( $options || {} ); |
| 1332 |
} |
| 1333 |
} |
| 1334 |
}; |
| 1335 |
|
| 1336 |
$@ && die "$@ while trying to import schema for $class"; |
| 1337 |
} |
| 1338 |
|
| 1339 |
=back |
| 1340 |
|
| 1341 |
=head2 Run-time type information |
| 1342 |
|
| 1343 |
It is possible to access the data structures that Class::Tangram uses |
| 1344 |
internally to verify attributes, create objects and so on. |
| 1345 |
|
| 1346 |
This should be considered a B<HIGHLY EXPERIMENTAL> interface to |
| 1347 |
B<INTERNALS> of Class::Tangram. |
| 1348 |
|
| 1349 |
Class::Tangram keeps seven internal hashes: |
| 1350 |
|
| 1351 |
=over |
| 1352 |
|
| 1353 |
=item C<%types> |
| 1354 |
|
| 1355 |
C<$types{$class}-E<gt>{$attribute}> is the tangram type of each attribute, |
| 1356 |
ie "ref", "iset", etc. See L<Tangram::Type>. |
| 1357 |
|
| 1358 |
=item C<%attribute_options> |
| 1359 |
|
| 1360 |
C<$attribute_options{$class}-E<gt>{$attribute}> is the options hash |
| 1361 |
for a given attribute. |
| 1362 |
|
| 1363 |
=item C<%required_attributes> |
| 1364 |
|
| 1365 |
C<$required_attributes{$class}-E<gt>{$attribute}> is the 'required' |
| 1366 |
option setting for a given attribute. |
| 1367 |
|
| 1368 |
=item C<%check> |
| 1369 |
|
| 1370 |
C<$check{$class}-E<gt>{$attribute}> is a function that will be passed |
| 1371 |
a reference to the value to be checked and either throw an exception |
| 1372 |
(die) or return true. |
| 1373 |
|
| 1374 |
=item C<%cleaners> |
| 1375 |
|
| 1376 |
C<$attribute_options{$class}-E<gt>{$attribute}> is a reference to a |
| 1377 |
destructor function for that attribute. It is called as an object |
| 1378 |
method on the object being destroyed, and should ensure that any |
| 1379 |
circular references that this object is involved in get cleared. |
| 1380 |
|
| 1381 |
=item C<%abstract> |
| 1382 |
|
| 1383 |
C<$abstract-E<gt>{$class}> is set if the class is abstract |
| 1384 |
|
| 1385 |
=item C<%init_defaults> |
| 1386 |
|
| 1387 |
C<$init_defaults{$class}-E<gt>{$attribute}> represents what an |
| 1388 |
attribute is set to automatically if it is not specified when an |
| 1389 |
object is created. If this is a scalar value, the attribute is set to |
| 1390 |
the value. If it is a function, then that function is called (as a |
| 1391 |
method) and should return the value to be placed into that attribute. |
| 1392 |
If it is a hash ref or an array ref, then that structure is COPIED in |
| 1393 |
to the new object. If you don't want that, you can do something like |
| 1394 |
this: |
| 1395 |
|
| 1396 |
[...] |
| 1397 |
flat_hash => { |
| 1398 |
attribute => { |
| 1399 |
init_default => sub { { key => "value" } }, |
| 1400 |
}, |
| 1401 |
}, |
| 1402 |
[...] |
| 1403 |
|
| 1404 |
Now, every new object will share the same hash for that attribute. |
| 1405 |
|
| 1406 |
=back |
| 1407 |
|
| 1408 |
There are currently four functions that allow you to access parts of |
| 1409 |
this information. |
| 1410 |
|
| 1411 |
=over |
| 1412 |
|
| 1413 |
=item Class::Tangram::attribute_options($class) |
| 1414 |
|
| 1415 |
Returns a hash ref to a data structure from attribute names to the |
| 1416 |
option hash for that attribute. |
| 1417 |
|
| 1418 |
=cut |
| 1419 |
|
| 1420 |
sub attribute_options($) { |
| 1421 |
my $class = shift; |
| 1422 |
return $attribute_options{$class}; |
| 1423 |
} |
| 1424 |
|
| 1425 |
=item Class::Tangram::attribute_types($class) |
| 1426 |
|
| 1427 |
Returns a hash ref from attribute names to the tangram type for that |
| 1428 |
attribute. |
| 1429 |
|
| 1430 |
=cut |
| 1431 |
|
| 1432 |
sub attribute_types($) { |
| 1433 |
my $class = shift; |
| 1434 |
return $types{$class}; |
| 1435 |
} |
| 1436 |
|
| 1437 |
=item Class::Tangram::required_attributes($class) |
| 1438 |
|
| 1439 |
Returns a hash ref from attribute names to the 'required' option setting for |
| 1440 |
that attribute. May also be called as a method, as in |
| 1441 |
C<$instance-E<gt>required_attributes>. |
| 1442 |
|
| 1443 |
=cut |
| 1444 |
|
| 1445 |
sub required_attributes($) { |
| 1446 |
my $class = ref $_[0] || $_[0]; |
| 1447 |
return $required_attributes{$class}; |
| 1448 |
} |
| 1449 |
|
| 1450 |
=item Class::Tangram::init_defaults($class) |
| 1451 |
|
| 1452 |
Returns a hash ref from attribute names to the default intial values for |
| 1453 |
that attribute. May also be called as a method, as in |
| 1454 |
C<$instance-E<gt>init_defaults>. |
| 1455 |
|
| 1456 |
=cut |
| 1457 |
|
| 1458 |
sub init_defaults($) { |
| 1459 |
my $class = ref $_[0] || $_[0]; |
| 1460 |
return $init_defaults{$class}; |
| 1461 |
} |
| 1462 |
|
| 1463 |
=item Class::Tangram::known_classes |
| 1464 |
|
| 1465 |
This function returns a list of all the classes that have had their |
| 1466 |
object schema imported by Class::Tangram. |
| 1467 |
|
| 1468 |
=cut |
| 1469 |
|
| 1470 |
sub known_classes { |
| 1471 |
return keys %types; |
| 1472 |
} |
| 1473 |
|
| 1474 |
=item Class::Tangram::is_abstract($class) |
| 1475 |
|
| 1476 |
This function returns true if the supplied class is abstract. |
| 1477 |
|
| 1478 |
=cut |
| 1479 |
|
| 1480 |
sub is_abstract { |
| 1481 |
my $class = shift; |
| 1482 |
$class eq "Class::Tangram" && ($class = shift); |
| 1483 |
|
| 1484 |
exists $cleaners{$class} or import_schema($class); |
| 1485 |
} |
| 1486 |
|
| 1487 |
=item Class::Tangram::class_bases($class) |
| 1488 |
|
| 1489 |
Returns an array ref of class names the given class inherits from. |
| 1490 |
|
| 1491 |
=cut |
| 1492 |
|
| 1493 |
sub class_bases($) { |
| 1494 |
my $class = shift; |
| 1495 |
return $bases{$class}; |
| 1496 |
} |
| 1497 |
|
| 1498 |
=item Class::Tangram::class_is_abstract($class) |
| 1499 |
|
| 1500 |
Returns a bool value if the given class is abstract. |
| 1501 |
|
| 1502 |
=cut |
| 1503 |
|
| 1504 |
sub class_is_abstract($) { |
| 1505 |
my $class = shift; |
| 1506 |
return exists $abstract{$class}; |
| 1507 |
} |
| 1508 |
|
| 1509 |
=item Class->set_init_default(attribute => $value); |
| 1510 |
|
| 1511 |
Sets the default value on an attribute for newly created "Class" |
| 1512 |
objects, as if it had been declared with init_default. Can be called |
| 1513 |
as a class or an instance method. |
| 1514 |
|
| 1515 |
=cut |
| 1516 |
|
| 1517 |
sub set_init_default { |
| 1518 |
my $invocant = shift; |
| 1519 |
my $class = ref $invocant || $invocant; |
| 1520 |
|
| 1521 |
exists $init_defaults{$class} or import_schema($class); |
| 1522 |
|
| 1523 |
while ( my ($attribute, $value) = splice @_, 0, 2) { |
| 1524 |
$init_defaults{$class}->{$attribute} = $value; |
| 1525 |
} |
| 1526 |
} |
| 1527 |
|
| 1528 |
=back |
| 1529 |
|
| 1530 |
=cut |
| 1531 |
|
| 1532 |
# a little embedded package |
| 1533 |
|
| 1534 |
package Tangram::Transient; |
| 1535 |
|
| 1536 |
BEGIN { |
| 1537 |
eval "use base qw(Tangram::Type)"; |
| 1538 |
if ( $@ ) { |
| 1539 |
# no tangram |
| 1540 |
} else { |
| 1541 |
$Tangram::Schema::TYPES{transient} = bless {}, __PACKAGE__; |
| 1542 |
} |
| 1543 |
} |
| 1544 |
|
| 1545 |
sub coldefs { } |
| 1546 |
|
| 1547 |
sub get_exporter { } |
| 1548 |
sub get_importer { } |
| 1549 |
|
| 1550 |
sub get_import_cols { |
| 1551 |
# print "Get_import_cols:" , Dumper \@_; |
| 1552 |
return (); |
| 1553 |
} |
| 1554 |
|
| 1555 |
=head1 SEE ALSO |
| 1556 |
|
| 1557 |
L<Tangram::Schema> |
| 1558 |
|
| 1559 |
B<A guided tour of Tangram, by Sound Object Logic.> |
| 1560 |
|
| 1561 |
http://www.soundobjectlogic.com/tangram/guided_tour/fs.html |
| 1562 |
|
| 1563 |
=head1 DEPENDENCIES |
| 1564 |
|
| 1565 |
The following modules are required to be installed to use |
| 1566 |
Class::Tangram: |
| 1567 |
|
| 1568 |
Set::Object => 1.02 |
| 1569 |
Pod::Constants => 0.11 |
| 1570 |
Test::Simple => 0.18 |
| 1571 |
Date::Manip => 5.21 |
| 1572 |
|
| 1573 |
Test::Simple and Date::Manip are only required to run the test suite. |
| 1574 |
|
| 1575 |
If you find Class::Tangram passes the test suite with earlier versions |
| 1576 |
of the above modules, please send me an e-mail. |
| 1577 |
|
| 1578 |
=head2 MODULE RELEASE |
| 1579 |
|
| 1580 |
This is Class::Tangram version 1.13. |
| 1581 |
|
| 1582 |
=head1 BUGS/TODO |
| 1583 |
|
| 1584 |
There should be more functions for breaking loops; in particular, a |
| 1585 |
standard function called C<drop_refs($obj)>, which replaces references |
| 1586 |
to $obj with the appropriate C<Tangram::RefOnDemand> object so that an |
| 1587 |
object can be unloaded via C<Tangram::Storage->unload()> and actually |
| 1588 |
have a hope of being reclaimed. Another function that would be handy |
| 1589 |
would be a deep "mark" operation for manual mark & sweep garbage |
| 1590 |
collection. |
| 1591 |
|
| 1592 |
Need to think about writing some functions using C<Inline> for speed. |
| 1593 |
One of these days... |
| 1594 |
|
| 1595 |
Allow C<init_default> values to be set in a default import function? |
| 1596 |
|
| 1597 |
ie |
| 1598 |
|
| 1599 |
use MyClassTangramObject -defaults => { foo => "bar" }; |
| 1600 |
|
| 1601 |
=head1 AUTHOR |
| 1602 |
|
| 1603 |
Sam Vilain, <sam@vilain.net> |
| 1604 |
|
| 1605 |
=head2 CREDITS |
| 1606 |
|
| 1607 |
# Some modifications |
| 1608 |
# Copyright © 2001 Micro Sharp Technologies, Inc., Vancouver, WA, USA |
| 1609 |
# Author: Karl M. Hegbloom <karlheg@microsharp.com> |
| 1610 |
# Perl Artistic Licence. |
| 1611 |
|
| 1612 |
Many thanks to Charles Owens and David Wheeler for their feedback, |
| 1613 |
ideas, patches and bug testing. |
| 1614 |
|
| 1615 |
=cut |
| 1616 |
|
| 1617 |
69; |
| 1618 |
|
| 1619 |
__END__ |
| 1620 |
|
| 1621 |
# From old SYNOPSIS, I decided it was too long. A lot of |
| 1622 |
# the information here needs to be re-integrated into the |
| 1623 |
# POD. |
| 1624 |
|
| 1625 |
package Project; |
| 1626 |
|
| 1627 |
# here's where we build the individual object schemas into |
| 1628 |
# a Tangram::Schema object, which the Tangram::Storage |
| 1629 |
# class uses to know which tables and columns to find |
| 1630 |
# objects. |
| 1631 |
use Tangram::Schema; |
| 1632 |
|
| 1633 |
# TIMTOWTDI - this is the condensed manpage version :) |
| 1634 |
my $dbschema = Tangram::Schema->new |
| 1635 |
({ classes => |
| 1636 |
[ 'Orange' => { fields => $Orange::fields }, |
| 1637 |
'MyObject' => { fields => $MyObject::schema }, ]}); |
| 1638 |
|
| 1639 |
sub schema { $dbschema }; |
| 1640 |
|
| 1641 |
package main; |
| 1642 |
|
| 1643 |
# See Tangram::Relational for instructions on using |
| 1644 |
# "deploy" to create the database this connects to. You |
| 1645 |
# only have to do this if you want to write the objects to |
| 1646 |
# a database. |
| 1647 |
use Tangram::Relational; |
| 1648 |
my ($dsn, $u, $p); |
| 1649 |
my $storage = Tangram::Relational->connect |
| 1650 |
(Project->schema, $dsn, $u, $p); |
| 1651 |
|
| 1652 |
# Create an orange |
| 1653 |
my $orange = Orange->new( |
| 1654 |
juiciness => 8, |
| 1655 |
type => 'Florida', |
| 1656 |
tag => '', # required |
| 1657 |
); |
| 1658 |
|
| 1659 |
# Store it |
| 1660 |
$storage->insert($orange); |
| 1661 |
|
| 1662 |
# This is how you get values out of the objects |
| 1663 |
my $juiciness = $orange->juiciness; |
| 1664 |
|
| 1665 |
# a "ref" must be set to a blessed object, any object |
| 1666 |
my $grower = bless { name => "Joe" }, "Farmer"; |
| 1667 |
$orange->set_grower ($grower); |
| 1668 |
|
| 1669 |
# these are all illegal - type checking is fairly strict |
| 1670 |
my $orange = eval { Orange->new; }; print $@; |
| 1671 |
eval { $orange->set_juiciness ("Yum"); }; print $@; |
| 1672 |
eval { $orange->set_segments (31); }; print $@; |
| 1673 |
eval { $orange->set_grower ("Mr. Nice"); }; print $@; |
| 1674 |
|
| 1675 |
# Demonstrate some "required" functionality |
| 1676 |
eval { $orange->set_type (''); }; print $@; |
| 1677 |
eval { $orange->set_type (undef); }; print $@; |
| 1678 |
eval { $orange->set_tag (undef); }; print $@; |
| 1679 |
|
| 1680 |
# this works too, but is slower |
| 1681 |
$orange->get( "juiciness" ); |
| 1682 |
$orange->set( juiciness => 123, |
| 1683 |
segments => 17 ); |
| 1684 |
|
| 1685 |
# Re-configure init_default - make each new orange have a |
| 1686 |
# random juiciness |
| 1687 |
$orange->set_init_default( juiciness => sub { int(rand(45)) } ); |