App-Glacier
view release on metacpan or search on metacpan
lib/App/Glacier/DB/GDBM.pm view on Meta::CPAN
package App::Glacier::DB::GDBM;
use strict;
use warnings;
use GDBM_File;
use Carp;
use File::Basename;
use File::Path qw(make_path);
# Avoid coredumps in threaded code.
# See https://rt.perl.org/Public/Bug/Display.html?id=61912.
sub CLONE_SKIP { 1 }
sub new {
my $class = shift;
local %_ = @_;
my $file = delete $_{file} // croak "filename is required";
unless (-f $file) {
if (defined(my $create = delete $_{create})) {
if (ref($create) eq 'CODE') {
$create = &{$create}();
}
return undef unless $create;
}
my $dir = dirname($file);
unless (-d $dir) {
make_path($dir, {error=>\my $err});
if (@$err) {
for my $diag (@$err) {
my ($filename, $message) = %$diag;
$filename = $dir if ($filename eq '');
carp("error creating $filename: $message");
}
croak("failed to create $dir");
}
}
}
my $self = bless {}, $class;
$self->{_filename} = $file;
$self->{_mode} = delete $_{mode} || 0644;
$self->{_retries} = delete $_{retries} || 10;
$self->{_nref} = 0;
$self->{_deleted} = [];
return $self;
}
my %lexicon = (
backend => 1,
file => { mandatory => 1 },
mode => { default => 0644 },
ttl => { default => 72000, check => \&App::Glacier::Command::ck_number },
encoding => { default => 'json' }
);
sub configtest {
my ($class, $cfg, @path) = @_;
$cfg->lint(\%lexicon, @path);
}
# Tie in the database, run $code, and untie it again. Correctly handle
# nested invocations to avoid deadlocking.
sub _tied {
my ($self, $code) = @_;
croak "argument must be a CODE ref" unless ref($code) eq 'CODE';
if ($self->{_nref}++ == 0) {
my $n = 0;
while (! tie %{$self->{_map}}, 'GDBM_File', $self->{_filename},
GDBM_WRCREAT, $self->{_mode}) {
if ($n++ > $self->{_retries}) {
croak "can't open file $self->{_filename}: $!";
( run in 1.190 second using v1.01-cache-2.11-cpan-fe3c2283af0 )