Acq: EDI omnibus bugfix package
authorLebbeous Fogle-Weekley <lebbeous@esilibrary.com>
Wed, 23 Jan 2013 18:07:33 +0000 (13:07 -0500)
committerBill Erickson <berick@esilibrary.com>
Thu, 14 Feb 2013 20:07:01 +0000 (15:07 -0500)
EDIReader bugfix - Process misc charges better
    MOA+304 seems to be an encoding peculiar to Ingram for processing fees.

Bubble up errors better from invoice processing
    Errors were getting logged, but weren't linked on the related
    acq.edi_message row in the database like they could have been to make
    debugging easier.

    This incidentally elevates one possible message from warning level to
    error, but even as a warning it was stopping the processing of the
    invoice in question there.  So there's no meaningful difference.

Try harder to associate incoming EDI messages with exact right account
    Sites use many very nearly identical EDI accounts (same host and
    credentials) with different values only for the label and the vendcode.
    This allows mapping of an order to a profile on the vendor side.

    The problem with this is that the edi_fetcher and the processes it
    kicks off did not know how to map incoming messages to the right
    account based on vendcode.  That code simply iterated through
    accounts, using host information and login credentials, and grabbing
    what it can find, as if
    there will be no other Evergreen-side EDI "accounts" with the same
    hostname and loging credentials.

    This should help with that.

Style and whitespace cleanups in O::A::Acq::EDI.pm
    All I could stand before I just couldn't take it anymore.

New PO template created malformed JSON in the INC_COPIES=0 case
    And now it no longer should.

Allow order responses and invoices to omit PO repetition in lineitem refs
    Usually vendor documents have bits of EDI that look like:
    RFF+LI:100/123
    where 100 is a PO number and 123 is a lineitem number.

    Sometimes, for some documents, B&T at least will omit the '100/' part.
    This is fine because we don't really need that number repeated for
    every lineitem.  We do need this change so that our EDI reader code
    can deal with the omission, though.

Be more liberal reading EDIFACT message reference number
    Spec, if I read it correctly, says that this is alphanumeric, not just
    numeric, and ULS is one vendor I've seen taking advantage of letters
    and numbers in that space.

    This commit makes the relevant regex in our EDIReader appropriately
    more tolerant.

Fixes to new vencode parsing for incoming EDI messages

Prevent problem in preventing EDI re-retrieves
    The query we were using before would needlessly transfer large objects,
    potentially hitting Jabber message size limits.

    We're just testing for the existence of such objects, so we need no
    more than a single ID in the result.

Fix EDI invoices for ULS, improve troubleshootability
    1) Taxes appear in different, but still valid way in ULS invoices than
    in invoices from other vendors observed to date.

    2) Invoices from ULS use MOA 203 to indicate unit price instead of the
    usual meaning of whole-lineitem price.

    3) Now abuse acq.invoice.note to leave better troubleshooting
    breadcrumbs.

Invoices from EDI had unsavable invoice_items attached
    Deal with this by letting us create fund_debits a little later than in
    the previous workflow. We have to, because the EDI-level stuff creating
    the invoice doesn't know what fund we want to target for taxes and misc
    charges.

    The problem we're fixing manifested by showing an alert() dialog about
    ACQ_FUND_DEBIT_NOT_FOUND.

Signed-off-by: Lebbeous Fogle-Weekley <lebbeous@esilibrary.com>
Signed-off-by: Bill Erickson <berick@esilibrary.com>
Open-ILS/src/perlmods/lib/OpenILS/Application/Acq/EDI.pm
Open-ILS/src/perlmods/lib/OpenILS/Application/Acq/Invoice.pm
Open-ILS/src/perlmods/lib/OpenILS/Utils/EDIReader.pm
Open-ILS/src/sql/Pg/950.data.seed-values.sql

index d898a8c..4d8edcf 100644 (file)
@@ -47,19 +47,14 @@ my %map = (
     path     => 'remote_path',
 );
 
+my $VENDOR_KLUDGE_MAP = {
+    INVOIC => {
+        amount_billed_is_per_unit => [1699342]
+    },
+    ORDRSP => {
+    }
+};
 
