The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
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: