From Code to Community: Sponsoring The Perl and Raku Conference 2025 Learn more

#
# Copyright 2009-2013 MongoDB, Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# ABSTRACT: A Mongo GridFS file
use version;
our $VERSION = 'v0.999.998.5';
use Moose;
use Types::Standard -types;
use namespace::clean -except => 'meta';
has _grid => (
is => 'ro',
isa => InstanceOf['MongoDB::GridFS'],
required => 1,
);
#pod =attr info
#pod
#pod A hash reference of metadata saved with this file.
#pod
#pod =cut
has info => (
is => 'ro',
isa => HashRef,
required => 1,
);
#pod =method print
#pod
#pod $written = $file->print($fh);
#pod $written = $file->print($fh, $length);
#pod $written = $file->print($fh, $length, $offset)
#pod
#pod Writes the number of bytes specified from the offset specified
#pod to the given file handle. If no C<$length> or C<$offset> are
#pod given, the entire file is written to C<$fh>. Returns the number
#pod of bytes written.
#pod
#pod =cut
sub print {
my ($self, $fh, $length, $offset) = @_;
$offset ||= 0;
$length ||= 0;
my ($written, $pos) = (0, 0);
my $start_pos = $fh->getpos();
$self->_grid->chunks->ensure_index(Tie::IxHash->new(files_id => 1, n => 1), { safe => 1, unique => 1 });
my $cursor = $self->_grid->chunks->query({"files_id" => $self->info->{"_id"}})->sort({"n" => 1});
if ( !$cursor->has_next ) {
MongoDB::GridFSError->throw(
sprintf( "GridFS file corrupt: no chunks found for file ID '%s'",
$self->info->{_id} )
);
}
while ((my $chunk = $cursor->next) && (!$length || $written < $length)) {
my $len = length $chunk->{'data'};
# if we are cleanly beyond the offset
if (!$offset || $pos >= $offset) {
if (!$length || $written + $len < $length) {
$fh->print($chunk->{"data"});
$written += $len;
$pos += $len;
}
else {
$fh->print(substr($chunk->{'data'}, 0, $length-$written));
$written += $length-$written;
$pos += $length-$written;
}
next;
}
# if the offset goes to the middle of this chunk
elsif ($pos + $len > $offset) {
# if the length of this chunk is smaller than the desired length
if (!$length || $len <= $length-$written) {
$fh->print(substr($chunk->{'data'}, $offset-$pos, $len-($offset-$pos)));
$written += $len-($offset-$pos);
$pos += $len-($offset-$pos);
}
else {
$fh->print(substr($chunk->{'data'}, $offset-$pos, $length));
$written += $length;
$pos += $length;
}
next;
}
# if the offset is larger than this chunk
$pos += $len;
}
$fh->setpos($start_pos);
return $written;
}
#pod =method slurp
#pod
#pod $all = $file->slurp
#pod $bytes = $file->slurp($length);
#pod $bytes = $file->slurp($length, $offset);
#pod
#pod Return the number of bytes specified from the offset specified. If no
#pod C<$length> or C<$offset> are given, the entire file is returned.
#pod
#pod =cut
sub slurp {
my ($self,$length,$offset) = @_;
my $bytes = '';
my $fh = new IO::File \$bytes,'+>';
my $written = $self->print($fh,$length,$offset);
# some machines don't set $bytes
if ($written and !length($bytes)) {
my $retval;
read $fh, $retval, $written;
return $retval;
}
return $bytes;
}
__PACKAGE__->meta->make_immutable;
1;
__END__
=pod
=encoding UTF-8
=head1 NAME
MongoDB::GridFS::File - A Mongo GridFS file
=head1 VERSION
version v0.999.998.5
=head1 SYNOPSIS
use MongoDB::GridFS::File;
$outfile = IO::File->new("outfile", "w");
$file = $grid->find_one;
$file->print($outfile);
=head1 USAGE
=head2 Error handling
Unless otherwise explictly documented, all methods throw exceptions if
an error occurs. The error types are documented in L<MongoDB::Error>.
To catch and handle errors, the L<Try::Tiny> and L<Safe::Isa> modules
are recommended:
use Try::Tiny;
use Safe::Isa; # provides $_isa
$bytes = try {
$file->slurp;
}
catch {
if ( $_->$_isa("MongoDB::TimeoutError" ) {
...
}
else {
...
}
};
To retry failures automatically, consider using L<Try::Tiny::Retry>.
=head1 ATTRIBUTES
=head2 info
A hash reference of metadata saved with this file.
=head1 METHODS
=head2 print
$written = $file->print($fh);
$written = $file->print($fh, $length);
$written = $file->print($fh, $length, $offset)
Writes the number of bytes specified from the offset specified
to the given file handle. If no C<$length> or C<$offset> are
given, the entire file is written to C<$fh>. Returns the number
of bytes written.
=head2 slurp
$all = $file->slurp
$bytes = $file->slurp($length);
$bytes = $file->slurp($length, $offset);
Return the number of bytes specified from the offset specified. If no
C<$length> or C<$offset> are given, the entire file is returned.
=head1 AUTHORS
=over 4
=item *
David Golden <david@mongodb.com>
=item *
Mike Friedman <friedo@mongodb.com>
=item *
Kristina Chodorow <kristina@mongodb.com>
=item *
Florian Ragwitz <rafl@debian.org>
=back
=head1 COPYRIGHT AND LICENSE
This software is Copyright (c) 2015 by MongoDB, Inc..
This is free software, licensed under:
The Apache License, Version 2.0, January 2004
=cut