The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
package Pixie::Store;

use strict;
our $VERSION="2.06";
my %typemap = ( memory => 'Pixie::Store::Memory',
                bdb => 'Pixie::Store::BerkeleyDB',
                dbi => 'Pixie::Store::DBI', );

use Scalar::Util qw/weaken/;
use Carp;

#use overload
#  '""' => 'as_string';

sub as_string {
  my $proto = shift;
  my $str;
  if (ref $proto) {
    $str .= ref($proto);
    $str .= ": $proto->{spec}" if $proto->{spec};
  }
  else {
    $str .= $proto;
  }
  return $str;
}

sub connect {
  my $proto = shift;
  my $spec = shift;
  my($type, $path) = split(':', $spec, 2);

  $type = lc($type);
  die "Invalid database spec" unless exists $typemap{$type};

  eval "require " . $typemap{$type};
  die $@ if $@;

  my $self = $typemap{$type}->connect($path,@_);
  $self->{spec} = $spec;
  return $self;
}

sub object_graph_for {
  my $self = shift;
  my $pixie = shift;

  my $graph = $pixie->get_object_named('PIXIE::Node Graph');
  unless ($graph) {
    $graph = Pixie::ObjectGraph->new;
    $pixie->bind_name('PIXIE::Node Graph' => $graph);
  }
  return $graph;
}


sub remove_from_store {
  my $self = shift;
  my($oid) = @_;

  $self->remove_from_rootset($oid)
       ->_delete($oid);
}

# Low level locking

sub locked_set {
    my $self = shift;
    return $self->{locked_set} ||= {};
}

sub lock_object_for {
    my $self = shift;
    my($oid, $locker) = @_;
    $self->locked_set->{ $oid } = $locker->_oid;
    return 1;
}

sub unlock_object_for {
    my $self = shift;
    my $oid = shift;
    delete $self->locked_set->{$oid};
}

sub release_all_locks {
    my $self = shift;
    my $locked_set = $self->locked_set;
    $self->unlock_object_for(@$_)
        for map {[$_ => $locked_set->{$_}]}
            keys %$locked_set;
    return $self;
}

sub add_to_rootset {
  my $self = shift;
  my $thing = shift;
  $self->_add_to_rootset($thing) unless $self->is_hidden($thing);
}

sub is_hidden {
  my $self = shift;
  my $thing = shift;

  $thing->PIXIE::oid =~ /^<NAME:PIXIE::/;
}

sub _add_to_rootset { $_[0]->subclass_responsibility(@_) }

sub remove_from_rootset {
  $_[0]->subclass_responsibility($_[0]);
}

sub rootset {
  $_[0]->subclass_responsibility;
}

sub lock     { $_[0]->subclass_responsibility(@_) }
sub unlock   { $_[0]->subclass_responsibility(@_) }
sub rollback { $_[0]->subclass_responsibility(@_) }


sub lock_for_GC {
  my $self = shift;
  $self->lock;
}

sub unlock_after_GC {
  my $self = shift;
  $self->unlock;
}

sub subclass_responsibility {
  my $self = shift;
  require Carp;
  Carp::carp( (caller(1))[3], " not implemented for ", ref($self) );
  return wantarray ? @_ : $_[-1];
}

sub DESTROY {
    my $self = shift;
    $self->release_all_locks;
}

1;

__END__

=head1 NAME

Pixie::Store -- Abstract interface to physical storage

=head1 SYNOPSIS

In a deploy script:

  use Pixie::Store::DBI;

  # Setup the datastore.
  Pixie::Store::DBI->deploy('dbi:mysql:dbname=foo',
                            user => 'wibble',
                            pass => 'plib',
                            object_table => 'object');

In a pixie client:

  use Pixie::Store::MySubclass;
  use Pixie;

  my $pixie = Pixie->connect('prefix:myspec',
                             user => 'bill',
                             pass => 'flobadob');


=head1 DESCRIPTION

Pixie::Store provides pixie with an abstracted interface to the
physical storage used to actually store the objects that Pixie
manages. It is not a 'public' class; most Pixie users will never have
to touch it except maybe to call the C<deploy> method of an
appropriate subclass.

However, if you want to add another storage medium to Pixie, start
here. (If you want to add specific methods for storing in a particular
RDBMS, you should take a look at L<DBIx::AnyDBD> before diving into
Pixie::Store::DBI::Default and its woefully underdocumented friends.

=head2 The Public Interface

There is no public interface to Pixie::Store. However, where
appropriate, Pixie::Store subclasses may implement a C<deploy> method
which should be responsible for setting up a suitable storage
structure which can be connected to later. 

=head2 The Subclassable Interface

Pixie::Store implements almost no methods itself, except for a
'connect' factory method, which takes a 'storage spec' (similar in
form to the classic DBI data source spec), works out which concrete
subclass to use for the real connection, loads it if necessary and
uses that to build a store object.

But Pixie proper depends on the following methods existing and working
as described.

=over 4

=item connect(SPEC, @ARGS)

Makes the actual connection and returns an object of the appropriate
class. The only fixed part of the interface is that the storage spec
shall come first, and the only fixed part of that is that storage
specs tend to look like 'id:...'. The 'id:' tag is used by
C<Pixie::Store::connect> to identify which subclass to
instantiate. L</The Typemap> has more details of how that works.

=item clear

Empties the datastore, removes all stored objects and any associated
metadata. Use with caution. (It is remarkably handy when one is
writing test scripts though...)

=item store_at( OID, FLATTENED_OBJECT )

Take the FLATTENED_OBJECT and stash it where it can be found via the
given OID. The FLATTENED_OBJECT is guaranteed to be an arbitrarily
long string of bytes (just to make life easy...). An OID is a string
of up to 255 characters. Overwrites any existing entry at that OID.

=item get_object_at( OID )

Returns the object associated with the given OID if it exists; returns
undef/the empty list if no object can be found and throws an exception
if it finds more than one object associated with that OID. (OIDs are
supposed to be unique after all).

=item delete( OID )

Deletes the object associated with OID. Returns true if an object
existed, or false if there was no such object.

=item lock

Possibly misnamed. Locks the database so that nobody else can
interfere. (Actually, it is often implemented as 'begin
transaction'...).

=item unlock

Again, possibly misnamed. Ensures that all the changes that have been
inserted really have been inserted, and frees the database for other
users. Should possibly be called 'commit'.

=item rollback

Rolls the database back to the state it was in at the last 'lock'. Not
misnamed. (Hurrah).

=back

=head2 The Typemap

Once you have subclassed Pixie::Store you need to let it know about
your new subclass so it can make C<connect> work. To do that, pick an
appropriate prefix string to identify your subclass and add something
like the following -- after the C<use base 'Pixie::Store';>
part, or things will break -- to your code:

  $Pixie::Store::typemap{prefix} = __PACKAGE__;

Once you have done this, the code given in the synopsis should work,
as if by magic.

=head1 AUTHORS

James Duncan <james@fotango.com>, Piers Cawley <pdcawley@bofh.org.uk>
and Leon Brocard <acme@astray.com>.

=head1 COPYRIGHT

Copyright 2002 Fotango Ltd

This software is released under the same license as Perl itself.