# ================================================================== # Gossamer Threads Module Library - http://gossamer-threads.com/ # # GT::SQL::Table # Author: Jason Rhinelander # CVS Info : # $Id: Rebuild.pm,v 1.10 2005/04/06 23:11:08 jagerman Exp $ # # Copyright (c) 2004 Gossamer Threads Inc. All Rights Reserved. # ================================================================== # # Description: # This goes hand in hand with GT::SQL::Tree and is very useful in # turning an existing table without the root, and/or depth columns # into a GT::SQL::Tree-compatible format. # package GT::SQL::Tree::Rebuild; # =============================================================== use strict; use vars qw/$DEBUG $VERSION $ERROR_MESSAGE @ISA $AUTOLOAD/; use constants TREE_COLS_ROOT => 0, TREE_COLS_FATHER => 1, TREE_COLS_DEPTH => 2; @ISA = qw/GT::SQL::Base/; $DEBUG = 0; $VERSION = sprintf "%d.%03d", q$Revision: 1.10 $ =~ /(\d+)\.(\d+)/; $ERROR_MESSAGE = 'GT::SQL'; # New returns a GT::SQL::Tree::Rebuild object when you can pass to GT::SQL::Tree. # When you are adding a tree to an existing table, but the table does not have # the root and/or depth columns, you get a Rebuild object, then pass it to # ->add_tree so that your tree can be built anyway. # You need to call new with the following options: # table => $Table_object # missing_root => sub { ... }, # Only if you are missing the root. The code reference should return the pk of the root. # missing_depth => 1, # Only if you are missing the depth. The code reference should return the depth of the node. # missing_father => 1, # Only if you are missing the father. The code reference should return the pk of the father. # cols => [...], # The columns you want %row (discussed below) to contain # # The code references are passed two arguments: # \%row, # A row from the table. If using the cols option, it will only have those columns. # $table_object, # This is the same object you pass to new() # \%all # This is a hash reference of all rows; each key is a primary key, each value the row. This may or may not be of use to you. # # For depth, %all will have root and father ids set, for roots father ID's will be set. # # NOTE: The father, root, and depth columns must exist beforehand. sub new { my $this = shift; my $opts = $this->common_param(@_) or return $this->error(BADARGS => FATAL => '$obj->new(HASH or HASH REF)'); my $self = bless {}, $this; $self->{table} = $opts->{table} or return $self->error(BADARGS => FATAL => '$obj->new({ ... table => $table_obj ... })'); for (qw(missing_root missing_depth missing_father)) { next unless exists $opts->{$_}; $self->{$_} = $opts->{$_}; ref $self->{$_} eq 'CODE' or return $self->error(BADARGS => FATAL => '$obj->new({ ... ' . $_ . ' => sub { ... } ... })'); } $self->{cols} = $opts->{cols} if $opts->{cols}; $self->{cols} = [$self->{cols}] if $self->{cols} and not ref $self->{cols}; $self->{cols} ||= []; $self->{order_by} = $opts->{order_by} if $opts->{order_by}; $self->{missing_root} or $self->{missing_depth} or $self->{missing_father} or return $self->error(BADARGS => FATAL => 'At least one of "missing_root", "missing_depth", or "missing_father" must be passed to $obj->new({ ... })'); $self->{_debug} = $opts->{debug} || $DEBUG || 0; $self; } # Called internally by the GT::SQL::Tree object. This does all the calculations. # Note that this only rebuilds the table itself, GT::SQL::Tree->create will still # have to create its tree table. sub _rebuild { my ($self, $pk, $root_col, $father_col, $depth_col) = @_; my $table = $self->{table}; my $count = $table->count(); for (my $i = 0; $i < $count; $i += 10000) { $table->select_options("ORDER BY $self->{order_by}") if exists $self->{order_by}; $table->select_options("LIMIT 10000" . ($i ? " OFFSET $i" : "")); my $sth = $table->select(@{$self->{cols}}); while (my $row = $sth->fetchrow_hashref) { my %update; if ($self->{missing_father}) { my $father_id = $self->{missing_father}->($row, $table); $update{$father_col} = $father_id unless $row->{$father_col} == $father_id; $row->{$father_col} = $father_id; } if ($self->{missing_root}) { my $root_id = $self->{missing_root}->($row, $table); $update{$root_col} = $root_id unless $row->{$root_col} == $root_id; $row->{$root_col} = $root_id; } if ($self->{missing_depth}) { my $depth = $self->{missing_depth}->($row, $table); $update{$depth_col} = $depth unless $row->{$depth_col} == $depth; $row->{$depth_col} = $depth; } $table->update(\%update, { $pk => $row->{$pk} }) if keys %update; # If the new value(s) is/are 0, like the default value(s), %update will be empty } } return 1; } 1; __END__ =head1 NAME GT::SQL::Tree::Rebuild - Helps to turn a table into one usable by GT::SQL::Tree. =head1 SYNOPSIS use GT::SQL::Tree; use GT::SQL::Tree::Rebuild; my $rebuild = GT::SQL::Tree::Rebuild->new( table => $DB->table('MyTable'), missing_root => \&root_code, missing_father => \&father_code, missing_depth => \&depth_code, order_by => 'column_name' ); $DB->editor('MyTable')->add_tree(root => $root_col, father => $father_col, depth => $depth_col, rebuild => $rebuild); =head1 DESCRIPTION GT::SQL::Tree::Rebuild is designed to go hand-in-hand with GT::SQL::Tree and aids in turning an existing table into one with the neccessary root, father and depth columns needed by GT::SQL::Tree. The main purpose is to do a one-shot conversion of a table to make it compatible with GT::SQL::Tree. =head2 new - Create a Rebuild object There is only one method that is called - new. You pass the arguments needed and get back a GT::SQL::Tree::Rebuild object. This object should then be passed into GT::SQL::Tree->create (typically via C<$editor-Eadd_tree()>) new() takes a hash with up to 4 argument pairs: "table" (required), and one or more of "missing_root", "missing_father", or "missing_depth". The values are explained below. =over 4 =item table Required. You specify the table object for the table to rebuild. For example, if you are going to add a tree to the "Category" table, you provide the "Category" table object here. =item cols By default, an entire row will be returned. To speed up the process and lower the memory usage, you can use the C option, which specifies the columns to select for $row. It is recommended that you only select columns that you need as doing so will definately save time and memory. =item missing_father, missing_root, missing_depth Each of these arguments takes a code reference as its value. The arguments to the code references are as follows: =over 4 =item $row The first argument is a hash reference of the row being examined. Your job, in the code reference, is to examine $row and determine the missing value, depending on which code reference is being called. missing_root needs to return the root_id for this row; missing_father needs to return the father_id, and the missing_depth code reference should return the depth for the row. =item $table The second argument passed to the code references is the same table object that you pass into new(), which you can select from if neccessary. =back =item missing_father The C code reference is called first - before C and C. The code reference is called as described above and should return the ID of the father of the row passed in. A false return (0 or undef) is interpreted as meaning that this is a root and therefore has no father. =item missing_root C has to return the root of the row passed in. This is called after C, so the $row will contain whatever you returned in C in the father ID column. Of course, this only applies if using both C and C. =item missing_depth C has to return the depth of the row passed in. This is called last, so if you are also using C and/or C, you will have whatever was returned by those code refs available in the $row. =item order_by The query done to retrieve records can be sorted using the C option. It should be anything valid for "ORDER BY _____". Often it can be useful to have your results returned in a certain order - for example: order_by => 'depth_column ASC' would insure that parents come before roots. Of course, this example wouldn't work if you are using "missing_depth" since none of the depth values will be set. =back Once you have a GT::SQL::Tree::Rebuild object, you should pass it into Ccreate> (which typically involves passing it into C<$editor-Eadd_tree()>, which passed it through). Before calculating the tree, GT::SQL::Tree will call on the rebuild object to reproduce the father, root, and/or depth columns (whichever you specified). =head1 COPYRIGHT Copyright (c) 2004 Gossamer Threads Inc. All Rights Reserved. http://www.gossamer-threads.com/ =head1 VERSION Revision: $Id: Rebuild.pm,v 1.10 2005/04/06 23:11:08 jagerman Exp $ =cut