#!/usr/bin/perl # 2008 Kyle Hall # This file is part of Koha. # # Koha is free software; you can redistribute it and/or modify it # under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 3 of the License, or # (at your option) any later version. # # Koha is distributed in the hope that it will be useful, but # WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with Koha; if not, see . # use Modern::Perl; use CGI qw ( -utf8 ); use C4::Output qw( output_html_with_http_headers ); use C4::Auth qw( get_template_and_user ); use C4::Context; use C4::Accounts; use C4::Circulation qw( barcodedecode GetOpenIssue AddRenewal AddIssue MarkIssueReturned ); use C4::Items qw( ModDateLastSeen ); use C4::Members; use C4::Stats; use C4::BackgroundJob; use Koha::UploadedFiles; use Koha::Account; use Koha::Checkouts; use Koha::Patrons; use Date::Calc qw( Date_to_Days ); use constant DEBUG => 0; # this is the file version number that we're coded against. my $FILE_VERSION = '1.0'; our $query = CGI->new; my ($template, $loggedinuser, $cookie) = get_template_and_user({ template_name => "offline_circ/process_koc.tt", query => $query, type => "intranet", flagsrequired => { circulate => "circulate_remaining_permissions" }, }); my $fileID=$query->param('uploadedfileid'); my $runinbackground = $query->param('runinbackground'); my $completedJobID = $query->param('completedJobID'); my %cookies = CGI::Cookie->fetch(); my $sessionID = $cookies{'CGISESSID'}->value; ## 'Local' globals. our $dbh = C4::Context->dbh(); our @output = (); ## For storing messages to be displayed to the user if ($completedJobID) { my $job = C4::BackgroundJob->fetch($sessionID, $completedJobID); my $results = $job->results(); $template->param(transactions_loaded => 1); $template->param(messages => $results->{results}); } elsif ($fileID) { my $upload = Koha::UploadedFiles->find( $fileID ); my $fh = $upload? $upload->file_handle: undef; my $filename = $upload? $upload->filename: undef; my @input_lines = $fh? <$fh>: (); $fh->close if $fh; my $job = undef; if ($runinbackground) { my $job_size = scalar(@input_lines); $job = C4::BackgroundJob->new($sessionID, $filename, '/cgi-bin/koha/offline_circ/process_koc.pl', $job_size); my $jobID = $job->id(); # fork off if (my $pid = fork) { # parent # return job ID as JSON # prevent parent exiting from # destroying the kid's database handle # FIXME: according to DBI doc, this may not work for Oracle $dbh->{InactiveDestroy} = 1; my $reply = CGI->new(""); print $reply->header(-type => 'text/html'); print '{"jobID":"' . $jobID . '"}'; exit 0; } elsif (defined $pid) { # child # close STDOUT to signal to Apache that # we're now running in the background close STDOUT; close STDERR; } else { # fork failed, so exit immediately # fork failed, so exit immediately warn "fork failed while attempting to run offline_circ/process_koc.pl as a background job"; exit 0; } # if we get here, we're a child that has detached # itself from Apache } my $header_line = shift @input_lines; my $file_info = parse_header_line($header_line); if ($file_info->{'Version'} ne $FILE_VERSION) { push @output, { message => 1, ERROR_file_version => 1, upload_version => $file_info->{'Version'}, current_version => $FILE_VERSION }; } my $i = 0; foreach my $line (@input_lines) { $i++; my $command_line = parse_command_line($line); # map command names in the file to subroutine names my %dispatch_table = ( issue => \&kocIssueItem, 'return' => \&kocReturnItem, payment => \&kocMakePayment, ); # call the right sub name, passing the hashref of command_line to it. if ( exists $dispatch_table{ $command_line->{'command'} } ) { $dispatch_table{ $command_line->{'command'} }->($command_line); } else { warn "unknown command: '$command_line->{command}' not processed"; } if ($runinbackground) { $job->progress($i); } } if ($runinbackground) { $job->finish({ results => \@output }) if defined($job); } else { $template->param(transactions_loaded => 1); $template->param(messages => \@output); } } output_html_with_http_headers $query, $cookie, $template->output; =head1 FUNCTIONS =head2 parse_header_line parses the header line from a .koc file. This is the line that specifies things such as the file version, and the name and version of the offline circulation tool that generated the file. See L for more information. pass in a string containing the header line (the first line from th file). returns a hashref containing the information from the header. =cut sub parse_header_line { my $header_line = shift; chomp($header_line); $header_line =~ s/\r//g; my @fields = split( /\t/, $header_line ); my %header_info = map { split( /=/, $_ ) } @fields; return \%header_info; } =head2 parse_command_line =cut sub parse_command_line { my $command_line = shift; chomp($command_line); $command_line =~ s/\r//g; my ( $timestamp, $command, @args ) = split( /\t/, $command_line ); my ( $date, $time, $id ) = split( /\s/, $timestamp ); my %command = ( date => $date, time => $time, id => $id, command => $command, ); # set the rest of the keys using a hash slice my $argument_names = arguments_for_command($command); @command{@$argument_names} = @args; return \%command; } =head2 arguments_for_command fetches the names of the columns (and function arguments) found in the .koc file for a particular command name. For instance, the C command requires a C and C. In that case this function returns a reference to the list C. parameters: the command name returns: listref of column names. =cut sub arguments_for_command { my $command = shift; # define the fields for this version of the file. my %format = ( issue => [qw( cardnumber barcode )], return => [qw( barcode )], payment => [qw( cardnumber amount )], ); return $format{$command}; } sub kocIssueItem { my $circ = shift; $circ->{barcode} = barcodedecode( $circ->{barcode} ) if $circ->{barcode}; my $branchcode = C4::Context->userenv->{branch}; my $patron = Koha::Patrons->find( { cardnumber => $circ->{cardnumber} } ); my $borrower = $patron->unblessed; my $item = Koha::Items->find({ barcode => $circ->{barcode} }); my $issue = Koha::Checkouts->find( { itemnumber => $item->itemnumber } ); my $biblio = $item->biblio; if ( $issue ) { ## Item is currently checked out to another person. #warn "Item Currently Issued."; my $issue = GetOpenIssue( $item->itemnumber ); # FIXME Hum? That does not make sense, if it's in the issue table, the issue is open (i.e. returndate is null) if ( $issue->{'borrowernumber'} eq $borrower->{'borrowernumber'} ) { ## Issued to this person already, renew it. #warn "Item issued to this member already, renewing."; C4::Circulation::AddRenewal( $issue->{'borrowernumber'}, # borrowernumber $item->itemnumber, # itemnumber undef, # branch undef, # datedue - let AddRenewal calculate it automatically $circ->{'date'}, # issuedate ) unless (DEBUG); push @output, { renew => 1, title => $biblio->title, biblionumber => $biblio->biblionumber, barcode => $item->barcode, firstname => $borrower->{ 'firstname' }, surname => $borrower->{ 'surname' }, borrowernumber => $borrower->{'borrowernumber'}, cardnumber => $borrower->{'cardnumber'}, datetime => $circ->{ 'datetime' } }; } else { #warn "Item issued to a different member."; #warn "Date of previous issue: $issue->{'issuedate'}"; #warn "Date of this issue: $circ->{'date'}"; my ( $i_y, $i_m, $i_d ) = split( /-/, $issue->{'issuedate'} ); my ( $c_y, $c_m, $c_d ) = split( /-/, $circ->{'date'} ); if ( Date_to_Days( $i_y, $i_m, $i_d ) < Date_to_Days( $c_y, $c_m, $c_d ) ) { ## Current issue to a different persion is older than this issue, return and issue. C4::Circulation::AddIssue( $borrower, $circ->{'barcode'}, undef, undef, $circ->{'date'} ) unless ( DEBUG ); push @output, { issue => 1, title => $biblio->title, biblionumber => $biblio->biblionumber, barcode => $item->barcode, firstname => $borrower->{ 'firstname' }, surname => $borrower->{ 'surname' }, borrowernumber => $borrower->{'borrowernumber'}, cardnumber => $borrower->{'cardnumber'}, datetime => $circ->{ 'datetime' } }; } else { ## Current issue is *newer* than this issue, write a 'returned' issue, as the item is most likely in the hands of someone else now. #warn "Current issue to another member is newer. Doing nothing"; ## This situation should only happen of the Offline Circ data is *really* old. ## FIXME: write line to old_issues and statistics } } } else { ## Item is not checked out to anyone at the moment, go ahead and issue it C4::Circulation::AddIssue( $borrower, $circ->{'barcode'}, undef, undef, $circ->{'date'} ) unless ( DEBUG ); push @output, { issue => 1, title => $biblio->title, biblionumber => $biblio->biblionumber, barcode => $item->barcode, firstname => $borrower->{ 'firstname' }, surname => $borrower->{ 'surname' }, borrowernumber => $borrower->{'borrowernumber'}, cardnumber => $borrower->{'cardnumber'}, datetime =>$circ->{ 'datetime' } }; } } sub kocReturnItem { my ( $circ ) = @_; $circ->{barcode} = barcodedecode( $circ->{barcode} ) if $circ->{barcode}; my $item = Koha::Items->find({ barcode => $circ->{barcode} }); my $biblio = $item->biblio; my $borrowernumber = _get_borrowernumber_from_barcode( $circ->{'barcode'} ); if ( $borrowernumber ) { my $patron = Koha::Patrons->find( $borrowernumber ); C4::Circulation::MarkIssueReturned( $borrowernumber, $item->itemnumber, $circ->{'date'}, $patron->privacy ); $item->onloan(undef)->store; ModDateLastSeen( $item->itemnumber ); push @output, { return => 1, title => $biblio->title, biblionumber => $biblio->biblionumber, barcode => $item->barcode, borrowernumber => $patron->borrowernumber, firstname => $patron->firstname, surname => $patron->surname, cardnumber => $patron->cardnumber, datetime => $circ->{'datetime'} }; } else { push @output, { ERROR_no_borrower_from_item => 1, badbarcode => $circ->{'barcode'} }; } } sub kocMakePayment { my ($circ) = @_; my $cardnumber = $circ->{cardnumber}; my $amount = $circ->{amount}; my $patron = Koha::Patrons->find( { cardnumber => $cardnumber } ); Koha::Account->new( { patron_id => $patron->id } ) ->pay( { amount => $amount, interface => C4::Context->interface } ); push @output, { payment => 1, amount => $circ->{'amount'}, firstname => $patron->firstname, surname => $patron->surname, cardnumber => $patron->cardnumber, borrower => $patron->id, }; } =head2 _get_borrowernumber_from_barcode pass in a barcode get back the borrowernumber of the patron who has it checked out. undef if that can't be found =cut sub _get_borrowernumber_from_barcode { my $barcode = shift; return unless $barcode; my $item = Koha::Items->find({ barcode => $barcode }); return unless $item; my $issue = Koha::Checkouts->find( { itemnumber => $item->itemnumber } ); return unless $issue; return $issue->borrowernumber; }