-## Just for debugging stuff:
-sub add_a_msg {
-    my ($self, $conn) = @_;
-    my $e = new_editor(xact=>1);
-    my $incoming = Fieldmapper::acq::edi_message->new;
-    $incoming->edi("This is content");
-    $incoming->account(1);
-    $incoming->remote_file('in/some_file.edi');
-    $e->create_acq_edi_message($incoming);;
-    $e->commit;
-}
-# __PACKAGE__->register_method( method => 'add_a_msg', api_name => 'open-ils.acq.edi.add_a_msg');  # debugging
 
 __PACKAGE__->register_method(
        method    => 'retrieve',
@@ -93,71 +88,109 @@ sub retrieve_core {
     foreach my $account (@$set) {
         my $count = 0;
         my $server;
-        $logger->info("EDI check for vendor " . ++$vcount . " of " . scalar(@$set) . ": " . $account->host);
-        unless ($server = __PACKAGE__->remote_account($account)) {   # assignment, not comparison
-            $logger->err(sprintf "Failed remote account mapping for %s (%s)", $account->host, $account->id);
+        $logger->info(
+            "EDI check for vendor " .
+            ++$vcount . " of " . scalar(@$set) . ": " . $account->host
+        );
+        unless ($server = __PACKAGE__->remote_account($account)) { # assignment
+            $logger->err(
+                sprintf "Failed remote account mapping for %s (%s)",
+                $account->host, $account->id
+            );
             next;
         };
-#       my $rf_starter = './';  # default to current dir
+
         if ($account->in_dir) { 
             if ($account->in_dir =~ /\*+.*\//) {
-                $logger->err("EDI in_dir has a slash after an asterisk in value: '" . $account->in_dir . "'.  Skipping account with indeterminate target dir!");
+                $logger->err(
+                    "EDI in_dir has a slash after an asterisk in value: '" .
+                    $account->in_dir .
+                    "'.  Skipping account with indeterminate target dir!"
+                );
                 next;
             }
-#           $rf_starter = $account->in_dir;
-#           $rf_starter =~ s/((\/)?[^\/]*)\*+[^\/]*$//;  # kill up to the first (possible) slash before the asterisk: keep the preceeding static dir
-#           $rf_starter .= '/' if $rf_starter or $2;   # recap the dir, or replace leading "/" if there was one (but don't add if empty)
         }
+
         my @files    = ($server->ls({remote_file => ($account->in_dir || './')}));
         my @ok_files = grep {$_ !~ /\/\.?\.$/ and $_ ne '0'} @files;
         $logger->info(sprintf "%s of %s files at %s/%s", scalar(@ok_files), scalar(@files), $account->host, $account->in_dir);   
-        # $server->remote_path(undef);
+
         foreach my $remote_file (@ok_files) {
-            # my $remote_file = $rf_starter . $_;
             my $description = sprintf "%s/%s", $account->host, $remote_file;
             
-            # deduplicate vs. acct/filenames already in DB
-            my $hits = $e->search_acq_edi_message([
-                {
-                    account     => $account->id,
-                    remote_file => $remote_file,
-                    status      => {'in' => [qw/ processed /]},     # if it never got processed, go ahead and get the new one (try again)
-                    # create_time => 'NOW() - 60 DAYS',     # if we wanted to allow filenames to be reused after a certain time
-                    # ideally we would also use the date from FTP, but that info isn't available via RemoteAccount
-                }
-                # { flesh => 1, flesh_fields => {...}, }
-            ]);
-            if (scalar(@$hits)) {
+            # deduplicate vs. acct/filenames already in DB.
+            #
+            # The reason we match against host/username/password/in_dir
+            # is that there may be many variant accounts that point to the
+            # same FTP site and credentials.  If we only checked based on
+            # acq.edi_account.id, we'd not find out in those cases that we've
+            # already processed the same file before.
+            my $hits = $e->search_acq_edi_message(
+                [
+                    {
+                        "+acqedi" => {
+                            host => $account->host,
+                            username => $account->username,
+                            password => $account->password,
+                            in_dir => $account->in_dir
+                        },
+                        remote_file => $remote_file,
+                        status      => {'in' => [qw/ processed /]},
+                    },
+                    { join => {"acqedi" => {}}, limit => 1 }
+                ], { idlist => 1 }
+            );
+
+            if (!$hits) {
+                my $msg = "EDI: test for already-retrieved files yielded " .
+                    "event " . $e->event->{textcode};
+                $logger->warn($msg);
+                warn $msg;
+                return $e->die_event;
+            }
+
+            if (@$hits) {
                 $logger->debug("EDI: $remote_file already retrieved.  Skipping");
                 warn "EDI: $remote_file already retrieved.  Skipping";
                 next;
             }
 
             ++$count;
-            $max and $count > $max and last;
-            $logger->info(sprintf "%s of %s targets: %s", $count, scalar(@ok_files), $description);
-            print sprintf "%s of %s targets: %s\n", $count, scalar(@ok_files), $description;
+            if ($max and $count > $max) {
+                last;
+            }
+
+            $logger->info(
+                sprintf "%s of %s targets: %s",
+                    $count, scalar(@ok_files), $description
+            );
+            printf("%d of %d targets: %s\n", $count, scalar(@ok_files), $description);
             if ($test) {
                 push @return, "test_$count";
                 next;
             }
             my $content;
             my $io = IO::Scalar->new(\$content);
-            unless ( $server->get({remote_file => $remote_file, local_file => $io}) ) {
+
+            unless (
+                $server->get({remote_file => $remote_file, local_file => $io})
+            ) {
                 $logger->error("(S)FTP get($description) failed");
                 next;
             }
-            my $incoming = __PACKAGE__->process_retrieval($content, $remote_file, $server, $account->id);
-#           $server->delete(remote_file => $_);   # delete remote copies of saved message
+
+            my $incoming = __PACKAGE__->process_retrieval(
+                $content, $remote_file, $server, $account->id
+            );
+
             push @return, @$incoming;
         }
     }
     return \@return;
 }
 
-# my $msg_ids = OpenILS::Application::Acq::EDI->process_retrieval(
-#   $file_content, $remote_filename, $server, $account_id, $editor);
 
+# procses_retrieval() returns a reference to a list of acq.edi_message IDs
 sub process_retrieval {
     my ($class, $content, $filename, $server, $account_or_id) = @_;
     $content or return;
@@ -208,6 +241,7 @@ sub process_retrieval {
         $e->xact_begin;
         $incoming = $e->retrieve_acq_edi_message($incoming->id);
         if ($@) {
+            $logger->error($@);
             $incoming->status('proc_error');
             $incoming->error($@);
         } else {
@@ -224,63 +258,95 @@ sub process_retrieval {
 
 # ->send_core
 # $account     is a Fieldmapper object for acq.edi_account row
-# $messageset  is an arrayref with acq.edi_message.id values
+# $message_ids is an arrayref with acq.edi_message.id values
 # $e           is optional editor object
 sub send_core {
     my ($class, $account, $message_ids, $e) = @_;    # $e is a working editor
 
-    ($account and scalar @$message_ids) or return;
+    return unless $account and @$message_ids;
     $e ||= new_editor();
 
     $e->xact_begin;
     my @messageset = map {$e->retrieve_acq_edi_message($_)} @$message_ids;
     $e->xact_rollback;
     my $m_count = scalar(@messageset);
-    (scalar(@$message_ids) == $m_count) or
+    if (@$message_ids != $m_count) {
         $logger->warn(scalar(@$message_ids) - $m_count . " bad IDs passed to send_core (ignored)");
+    }
 
     my $log_str = sprintf "EDI send to edi_account %s (%s)", $account->id, $account->host;
     $logger->info("$log_str: $m_count message(s)");
-    $m_count or return;
+    return unless $m_count;
 
     my $server;
     my $server_error;
-    unless ($server = __PACKAGE__->remote_account($account, 1)) {   # assignment, not comparison
+    unless ($server = __PACKAGE__->remote_account($account, 1)) { # assignment
         $logger->error("Failed remote account connection for $log_str");
         $server_error = 1;
-    };
+    }
+
     foreach (@messageset) {
         $_ or next;     # we already warned about bum ids
         my ($res, $error);
         if ($server_error) {
-            $error = "Server error: Failed remote account connection for $log_str"; # already told $logger, this is to update object below
+            # We already told $logger; this is to update object below
+            $error = "Server error: Failed remote account connection ".
+                "for $log_str";
         } elsif (! $_->edi) {
-            $logger->error("Message (id " . $_->id. ") for $log_str has no EDI content");
+            $logger->error(
+                "Message (id " . $_->id. ") for $log_str has no EDI content"
+            );
             $error = "EDI empty!";
-        } elsif ($res = $server->put({remote_path => $account->path, content => $_->edi, single_ext => 1})) {
+        } elsif (
+            $res = $server->put({
+                remote_path => $account->path, content => $_->edi,
+                    single_ext => 1
+            })
+        ) {
             #  This is the successful case!
             $_->remote_file($res);
             $_->status('complete');
-            $_->process_time('NOW');    # For outbound files, sending is the end of processing on the EG side.
+            $_->process_time('NOW');
+
+            # For outbound files, sending is the end of processing on
+            # the EG side.
+
             $logger->info("Sent message (id " . $_->id. ") via $log_str");
         } else {
-            $logger->error("(S)FTP put to $log_str FAILED: " . ($server->error || 'UNKOWNN'));
+            $logger->error(
+                "(S)FTP put to $log_str FAILED: " .
+                ($server->error || 'UNKOWNN')
+            );
             $error = "put FAILED: " . ($server->error || 'UNKOWNN');
         }
+
         if ($error) {
             $_->error($error);
             $_->error_time('NOW');
         }
+
         $logger->info("Calling update_acq_edi_message");
         $e->xact_begin;
+
         unless ($e->update_acq_edi_message($_)) {
-             $logger->error("EDI send_core update_acq_edi_message failed for message object: " . Dumper($_));
-             OpenILS::Application::Acq::EDI::Translator->debug_file(Dumper($_              ), '/tmp/update_acq_edi_message.FAIL');
-             OpenILS::Application::Acq::EDI::Translator->debug_file(Dumper($_->to_bare_hash), '/tmp/update_acq_edi_message.FAIL.to_bare_hash');
+             $logger->error(
+                 "EDI send_core update_acq_edi_message failed " .
+                 "for message object: " . Dumper($_)
+             );
+
+             OpenILS::Application::Acq::EDI::Translator->debug_file(
+                 Dumper($_),
+                 '/tmp/update_acq_edi_message.FAIL'
+             );
+             OpenILS::Application::Acq::EDI::Translator->debug_file(
+                 Dumper($_->to_bare_hash),
+                 '/tmp/update_acq_edi_message.FAIL.to_bare_hash'
+             );
         }
+
         # There's always an update, even if we failed.
         $e->xact_commit;
-        __PACKAGE__->record_activity($account, $e);  # There's always an update, even if we failed.
+        __PACKAGE__->record_activity($account, $e);
     }
     return \@messageset;
 }
@@ -288,17 +354,23 @@ sub send_core {
 #  attempt_translation does not touch the DB, just the object.  
 sub attempt_translation {
     my ($class, $edi_message, $to_edi) = @_;
-    my $tran  = translator();
-    my $ret   = $to_edi ? $tran->json2edi($edi_message->jedi) : $tran->edi2json($edi_message->edi);
-#   $logger->error("json: " . Dumper($json)); # debugging
 
-    if (not $ret or (! ref($ret)) or $ret->is_fault) {      # RPC::XML::fault on failure
+    my $ret = $to_edi ? translator->json2edi($edi_message->jedi) :
+        translator->edi2json($edi_message->edi);
+
+    if (not $ret or (! ref($ret)) or $ret->is_fault) {
+        # RPC::XML::fault on failure
+
         $edi_message->status('trans_error');
         $edi_message->error_time('NOW');
-        my $pre = "EDI Translator " . ($to_edi ? 'json2edi' : 'edi2json') . " failed";
+        my $pre = "EDI Translator " .
+            ($to_edi ? 'json2edi' : 'edi2json') . " failed";
+
         my $message = ref($ret) ? 
-                      ("$pre, Error " . $ret->code . ": " . __PACKAGE__->nice_string($ret->string)) :
-                      ("$pre: "                           . __PACKAGE__->nice_string($ret)        ) ;
+            ("$pre, Error " . $ret->code . ": " .
+                __PACKAGE__->nice_string($ret->string)) :
+            ("$pre: " . __PACKAGE__->nice_string($ret)) ;
+
         $edi_message->error($message);
         $logger->error($message);
         return;
@@ -312,6 +384,7 @@ sub attempt_translation {
     } else {
         $edi_message->jedi($ret->value);   # translator returns an object
     }
+
     return $edi_message;
 }
 
@@ -331,7 +404,6 @@ sub retrieve_vendors {
             }
         }
     ]);
-#   {"id":{"!=":null},"+acqpro":{"active":"t"}}, {"join":"acqpro", "flesh_fields":{"acqedi":["provider"]},"flesh":1}
 }
 
 # This is the SRF-exposed call, so it does checkauth
@@ -425,18 +497,95 @@ sub nice_string {
     # return substr($string,0,$head) . "... " . substr($string, -1*$tail);
 }
 
+# process_message_buyer() is used in processing both INVOIC
+# messages as well as ORDRSP ones.  As such, the $eg_inv parameter is
+# optional.
+sub process_message_buyer {
+    my ($class, $e, $msg_hash, $message,  $log_prefix, $eg_inv) = @_;
+
+    # some vendors encode the account number as the SAN.
+    # starting with the san value, then the account value, 
+    # treat each as a san, then an acct number until the first success
+    for my $buyer ( ($msg_hash->{buyer_san}, $msg_hash->{buyer_acct}) ) {
+        next unless $buyer;
+
+        # some vendors encode the SAN as "$SAN $vendcode"
+        my $vendcode;
+        ($buyer, $vendcode) = $buyer =~ /(\S+)\s*(\S+)?$/;
+
+        my $addr = $e->search_actor_org_address(
+            {valid => "t", san => $buyer})->[0];
+
+        if ($addr) {
+
+            $eg_inv->receiver($addr->org_unit) if $eg_inv;
+
+            my $orig_acct = $e->retrieve_acq_edi_account($message->account);
+
+            if (defined($vendcode) and ($orig_acct->vendcode ne $vendcode)) {
+                # The vendcode can give us the opportunity to change the
+                # acq.edi_account with which our acq.edi_message is associated
+                # in case it's wrong.
+
+                my $other_accounts = $e->search_acq_edi_account(
+                    {
+                        vendcode => $vendcode,
+                        host => $orig_acct->host,
+                        username => $orig_acct->username,
+                        password => $orig_acct->password,
+                        in_dir => $orig_acct->in_dir
+                    }
+                );
+
+                if (@$other_accounts) {
+                    # We can update this object because the caller saves
+                    # it with cstore later.
+                    $message->account($other_accounts->[0]->id);
+
+                    $logger->info(
+                        $log_prefix . sprintf(
+                            "changing edi_account from %d to %d based on " .
+                            "vendcode '%s'",
+                            $orig_acct->id, $message->account, $vendcode
+                        )
+                    );
+                }
+            }
+
+            last;
+
+        } elsif ($eg_inv) {
+
+            my $acct = $e->search_acq_edi_account({vendacct => $buyer})->[0];
+
+            if ($acct) {
+                $eg_inv->receiver($acct->owner);
+                last;
+            }
+        }
+    }
+}
+
 # parts of this process can fail without the entire
 # thing failing.  If a catastrophic error occurs,
 # it will occur via die.
 sub process_parsed_msg {
     my ($class, $account, $incoming, $msg_hash) = @_;
 
+    # INVOIC
     if ($incoming->message_type eq 'INVOIC') {
         return $class->create_acq_invoice_from_edi(
             $msg_hash, $account->provider, $incoming);
     }
 
     # ORDRSP
+
+    #  First do this for the whole message...
+    $class->process_message_buyer(
+        new_editor, $msg_hash, $incoming, "ORDRSP processing"
+    );
+
+    #  ... now do this stuff per-lineitem.
     for my $li_hash (@{$msg_hash->{lineitems}}) {
         my $e = new_editor(xact => 1);
 
@@ -686,13 +835,26 @@ sub edi_date_to_iso {
 }
 
 
+# Return hash with a key for every kludge that should apply for this
+# msg_type (INVOIC,ORDRSP) and this vendor SAN.
+sub get_kludges {
+    my ($class, $msg_type, $vendor_san) = @_;
+
+    my @kludges;
+    while (my ($kludge, $vendors) = each %{$VENDOR_KLUDGE_MAP->{$msg_type}}) {
+        push @kludges, $kludge if grep { $_ eq $vendor_san } @$vendors;
+    }
+
+    return map { $_ => 1 } @kludges;
+}
+
 # create_acq_invoice_from_edi() does what it sounds like it does for INVOIC
 # messages.  For similar operation on ORDRSP messages, see the guts of
 # process_jedi().
 # Return boolean success indicator.
 sub create_acq_invoice_from_edi {
-    my ($class, $invoice, $provider, $message) = @_;
-    # $invoice is O::U::EDIReader hash
+    my ($class, $msg_data, $provider, $message) = @_;
+    # $msg_data is O::U::EDIReader hash
     # $provider is only a pkey
     # $message is Fieldmapper::acq::edi_message
 
@@ -701,71 +863,59 @@ sub create_acq_invoice_from_edi {
     my $log_prefix = "create_acq_invoice_from_edi(..., <acq.edi_message #" .
         $message->id . ">): ";
 
+    my %msg_kludges;
+    if ($msg_data->{vendor_san}) {
+        %msg_kludges = $class->get_kludges('INVOIC', $msg_data->{vendor_san});
+    } else {
+        $logger->warn($log_prefix . "no vendor_san field!");
+    }
+
     my $eg_inv = Fieldmapper::acq::invoice->new;
 
+    # Some troubleshooting aids.  Yeah we should have made appropriate links
+    # for this in the schema, but this is better than nothing.  Probably
+    # *don't* try to i18n this.
+    $eg_inv->note("Generated from acq.edi_message #" . $message->id . ".");
+    if (%msg_kludges) {
+        $eg_inv->note(
+            $eg_inv->note .
+            " Vendor kludges: " . join(", ", keys(%msg_kludges)) . "."
+        );
+    }
+
     $eg_inv->provider($provider);
     $eg_inv->shipper($provider);    # XXX Do we really have a meaningful way to
                                     # distinguish provider and shipper?
     $eg_inv->recv_method("EDI");
 
     $eg_inv->recv_date(
-        $class->edi_date_to_iso($invoice->{invoice_date}));
-
-
-    # some vendors encode the account number as the SAN.
-    # starting with the san value, then the account value, 
-    # treat each as a san, then an acct number until the first success
-    for my $buyer ( ($invoice->{buyer_san}, $invoice->{buyer_acct}) ) {
-        next unless $buyer;
-
-        # some vendors encode the SAN as "$SAN $vendcode"
-        $buyer =~ s/\s.*//g;
-
-        my $addr = $e->search_actor_org_address(
-            {valid => "t", san => $buyer})->[0];
+        $class->edi_date_to_iso($msg_data->{invoice_date}));
 
-        if ($addr) {
-
-            $eg_inv->receiver($addr->org_unit);
-            last;
-
-        } else {
-
-            my $acct = $e->search_acq_edi_account({vendacct => $buyer})->[0];
 
-            if ($acct) {
-                $eg_inv->receiver($acct->owner);
-                last;
-            }
-        }
-    }
+    $class->process_message_buyer($e, $msg_data, $message, $log_prefix, $eg_inv);
 
     if (!$eg_inv->receiver) {
-        $logger->error($log_prefix . 
+        die($log_prefix .
             sprintf("unable to determine buyer (org unit) in invoice; ".
                 "buyer_san=%s; buyer_acct=%s",
-                ($invoice->{buyer_san} || ''), 
-                ($invoice->{buyer_acct} || '')
+                ($msg_data->{buyer_san} || ''), 
+                ($msg_data->{buyer_acct} || '')
             )
         );
-        return 0;
     }
 
-    $eg_inv->inv_ident($invoice->{invoice_ident});
+    $eg_inv->inv_ident($msg_data->{invoice_ident});
 
     if (!$eg_inv->inv_ident) {
-        $logger->error(
-            $log_prefix . "no invoice ID # in INVOIC message; " . shift
-        );
-        return 0;
+        die($log_prefix . "no invoice ID # in INVOIC message; " . shift);
     }
 
     my @eg_inv_entries;
     my @eg_inv_cancel_lis;
 
-    $message->purchase_order($invoice->{purchase_order});
+    $message->purchase_order($msg_data->{purchase_order});
 
-    for my $lineitem (@{$invoice->{lineitems}}) {
+    for my $lineitem (@{$msg_data->{lineitems}}) {
         my $li_id = $lineitem->{id};
 
         if (!$li_id) {
@@ -776,9 +926,7 @@ sub create_acq_invoice_from_edi {
         my $li = $e->retrieve_acq_lineitem($li_id);
 
         if (!$li) {
-            $logger->warn($log_prefix . 
-                "no LI found with ID: $li_id : " . $e->event);
-            return 0;
+            die($log_prefix . "no LI found with ID: $li_id : " . $e->event);
         }
 
         my ($quant) = grep {$_->{code} eq '47'} @{$lineitem->{quantities}};
@@ -794,6 +942,8 @@ sub create_acq_invoice_from_edi {
         # and $lineitem->{gross_unit_price}
         my $lineitem_price = $lineitem->{amount_billed};
 
+        $lineitem_price *= $quantity if $msg_kludges{amount_billed_is_per_unit};
+
         # if the top-level PO value is unset, get it from the first LI
         $message->purchase_order($li->purchase_order)
             unless $message->purchase_order;
@@ -821,6 +971,10 @@ sub create_acq_invoice_from_edi {
         push @eg_inv_cancel_lis, 
             {lineitem => $li, quantity => $quantity} 
             if $li->cancel_reason;
+
+        # The EDIReader class does detect certain per-lineitem taxes, but
+        # we'll ignore them for now, as the only sample invoices I've yet seen
+        # containing them also had a final cumulative tax at the end.
     }
 
     my @eg_inv_items;
@@ -828,31 +982,33 @@ sub create_acq_invoice_from_edi {
     my %charge_type_map = (
         'TX' => ['TAX', 'Tax from electronic invoice'],
         'CA' => ['PRO', 'Cataloging services'], 
-        'DL' => ['SHP', 'Delivery']
-    );
+        'DL' => ['SHP', 'Delivery'],
+        'GST' => ['TAX', 'Goods and services tax']
+    ); # XXX i18n, somehow
 
-    for my $charge (@{$invoice->{misc_charges}}) {
+    for my $charge (@{$msg_data->{misc_charges}}, @{$msg_data->{taxes}}) {
         my $eg_inv_item = Fieldmapper::acq::invoice_item->new;
 
-        my $amount = $charge->{charge_amount};
+        my $amount = $charge->{amount};
 
         if (!$amount) {
             $logger->warn($log_prefix . "charge with no amount");
             next;
         }
 
-        my $map = $charge_type_map{$charge->{charge_type}};
+        my $map = $charge_type_map{$charge->{type}};
 
         if (!$map) {
             $map = [
                 'PRO',
-                'Unknown charge type ' .  $charge->{charge_type}
+                'Unknown charge type ' .  $charge->{type}
             ];
         }
 
         $eg_inv_item->inv_item_type($$map[0]);
-        $eg_inv_item->note($$map[1]);
+        $eg_inv_item->title($$map[1]);  # title is user-visible; note isn't.
         $eg_inv_item->cost_billed($amount);
+        $eg_inv_item->amount_paid($amount);
 
         push @eg_inv_items, $eg_inv_item;
     }
@@ -865,16 +1021,12 @@ sub create_acq_invoice_from_edi {
 
     # save changes to acq.edi_message row
     if (not $e->update_acq_edi_message($message)) {
-        $logger->error(
-            $log_prefix . "couldn't update edi_message " . $message->id
-        );
-        return 0;
+        die($log_prefix . "couldn't update edi_message " . $message->id);
     }
 
     # create EG invoice
     if (not $e->create_acq_invoice($eg_inv)) {
-        $logger->error($log_prefix . "couldn't create invoice: " . $e->event);
-        return 0;
+        die($log_prefix . "couldn't create invoice: " . $e->event);
     }
 
     # Now we have a pkey for our EG invoice, so set the invoice field on all
@@ -883,11 +1035,10 @@ sub create_acq_invoice_from_edi {
     foreach (@eg_inv_entries) {
         $_->invoice($eg_inv_id);
         if (not $e->create_acq_invoice_entry($_)) {
-            $logger->error(
+            die(
                 $log_prefix . "couldn't create entry against lineitem " .
                 $_->lineitem . ": " . $e->event
             );
-            return 0;
         }
     }
 
@@ -895,10 +1046,7 @@ sub create_acq_invoice_from_edi {
     foreach (@eg_inv_items) {
         $_->invoice($eg_inv_id);
         if (not $e->create_acq_invoice_item($_)) {
-            $logger->error(
-                $log_prefix . "couldn't create inv item: " . $e->event
-            );
-            return 0;
+            die($log_prefix . "couldn't create inv item: " . $e->event);
         }
     }
 
@@ -944,10 +1092,9 @@ sub create_acq_invoice_from_edi {
 
             $lid->clear_cancel_reason;
             unless ($e->update_acq_lineitem_detail($lid)) {
-                $logger->error($log_prefix . 
+                die($log_prefix .
                     "couldn't clear lid cancel reason: ". $e->die_event
                 );
-                return 0;
             }
         }
 
@@ -956,10 +1103,9 @@ sub create_acq_invoice_from_edi {
         $li->edit_time('now'); 
 
         unless ($e->update_acq_lineitem($li)) {
-            $logger->error($log_prefix . 
+            die($log_prefix .
                 "couldn't clear li cancel reason: ". $e->die_event
             );
-            return 0;
         }
     }
 
index 9d59b6a..7563e88 100644 (file)
@@ -10,6 +10,18 @@ use OpenILS::Event;
 my $U = 'OpenILS::Application::AppUtils';
 
 
+sub _prepare_fund_debit_for_inv_item {
+    my ($debit, $item, $e) = @_;
+    $debit->fund($item->fund);
+    $debit->amount($item->amount_paid);
+    $debit->origin_amount($item->amount_paid);
+    $debit->origin_currency_type(
+        $e->retrieve_acq_fund($item->fund)->currency_type
+    ); # future: cache funds locally
+    $debit->encumbrance('f');
+    $debit->debit_type('direct_charge');
+}
+
 __PACKAGE__->register_method(
        method => 'build_invoice_api',
        api_name        => 'open-ils.acq.invoice.update',
@@ -103,12 +115,7 @@ sub build_invoice_api {
                         $debit = Fieldmapper::acq::fund_debit->new;
                         $debit->isnew(1);
                     }
-                    $debit->fund($item->fund);
-                    $debit->amount($item->amount_paid);
-                    $debit->origin_amount($item->amount_paid);
-                    $debit->origin_currency_type($e->retrieve_acq_fund($item->fund)->currency_type); # future: cache funds locally
-                    $debit->encumbrance('f');
-                    $debit->debit_type('direct_charge');
+                    _prepare_fund_debit_for_inv_item($debit, $item, $e);
 
                     if($debit->isnew) {
                         $e->create_acq_fund_debit($debit) or return $e->die_event;
@@ -140,11 +147,30 @@ sub build_invoice_api {
 
 
             } elsif($item->ischanged) {
+                my $debit;
+
+                if (!$item->fund_debit) {
+                    # No fund_debit yet? Make one now.
+                    $debit = Fieldmapper::acq::fund_debit->new;
+                    $debit->isnew(1);
+
+                    _prepare_fund_debit_for_inv_item($debit, $item, $e);
+                } else {
+                    $debit = $e->retrieve_acq_fund_debit($item->fund_debit) or
+                        return $e->die_event;
+                }
 
-                my $debit = $e->retrieve_acq_fund_debit($item->fund_debit) or return $e->die_event;
                 $debit->amount($item->amount_paid);
                 $debit->fund($item->fund);
-                $e->update_acq_fund_debit($debit) or return $e->die_event;
+
+                if ($debit->isnew) {
+                    # Making a new debit, so make it and link our item to it.
+                    $e->create_acq_fund_debit($debit) or return $e->die_event;
+                    $item->fund_debit($e->data->id);
+                } else {
+                    $e->update_acq_fund_debit($debit) or return $e->die_event;
+                }
+
                 $e->update_acq_invoice_item($item) or return $e->die_event;
             }
         }
index 7096883..3a3ecc6 100644 (file)
@@ -17,9 +17,10 @@ use strict; use warnings;
 
 my $NEW_MSG_RE = '^UNH'; # starts a new message
 my $NEW_LIN_RE = '^LIN'; # starts a new line item
+my $END_ALL_LIN = '^UNS'; # no more lineitems after this
 
 my %edi_fields = (
-    message_type    => qr/^UNH\+\d+\+(\S{6})/,
+    message_type    => qr/^UNH\+[A-z0-9]+\+(\S{6})/,
     buyer_san       => qr/^NAD\+BY\+([^:]+)::31B/,
     buyer_acct      => qr/^NAD\+BY\+([^:]+)::91/,
     vendor_san      => qr/^NAD\+SU\+([^:]+)::31B/,
@@ -31,7 +32,7 @@ my %edi_fields = (
 );
 
 my %edi_li_fields = (
-    id      => qr/^RFF\+LI:\S+\/(\S+)/,
+    id      => qr/^RFF\+LI:(?:\S+\/)?(\d+)/,
     index   => qr/^LIN\+([^\+]+)/,
     amount_billed   => qr/^MOA\+203:([^:]+)/,
     net_unit_price  => qr/^PRI\+AAA:([^:]+)/,
@@ -54,8 +55,15 @@ my %edi_li_quant_fields = (
 );
 
 my %edi_charge_fields = (
-    charge_type   => qr/^ALC\+C\++([^\+]+)/,
-    charge_amount => qr/^MOA\+(8|131):([^:]+)/
+    type   => qr/^ALC\+C\++([^\+]+)/,
+    amount => qr/^MOA\+(?:8|131|304):([^:]+)/
+);
+
+# This may need to be liberalized later, but it works for the only example I
+# have so far.
+my %edi_tax_fields = (
+    type   => qr/^TAX\+7\+([^\+]+)/,
+    amount => qr/^MOA\+124:([^:]+)/
 );
 
 sub new {
@@ -92,7 +100,7 @@ sub read {
         # - starting a new message
 
         if (/$NEW_MSG_RE/) { 
-            $msg = {lineitems => [], misc_charges => []};
+            $msg = {lineitems => [], misc_charges => [], taxes => []};
             push(@msgs, $msg);
         }
 
@@ -139,7 +147,7 @@ sub read {
 
         # - starting a new misc. charge
 
-        if (/$edi_charge_fields{charge_type}/) {
+        if (/$edi_charge_fields{type}/) {
             $msg->{_current_charge} = {};
             push (@{$msg->{misc_charges}}, $msg->{_current_charge});
         }
@@ -152,6 +160,36 @@ sub read {
                     if /$edi_charge_fields{$field}/;
             }
         }
+
+        # - starting a new tax charge.  Taxes wind up on current lineitem if
+        # any, otherwise in the top-level taxes array
+
+        if (/$edi_tax_fields{type}/) {
+            $msg->{_current_tax} = {};
+            if ($msg->{_current_li}) {
+                $msg->{_current_li}{tax} = $msg->{_current_tax}
+            } else {
+                push (@{$msg->{taxes}}, $msg->{_current_tax});
+            }
+        }
+
+        # - extract tax field
+
+        if (my $tax = $msg->{_current_tax}) {
+            for my $field (keys %edi_tax_fields) {
+                ($tax->{$field}) = $_ =~ /$edi_tax_fields{$field}/
+                    if /$edi_tax_fields{$field}/;
+            }
+        }
+
+        # This helps avoid associating taxes and charges at the end of the
+        # message with the final lineitem inapporiately.
+        if (/$END_ALL_LIN/) {
+            # remove the state-maintenance keys
+            foreach (grep /^_/, keys %$msg) {
+                delete $msg->{$_};
+            }
+        }
     }
 
     # remove the state-maintenance keys
index d4272a5..6e4e460 100644 (file)
@@ -8145,10 +8145,10 @@ $$
                 [% FOR note IN ftx_vals -%] "[% note %]"[% UNLESS loop.last %], [% END %][% END %] 
             ],            
 
-            "quantity":[% li.lineitem_details.size %],
+            "quantity":[% li.lineitem_details.size %]
 
             [%- IF INC_COPIES -%]
-            "copies" : [
+            ,"copies" : [
                 [%- compressed_copies = [];
                     FOR lid IN li.lineitem_details;
                         fund = lid.fund.code;