#!/usr/bin/perl
#
# LJ::Cache class
# See perldoc documentation at the end of this file.
#
# -------------------------------------------------------------------------
#
# This package is released under the LGPL (GNU Library General Public License)
#
# A copy of the license has been included with the software as LGPL.txt.  
# If not, the license is available at:
#      http://www.gnu.org/copyleft/library.txt
#
# -------------------------------------------------------------------------
#

package LJ::Cache;

use strict;
use vars qw($VERSION @ISA @EXPORT @EXPORT_OK);

require Exporter;
require AutoLoader;

@ISA = qw(Exporter AutoLoader);
# Items to export into callers namespace by default. Note: do not export
# names by default without a very good reason. Use EXPORT_OK instead.
# Do not simply export all your public functions/methods/constants.
@EXPORT = qw(
	
);
$VERSION = '0.02';

# Preloaded methods go here.

sub new {
    my ($class, $args) = @_;
    my $self = {};
    bless $self, ref $class || $class;
    
    $self->init($args);
    return $self;
}

sub init {
    my ($self, $args) = @_;

    $self->{'head'} = 0;
    $self->{'tail'} = 0;
    $self->{'items'} = {};
    $self->{'size'} = 0;
    $self->{'maxsize'} = $args->{'maxsize'};
}

sub validate_list
{
    my ($self, $source) = @_;
    print "Validate list: $self->{'size'} (max: $self->{'maxsize'})\n";
    
    my $count = 1;
    if ($self->{'size'} && ! defined $self->{'head'}) {
	die "$source: no head pointer\n";
    }
    if ($self->{'size'} && ! defined $self->{'tail'}) {
	die "$source: no tail pointer\n";
    }
    if ($self->{'size'}) {
	print "  head: $self->{'head'}\n";
	print "  tail: $self->{'tail'}\n";
    }

    my $iter = $self->{'head'};
    my $last = undef;
    while ($count <= $self->{'size'}) {
	if (! defined $iter) {
	    die "$source: undefined iterator\n";
	}
	my $item = $self->{'items'}->{$iter};
	unless (defined $item) {
	    die "$source: item '$iter' isn't in items\n";
	}
	my $prevtext = $item->[0] || "--";
	my $nexttext = $item->[2] || "--";
	print "  #$count ($iter): [$prevtext, $item->[1], $nexttext]\n";
	if ($count == 1 && defined($item->[0])) {
	    die "$source: Head element shouldn't have previous pointer!\n";
	}
	if ($count == $self->{'size'} && defined($item->[2])) {
	    die "$source: Last element shouldn't have next pointer!\n";
	}
	if (defined $last && ! defined $item->[0]) {
	    die "$source: defined \$last but not defined previous pointer.\n";
	}
	if (! defined $last && defined $item->[0]) {
	    die "$source: not defined \$last but previous pointer defined.\n";
	}
	if (defined $item->[0] && defined $last && $item->[0] ne $last)
	{
	    die "$source: Previous pointer is wrong.\n";
	}

	$last = $iter;
	$iter = defined $item->[2] ? $item->[2] : undef;
	$count++;
    }
}

sub drop_tail
{
    my $self = shift;

    ## who's going to die?
    my $to_die = $self->{'tail'};

    ## set the tail to the item before the one dying.
    $self->{'tail'} = $self->{'items'}->{$to_die}->[0];

    ## adjust the forward pointer on the tail to be undef
    if (defined $self->{'tail'}) {
	undef $self->{'items'}->{$self->{'tail'}}->[2];
    }

    ## kill the item
    my $presize = scalar(keys %{$self->{'items'}});
    delete $self->{'items'}->{$to_die};
    my $postsize = scalar(keys %{$self->{'items'}});
    unless ($postsize == $presize-1) {
	die "Tail drop didn't work!\n";
    }

    ## shrink the overall size
    $self->{'size'}--;

}

sub print_list {
    my ($self) = @_;
    print "Size: $self->{'size'} (max: $self->{'maxsize'})\n";

    my $count = 1;
    my $iter = $self->{'head'};
    while (defined $iter) { #$count <= $self->{'size'}) {
	my $item = $self->{'items'}->{$iter};
	print "$count: $iter = $item->[1]\n";
	$iter = $item->[2];
 	$count++;
    }
}

sub get {
    my ($self, $key) = @_;

    if (exists $self->{'items'}->{$key}) 
    {
	my $item = $self->{'items'}->{$key};

	# promote this to the head
	unless ($self->{'head'} eq $key)
	{
	    if ($self->{'tail'} eq $key) {
		$self->{'tail'} = $item->[0];
	    }
	    # remove this element from the linked list.
	    my $next = $item->[2];
	    my $prev = $item->[0];
	    if (defined $next) { $self->{'items'}->{$next}->[0] = $prev; }
	    if (defined $prev) { $self->{'items'}->{$prev}->[2] = $next; }
	    
	    # make current head point backwards to this item
	    $self->{'items'}->{$self->{'head'}}->[0] = $key;
	    
	    # make this item point forwards to current head, and backwards nowhere
	    $item->[2] = $self->{'head'};
	    undef $item->[0];
	    
	    # make this the new head
	    $self->{'head'} = $key;
	}
	
	return $item->[1];
    }
    return undef;
}

sub set {
    my ($self, $key, $value) = @_;
    
    $self->drop_tail() if ($self->{'maxsize'} && 
			   $self->{'size'} == $self->{'maxsize'} &&
			   ! exists $self->{'items'}->{$key});
    
    if (exists $self->{'items'}->{$key}) {
	# update the value
	my $item = $self->{'items'}->{$key};
	$item->[1] = $value;
    }
    else {
	# stick it at the end, for now
	$self->{'items'}->{$key} = [undef, $value, undef];
	if ($self->{'size'}) {
	    $self->{'items'}->{$self->{'tail'}}->[2] = $key;
	    $self->{'items'}->{$key}->[0] = $self->{'tail'};
	} else {
	    $self->{'head'} = $key;
	}
	$self->{'tail'} = $key;
	$self->{'size'}++;
    }

    # this will promote it to the top:
    $self->get($key);
}

# Autoload methods go after =cut, and are processed by the autosplit program.

1;
__END__

=head1 NAME

LJ::Cache - LRU Cache

=head1 SYNOPSIS

  use LJ::Cache;
  my $cache = new LJ::Cache { 'maxsize' => 20 };
  my $value = $cache->get($key);
  unless (defined $value) {
      $val = "load some value";
      $cache->set($key, $value);
  }

=head1 DESCRIPTION

This class implements an LRU dictionary cache.  The two operations on it
are get() and set(), both of which promote the key being referenced to
the "top" of the cache, so it will stay alive longest.

When the cache is full and and a new item needs to be added, the oldest
one is thrown away.

You should be able to regenerate the data at any time, if get() 
returns undef.

This class is useful for caching information from a slower data source
while also keeping a bound on memory usage.

=head1 AUTHOR

Brad Fitzpatrick, bradfitz@bradfitz.com

=head1 SEE ALSO

perl(1).

=cut


syntax highlighted by Code2HTML, v. 0.9.1