package FileHash; use strict; use warnings; use Tie::Hash; use Data::Dumper; our @ISA = ("Tie::Hash"); sub TIEHASH { my ($class, $file) = @_; my $s = {}; $s->{DATA} = _read($file) || {}; $s->{FILE} = $file; bless $s, $class; } sub STORE { $_[0]->{DATA}{$_[1]} = $_[2] } sub CLEAR { $_[0]->{DATA} = () } sub FETCH { $_[0]->{DATA}{$_[1]} } sub FIRSTKEY { my $a = scalar keys %{$_[0]}; each %{$_[0]->{DATA}} } sub NEXTKEY { each %{$_[0]->{DATA}} } sub EXISTS { exists $_[0]->{DATA}{$_[1]} } sub DELETE { delete $_[0]->{DATA}{$_[1]} } sub DESTROY { _write ($_[0]) } # ----- private subs ----- sub _write { # still need to deal with race conditions my %data = %{$_[0]->{DATA}}; my $file = $_[0]->{FILE} ; open (FH, "> $file") or die "Can't open $file for FileHash _write: $!"; print FH Data::Dumper->Dump([\%data], ['*data']); close FH or die "Can't close $file for FileHash _write: $!"; } sub _read { # still need to deal with race conditions no strict; my $file = shift; return undef unless -f $file; unless (my $ret = do $file) { warn "couldn't parse $file: $@" if $@; warn "couldn't do $file: $!" unless defined $ret; } return \%data; } 1;