package IPC::SRLock::Sysv;
use namespace::autoclean;
use English qw( -no_match_vars );
use File::DataClass::Types qw( Object OctalNum PositiveInt );
use IPC::ShareLite qw( :lock );
use IPC::SRLock::Functions qw( Unspecified hash_from throw );
use Storable qw( nfreeze thaw );
use Time::HiRes qw( usleep );
use Try::Tiny;
use Moo;
extends q(IPC::SRLock::Base);
# Attribute constructors
my $_build__share = sub {
my $self = shift; my $share;
try { $share = IPC::ShareLite->new( '-key' => $self->lockfile,
'-create' => 1,
'-mode' => $self->mode,
'-size' => $self->size ) }
catch { throw "${_}: ${OS_ERROR}" };
return $share;
};
# Public attributes
has 'lockfile' => is => 'ro', isa => PositiveInt, default => 12_244_237;
has 'mode' => is => 'ro', isa => OctalNum, coerce => 1, default => '0666';
has 'size' => is => 'ro', isa => PositiveInt, default => 65_536;
# Private attributes
has '_share' => is => 'lazy', isa => Object, builder => $_build__share;
# Private functions
my $_store_share_data = sub {
my ($self, $data) = @_;
try { $self->_share->store( nfreeze $data ) }
catch { throw "${_}: ${OS_ERROR}" };
return 1;
};
my $_unlock_share = sub {
my $self = shift;
defined $self->_share->unlock or throw 'Failed to unset semaphore';
return;
};
my $_fetch_share_data = sub {
my ($self, $for_update, $async) = @_; my $data;
my $mode = $for_update ? LOCK_EX : LOCK_SH; $async and $mode |= LOCK_NB;
my $lock = $self->_share->lock( $mode );
defined $lock or throw 'Failed to set semaphore'; $lock or return;
try { $data = $self->_share->fetch; $data = $data ? thaw( $data ) : {} }
catch { throw "${_}: ${OS_ERROR}" };
not $for_update and $self->$_unlock_share;
return $data;
};
# Construction
sub BUILD {
my $self = shift; $self->_share; return;
}
# Public methods
sub list {
my $self = shift; my $data = $self->$_fetch_share_data; my $list = [];
while (my ($key, $info) = each %{ $data }) {
push @{ $list }, { key => $key,
pid => $info->{pid },
stime => $info->{stime },
timeout => $info->{timeout} };
}
return $list;
}
sub reset {
my $self = shift;
my $args = hash_from @_;
my $key = $args->{k} or throw Unspecified, [ 'key' ];
my $data = $self->$_fetch_share_data( 1 ); $key = "${key}";
my $found = delete $data->{ $key } and $self->$_store_share_data( $data );
$self->$_unlock_share;
$found or throw 'Lock [_1] not set', args => [ $key ];
return 1;
}
sub set {
my $self = shift; my $args = $self->_get_args( @_ ); my $start = time;
my $key = $args->{k}; my $pid = $args->{p}; my $timeout = $args->{t};
my $lock_set;
until ($lock_set) {
my ($lock, $lpid, $ltime, $ltimeout);
my $data = $self->$_fetch_share_data( 1, $args->{async} );
my $found = 0; my $now = time; my $timedout = 0;
if ($data) {
if (exists $data->{ $key } and $lock = $data->{ $key }) {
$lpid = $lock->{pid };
$ltime = $lock->{stime };
$ltimeout = $lock->{timeout};
if ($ltimeout and $now > $ltime + $ltimeout) {
$data->{ $key } = { pid => $pid,
stime => $now,
timeout => $timeout };
$lock_set = $self->$_store_share_data( $data );
$timedout = 1;
}
else { $found = 1 }
}
else {
$data->{ $key } = { pid => $pid,
stime => $now,
timeout => $timeout };
$lock_set = $self->$_store_share_data( $data );
}
$self->$_unlock_share;
}
not $lock_set and $args->{async} and return 0;
$timedout and $self->log->error
( $self->_timeout_error( $key, $lpid, $ltime, $ltimeout ) );
not $lock_set and $self->patience and $now > $start + $self->patience
and throw 'Lock [_1] timed out', args => [ $key ];
$found and usleep( 1_000_000 * $self->nap_time );
}
$self->log->debug( "Lock ${key} set by ${pid}" );
return 1;
}
1;
__END__
=pod
=head1 Name
IPC::SRLock::Sysv - Set / reset locks using System V IPC
=head1 Synopsis
use IPC::SRLock;
my $config = { type => q(sysv) };
my $lock_obj = IPC::SRLock->new( $config );
=head1 Description
Uses System V semaphores to lock access to a shared memory file
=head1 Configuration and Environment
This class defines accessors for these attributes:
=over 3
=item C<lockfile>
The key the the semaphore. Defaults to 12_244_237
=item C<mode>
Mode to create the shared memory file. Defaults to 0666
=item C<size>
Maximum size of a shared memory segment. Defaults to 65_536
=back
=head1 Subroutines/Methods
=head2 BUILD
Create the shared memory segment at construction time
=head2 list
List the contents of the lock table
=head2 reset
Delete a lock from the lock table
=head2 set
Set a lock in the lock table
=head1 Diagnostics
None
=head1 Dependencies
=over 3
=item L<File::DataClass>
=item L<IPC::ShareLite>
=item L<IPC::SRLock::Base>
=item L<Moo>
=item L<Storable>
=item L<Time::HiRes>
=item L<Try::Tiny>
=back
=head1 Incompatibilities
There are no known incompatibilities in this module
=head1 Bugs and Limitations
There are no known bugs in this module.
Please report problems to the address below.
Patches are welcome
=head1 Author
Peter Flanigan, C<< <pjfl@cpan.org> >>
=head1 License and Copyright
Copyright (c) 2015 Peter Flanigan. All rights reserved
This program is free software; you can redistribute it and/or modify it
under the same terms as Perl itself. See L<perlartistic>
This program is distributed in the hope that it will be useful,
but WITHOUT WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE
=cut
# Local Variables:
# mode: perl
# tab-width: 3
# End: