package App::Hashl; use strict; use warnings; use 5.010; use Digest::SHA; use Storable qw(nstore retrieve); our $VERSION = '1.00'; sub new { my ( $obj, %conf ) = @_; my $ref = { files => {}, ignored => {}, }; $ref->{config} = \%conf; $ref->{config}->{read_size} //= ( 2**20 ) * 4; # 4 MiB $ref->{version} = $VERSION; return bless( $ref, $obj ); } sub new_from_file { my ( $obj, $file ) = @_; my $ref = retrieve($file); if ( not defined $ref->{version} ) { $ref->{version} = '1.00'; } return bless( $ref, $obj ); } sub si_size { my ( $self, $bytes ) = @_; my @post = ( q{ }, qw(k M G T) ); while ( $bytes >= 1024 ) { $bytes /= 1024; shift @post; } return sprintf( '%6.1f%s', $bytes, $post[0] ); } sub hash_file { my ( $self, $file ) = @_; my $data; my $digest = Digest::SHA->new(1); # read() fails for empty files if ( ( stat($file) )[7] == 0 ) { return $digest->hexdigest; } if ( $self->{config}->{read_size} == 0 ) { $digest->addfile($file); return $digest->hexdigest; } #<<< perltidy has problems indenting 'or die' with tabs open( my $fh, '<', $file ) or die("Can't open ${file} for reading: $!\n"); binmode($fh) or die("Can't set binmode on ${file}: $!\n"); read( $fh, $data, $self->{config}->{read_size} ) or die("Can't read ${file}: $!\n"); close($fh) or die("Can't close ${file}: $!\n"); #>>> $digest->add($data); return $digest->hexdigest; } sub hash_in_db { my ( $self, $hash ) = @_; if ( $self->{ignored}->{$hash} ) { return '// ignored'; } for my $name ( $self->files() ) { my $file = $self->file($name); if ( $file->{hash} eq $hash ) { return $name; } } return; } sub file_in_db { my ( $self, $file ) = @_; return $self->hash_in_db( $self->hash_file($file) ); } sub read_size { my ($self) = @_; return $self->{config}->{read_size}; } sub db_version { my ($self) = @_; return $self->{version}; } sub file { my ( $self, $name ) = @_; return $self->{files}->{$name}; } sub delete_file { my ( $self, $name ) = @_; delete $self->{files}->{$name}; return 1; } sub files { my ($self) = @_; return keys %{ $self->{files} }; } sub add_file { my ( $self, %opt ) = @_; my $file = $opt{file}; my $path = $opt{path}; my ( $size, $mtime ) = ( stat($path) )[ 7, 9 ]; if ( $self->file($file) and $self->file($file)->{mtime} == $mtime and $self->file($file)->{size} == $size ) { return; } my $hash = $self->hash_file($path); if ( $self->{ignored}->{$hash} ) { if ( $opt{unignore} ) { $self->unignore($hash); } else { return; } } $self->{files}->{$file} = { hash => $hash, mtime => $mtime, size => $size, }; return 1; } sub ignored { my ($self) = @_; if ( exists $self->{ignored} ) { return keys %{ $self->{ignored} }; } return (); } sub ignore { my ( $self, $file, $path ) = @_; $self->delete_file($file); $self->{ignored}->{ $self->hash_file($path) } = 1; return 1; } sub unignore { my ( $self, $hash ) = @_; delete $self->{ignored}->{$hash}; return 1; } sub save { my ( $self, $file ) = @_; return nstore( $self, $file ); } 1; __END__ =head1 NAME App::Hashl - Partially hash files, check if files are equal etc. =head1 SYNOPSIS use App::Hashl; my $hashl = App::Hashl->new(); # or: App::Hashl->new_from_file($database_file); =head1 VERSION This manual documents App::Hashl version 1.00 =head1 DESCRIPTION App::Hashl contains utilities to hash the first n bytes of files, store and recall them, check if another file is already in the database and optionally ignore file hashes. =head1 METHODS =over =item $hashl = App::Hashl->new(I<%conf>) Returns a new B object. Accepted parameters are: =over =item B => I How many bytes of a file to consider for the hash. Defaults to 4 MiB (4 * 2**20 bytes). 0 means read the whole file. =back =item $hashl = App::Hashl->new_from_file(I<$file>) Returns the B object saved to I by a prior $hashl->save call. =item $hashl->si_size(I<$bytes>) Returns I as a human-readable SI-size, such as "1.0k", "50.7M", "2.1G". The returned string is always six characters long. =item $hashl->hash_file(I<$file>) Returns the SHA1 hash of the first few bytes (as configured via B) of I. Dies if I cannot be read. =item $hashl->hash_in_db(I<$hash>) Checks if I is in the database. If it is, returns the filename it is associated with. If it is ignored, returns "// ignored" (subject to change). Otherwise, returns false. =item $hashl->file_in_db(I<$file>) Checks if I's hash is in the database. For the return value, see B. =item $hashl->read_size() Returns the current read size. Note that once an B object has been created, it is not possible to change the read size. =item $hashl->file(I<$name>) Returns a hashref describing the file. The layout is as follows: hash => file's hash, mtime => mtime as UNIX timestamp, size => file size in bytes If I does not exist in the database, returns undef. =item $hashl->delete_file(I<$name>) Remove the file from the database. =item $hashl->files() Returns a list of all file names in the database. =item $hashl->add_file(I<%data>) Add a file to the database. Required keys in I<%data> are: =over =item B => I relateve file name to store in the database =item B => I Full path to the file =item B => B<0>|B<1> If true: do not skip ignored files, unignore and re-add them instead =back If the file already is in the database, it is only updated if both the file size and the mtime have changed. Returns true if the file was actually added to the database, false if it is ignored or already present (and up-to-date). =item $hashl->ignored() Returns a list of all ignored file hashes. =item $hashl->ignore(I<$file>, I<$path>) Removes I<$file> from the database and adds I<$path> to the list of ignored file hashes. =item $hashl->unignore(I<$path>) Unignore the hash of I<$path>. =item $hashl->save(I<$file>) Save the B object with all data to I<$file>. It can later be retrieved via B. =back =head1 DIAGNOSTICS None. =head1 DEPENDENCIES Digest::SHA(3pm); =head1 BUGS AND LIMITATIONS None known. =head1 AUTHOR Copyright (C) 2011 by Daniel Friesel Ederf@finalrewind.orgE =head1 LICENSE 0. You just DO WHAT THE FUCK YOU WANT TO.