Rose::HTMLx::Form::Related::RDBO::Metadata - RDBO metadata driver


Rose-HTMLx-Form-Related documentation Contained in the Rose-HTMLx-Form-Related distribution.

Index


Code Index:

NAME

Top

Rose::HTMLx::Form::Related::RDBO::Metadata - RDBO metadata driver

SYNOPSIS

Top

 see Rose::HTMLx::Form::Related::Metadata

METHODS

Top

Only overriden methods are described here.

discover_relationships

Implements RDBO relationship introspection.

AUTHOR

Top

Peter Karman, <karman at cpan.org>

BUGS

Top

Please report any bugs or feature requests to bug-rose-htmlx-form-related at rt.cpan.org, or through the web interface at http://rt.cpan.org/NoAuth/ReportBug.html?Queue=Rose-HTMLx-Form-Related. I will be notified, and then you'll automatically be notified of progress on your bug as I make changes.

SUPPORT

Top

You can find documentation for this module with the perldoc command.

    perldoc Rose::HTMLx::Form::Related

You can also look for information at:

* AnnoCPAN: Annotated CPAN documentation

http://annocpan.org/dist/Rose-HTMLx-Form-Related

* CPAN Ratings

http://cpanratings.perl.org/d/Rose-HTMLx-Form-Related

* RT: CPAN's request tracker

http://rt.cpan.org/NoAuth/Bugs.html?Dist=Rose-HTMLx-Form-Related

* Search CPAN

http://search.cpan.org/dist/Rose-HTMLx-Form-Related

ACKNOWLEDGEMENTS

Top

The Minnesota Supercomputing Institute http://www.msi.umn.edu/ sponsored the development of this software.

COPYRIGHT & LICENSE

Top


Rose-HTMLx-Form-Related documentation Contained in the Rose-HTMLx-Form-Related distribution.
package Rose::HTMLx::Form::Related::RDBO::Metadata;
use strict;
use base qw( Rose::HTMLx::Form::Related::Metadata );
use Carp;
use Data::Dump qw( dump );
use MRO::Compat;
use mro 'c3';

our $VERSION = '0.22';

sub discover_relationships {
    my $self = shift;

    my $debug = $self->form->debug;

    # if running under Catalyst (e.g.) get controller info
    my $app = $self->form->app_class || $self->form->app;

    # get relationship objects from RDBO
    my %seen;
    my @fks = $self->object_class->meta->foreign_keys;
    my @rel = $self->object_class->meta->relationships;
    my @rels;
    for my $r ( @fks, @rel ) {

        # screen out duplicates since FKs masquerade as Rels
        next if $seen{ $r->name }++;
        push( @rels, $r );
    }

    my @relinfos;

    # create our RelInfo objects
    for my $rdbo_rel (@rels) {

        my $relinfo = $self->relinfo_class->new;
        my $name    = $rdbo_rel->name;
        my $type    = $rdbo_rel->type;
        $relinfo->object_class( $self->object_class );
        $relinfo->name($name);
        $relinfo->method($name);
        $relinfo->type($type);
        $relinfo->label(
            $self->labels->{$name} || join(
                ' ', map { ucfirst($_) }
                    split( m/_/, $name )
            )
        );

        $debug and carp dump $relinfo;

        if ( $type eq 'many to many' ) {
            my $map_to    = $rdbo_rel->map_to;
            my $map_class = $rdbo_rel->map_class;
            $debug and carp "map_to = $map_to";
            $debug and carp "map_class = $map_class";
            $debug and carp dump $map_class->meta;
            my $foreign_rel = $map_class->meta->relationship($map_to);
            my $local_rel
                = $map_class->meta->relationship( $rdbo_rel->map_from );
            my @forcolmap = %{ $foreign_rel->column_map };
            $debug and carp dump \@forcolmap;
            my @loccolmap = %{ $local_rel->column_map };
            $relinfo->map_class($map_class);
            $relinfo->foreign_class( $foreign_rel->class );
            $relinfo->map_to($map_to);
            $relinfo->map_to_column( $forcolmap[0] );
            $relinfo->map_from_column( $loccolmap[0] );
            $relinfo->map_from( $rdbo_rel->map_from );
        }
        else {
            $relinfo->foreign_class( $rdbo_rel->class );
            $relinfo->cmap( { $rdbo_rel->column_map } );
        }

        if ($app) {

            $relinfo->app($app);

            # create URL and controller if available.
            my $prefix          = $self->object_class->schema_class_prefix;
            my $controller_name = $relinfo->foreign_class;
            if ( !$controller_name ) {
                croak "no foreign class in relinfo: " . dump $relinfo;
            }
            $controller_name =~ s/^${prefix}:://;
            $relinfo->controller_class(
                join( '::',
                    grep { defined($_) }
                        ( $self->controller_prefix, $controller_name ) )
            );
            if ( $relinfo->map_class ) {
                my $map_class_prefix
                    = $relinfo->map_class->schema_class_prefix;
                my $controller_name = $relinfo->map_class;
                $controller_name =~ s/^${map_class_prefix}:://;
                $relinfo->map_class_controller_class(
                    join( '::',
                        grep { defined($_) }
                            ( $self->controller_prefix, $controller_name ) )
                );
            }

            # only want a controller instance if $app is fully
            # initialized (not a class name)
            if ( ref $app ) {
                $relinfo->controller(
                    $app->controller( $relinfo->controller_class ) );
            }

        }

        push( @relinfos, $relinfo );

    }

    $self->relationships( \@relinfos );

}

sub show_related_field_using {
    my $self   = shift;
    my $fclass = shift or croak "foreign_object_class required";
    my $field  = shift or croak "field_name required";

    my $method = $self->next::method( $fclass, $field );
    return $method if $method;

    # find the first single-column unique char/varchar method name
    my @ukeys = $fclass->meta->unique_keys_column_names;
    if (@ukeys) {
        for my $k (@ukeys) {
            if ( scalar(@$k) == 1
                && $fclass->meta->column( $k->[0] )->type =~ m/char/ )
            {
                return $k->[0];    # TODO column alias ??
            }
        }
    }
    return undef;
}

1;

__END__