EDI Invoicing
authorLebbeous Fogle-Weekley <lebbeous@esilibrary.com>
Tue, 17 Jul 2012 21:59:23 +0000 (17:59 -0400)
committerBill Erickson <berick@esilibrary.com>
Wed, 1 Aug 2012 15:14:19 +0000 (11:14 -0400)
Includes patch for edi4r to make invoice parsing possible.

Also adds OpenILS::Utils::LooseEDI module.  This doesn't try to do the
validation and other helpful things that Business::EDI aims to do, but it
does let us get up and running relatively quickly at getting data from EDI
invoices if we know what we're looking for.

[Amended to add bits about tax, plus release note blurb.]

Signed-off-by: Lebbeous Fogle-Weekley <lebbeous@esilibrary.com>
Signed-off-by: Bill Erickson <berick@esilibrary.com>
Open-ILS/src/edi_translator/misc/edi4r-break-on-nil-seg.patch [new file with mode: 0644]
Open-ILS/src/perlmods/lib/OpenILS/Application/Acq/EDI.pm
Open-ILS/src/perlmods/lib/OpenILS/Utils/LooseEDI.pm [new file with mode: 0644]
docs/RELEASE_NOTES_NEXT/edi-invoices.txt [new file with mode: 0644]

diff --git a/Open-ILS/src/edi_translator/misc/edi4r-break-on-nil-seg.patch b/Open-ILS/src/edi_translator/misc/edi4r-break-on-nil-seg.patch
new file mode 100644 (file)
index 0000000..06a6294
--- /dev/null
@@ -0,0 +1,10 @@
+--- a/lib/edi4r.rb     2012-07-17 17:57:27.000000000 -0400
++++ b/lib/edi4r.rb     2012-07-17 17:51:48.000000000 -0400
+@@ -758,6 +758,7 @@
+       loop do
+         index += 1
+         seg = msg[index]
++        break if seg.nil?
+         next  if child_mode and seg.level > level+1 # other descendants
+         break if seg.level <= level
+         results << seg
index 3fc0883..277f76e 100644 (file)
@@ -16,6 +16,7 @@ use OpenILS::Utils::CStoreEditor q/new_editor/;
 use OpenILS::Utils::Fieldmapper;
 use OpenILS::Application::Acq::EDI::Translator;
 
+use OpenILS::Utils::LooseEDI;
 use Business::EDI;
 
 use Data::Dumper;
@@ -177,6 +178,9 @@ sub process_retrieval {
     $e->xact_commit;
     # refresh: send process_jedi the updated row
     $e->xact_begin;
+
+    # LFW: I really don't understand in what sense you could call this
+    # message 'outgoing', except from the vendor's point of view?
     my $outgoing = $e->retrieve_acq_edi_message($incoming->id);  # refresh again!
     $e->xact_rollback;
     my $res = __PACKAGE__->process_jedi($outgoing, $server, $account, $e);
@@ -409,6 +413,16 @@ our @noop_6063 = (21);
 # ->process_jedi($message, $server, $remote, $e)
 # $message is an edi_message object
 #
+# This method has lots of logic to process ORDRSP messages (and theoretically
+# OSTRPT messages) and to make changes based on those to EG acq objects.
+# If it gets an INVOIC message, it hands that off to
+# create_acq_invoice_from_edi() following a new model (this code all wants
+# cleaned-up/refactored).
+#
+# This method currently returns an array of message objects, but no callers use
+# that except in a boolean evaluation to test for success.  So don't count on
+# that array being there or containing anything specific in the future: it
+# might get changed.
 sub process_jedi {
     my ($class, $message, $server, $remote, $e) = @_;
     $message or return;
@@ -449,7 +463,7 @@ sub process_jedi {
 #   $obj->{body}->[0]->{ORDERS}->[0]->[0] eq 'UNH'
 
     $logger->info("EDI interchange body has " . scalar(@{$perl->{body}}) . " message(s)");
-    my @ok_msg_codes = qw/ORDRSP OSTRPT/;
+    my @ok_msg_codes = qw/ORDRSP OSTRPT INVOIC/;
     my @messages;
     my $i = 0;
     foreach my $part (@{$perl->{body}}) {
@@ -463,6 +477,20 @@ sub process_jedi {
                 $logger->warn("EDI interchange $i contains unhandled '$key' message.  Ignoring it.");
                 next;
             }
+            if ($key eq 'INVOIC') {
+                # XXX TODO Maybe subclass O::U::LooseEDI::Message as
+                # something like OpenILS::Acq::{VendorInvoice,OrderReponse},
+                # each one knowing how to read itself and update EG acq
+                # objects (not under OpenILS::Application perhaps).
+                my $invoice_message =
+                    new OpenILS::Utils::LooseEDI::Message($part->{$key});
+                push @messages, $invoice_message if
+                    $class->create_acq_invoice_from_edi(
+                        $e, $invoice_message, $remote->provider
+                    );
+                next;
+            }
+
             my $msg = __PACKAGE__->message_object($part->{$key}) or next;
             push @messages, $msg;
 
@@ -577,6 +605,255 @@ sub process_jedi {
     return \@messages;
 }
 
+
+# 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, $e, $invoice, $provider, $message) = @_;
+    # $invoice is O::U::LooseEDI::Message, representing the EDI invoice message.
+    # $provider is only a pkey
+    # $message is Fieldmapper::acq::edi_message
+
+    my $log_prefix = "create_acq_invoice_from_edi(..., <acq.edi_message #" .
+        $message->id . ">): ";
+
+    my $eg_inv = Fieldmapper::acq::invoice->new;
+
+    $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");
+
+    # Find the buyer's identifier in the invoice.
+    my $buyer_san;
+    foreach (@{$invoice->{SG2}}) {
+        my $nad = $_->{NAD}[0];
+        if ($nad->{3035} eq 'BY' and $nad->{C082}{3055} eq '91') {
+            $buyer_san = $nad->{C082}{3039};
+        }
+    }
+
+    if (not $buyer_san) {
+        $logger->error($log_prefix . "could not find buyer SAN in INVOIC");
+        return 0;
+    }
+
+    # Find the matching org unit based on SAN via 'aoa' table.
+    my $addrs =
+        $e->search_actor_org_address({valid => "t", san => $buyer_san});
+
+    if (not $addrs or not @$addrs) {
+        $logger->error(
+            $log_prefix . "couldn't find OU unit matching buyer SAN in INVOIC:".
+            $e->event
+        );
+        return 0;
+    }
+
+    # XXX Should we verify that this matches PO ordering agency later?
+    $eg_inv->receiver($addrs->[0]->org_unit);
+
+    try {
+        $eg_inv->inv_ident($invoice->{BGM}[0]{1004});
+    } catch Error with {
+        $logger->error(
+            $log_prefix . "no invoice ID # in INVOIC message; " . shift
+        );
+    }
+    return 0 unless $eg_inv->inv_ident;
+
+    my @eg_inv_entries;
+
+    # The invoice message will have once instance of segment group 25
+    # per lineitem.
+    foreach my $sg25 (@{ $invoice->{SG25} }) {
+        # quantity
+        my $c186 = $sg25->{QTY}[0]{C186};
+        my $quantity = $c186->{6060};
+        # $c186->{6411} will probably say 'PCE', but need we check it?
+
+        # identifiers (typically ISBN for us, and we may not need these)
+        my @identifiers = ();
+        #   from LIN...
+        try {
+            my $c212 = $sg25->{LIN}[0]{C212};
+            push @identifiers, [$c212->{7143}, $c212->{7140}] if
+                $c212 and ref $c212 eq 'HASH';
+        } catch Error with {
+            # move on
+        };
+
+        #   from PIA...
+        try {
+            foreach my $pia (@{ $sg25->{PIA} }) {
+                foreach my $h (@{$pia->{C212}}) {
+                    push @identifiers, [$h->{7143}, $h->{7140}];
+                }
+            }
+        } catch Error with {
+            # move on
+        };
+
+        # @identifiers now contains lists of, say,
+        # ['IB',   '0786222735'], # ISBN 10
+        # ['EN','9780786222735']  # ISBN 13
+
+        # Segment Group 26-47 are all descendants of SG25.
+
+        # Segment Group 26 concerns *lineitem* price (i.e, total for all copies
+        # on this lineitem).
+
+        my $lineitem_price = $sg25->{SG26}[0]{MOA}[0]{C516}{5004};
+
+        # Segment Group 28 concerns *unit* (lineitem detail) price.  We may
+        # not actually use this.  TBD.
+        my $per_unit_price;
+        foreach my $sg28 (@{$sg25->{SG28}}) {
+            my $c509 = $sg28->{PRI}[0]{C509};
+            my ($price_qualifier, $price_qualifier_type);
+            ($per_unit_price, $price_qualifier, $price_qualifier_type) = (
+                $c509->{5118}, $c509->{5125}, $c509->{5387}
+            );
+
+            # price_qualifier=AAA seems to be the price to use.  Otherwise,
+            # take what we can get.
+            last if $price_qualifier eq 'AAA';
+        }
+
+        # Segment Group 29 will have references to LI and PO numbers
+        my $acq_identifiers = {};
+        foreach my $sg29 (@{$sg25->{SG29}}) {
+            foreach my $rff (@{$sg29->{RFF}}) {
+                my $c506 = $rff->{C506};
+                if ($c506->{1153} eq 'ON') {
+                    $acq_identifiers->{po} = $c506->{1154};
+                } elsif ($c506->{1153} eq 'LI') {
+                    my ($po, $li) = split m./., $c506->{1154};
+                    if ($po and $li) {
+                        if ($acq_identifiers->{po}) {
+                            $logger->warn(
+                                $log_prefix .
+                                "RFFs within lineitem disagree on PO # ?"
+                            ) unless $acq_identifiers->{po} eq $po;
+                        }
+                        $acq_identifiers->{li} = $li;
+                        $acq_identifiers->{po} = $po;
+                    } else {
+                        $logger->warn(
+                            $log_prefix .
+                            "RFF 1154 doesn't match expectations (.+/.+) " .
+                            "where 1153 is 'LI'"
+                        );
+                    }
+                }
+            }
+        }
+
+        if ($acq_identifiers->{po}) {
+            # First PO number seen in INVOIC sets the purchase_order field for
+            # the entry in acq.edi_message (which model may need a rethink).
+
+            $message->purchase_order($acq_identifiers->{po}) unless
+                $message->purchase_order;
+        } else {
+            $logger->warn(
+                $log_prefix .
+                "SG29 missing or refers to no purchase order that we can tell"
+            );
+        }
+        if (not $acq_identifiers->{li}) {
+            $logger->warn(
+                $log_prefix .
+                "SG29 missing or refers to no lineitem that we can tell"
+            );
+        }
+
+        my $eg_inv_entry = Fieldmapper::acq::invoice_entry->new;
+        $eg_inv_entry->inv_item_count($quantity);
+
+        # XXX Validate by making sure the LI is on-order and belongs to
+        # the right provider and ordering agency and all that.
+        $eg_inv_entry->lineitem($acq_identifiers->{li});
+
+        # XXX Do we actually need to link to PO directly here?
+        $eg_inv_entry->purchase_order($acq_identifiers->{po});
+
+        # This is the total price for all units billed, not per-unit.
+        $eg_inv_entry->cost_billed($lineitem_price);
+
+        push @eg_inv_entries, $eg_inv_entry;
+    }
+
+    my @eg_inv_items;
+
+    # Find any taxes applied to the whole invoice.
+    try {
+        if ($invoice->{SG50}) {
+            foreach my $sg50 (@{ $invoice->{SG50} }) {
+                if ($sg50->{TAX} and $sg50->{MOA}) {
+                    my $tax_amount = $sg50->{MOA}[0]{C516}{5004};
+
+                    my $eg_inv_item = Fieldmapper::acq::invoice_item->new;
+                    $eg_inv_item->inv_item_type('TAX');
+                    $eg_inv_item->cost_billed($tax_amount);
+                    # XXX i18n somehow? or maybe omit the note.
+                    $eg_inv_item->note('Tax from electronic invoice');
+
+                    push @eg_inv_items, $eg_inv_item;
+                }
+            }
+        }
+    } catch Error with {
+        # move on
+    };
+
+    $e->xact_begin;
+
+    # 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;
+    }
+
+    # create EG invoice
+    if (not $e->create_acq_invoice($eg_inv)) {
+        $logger->error($log_prefix . "couldn't create invoice: " . $e->event);
+        return 0;
+    }
+
+    # Now we have a pkey for our EG invoice, so set the invoice field on all
+    # our entries according and create those too.
+    my $eg_inv_id = $e->data->id;
+    foreach (@eg_inv_entries) {
+        $_->invoice($eg_inv_id);
+        if (not $e->create_acq_invoice_entry($_)) {
+            $logger->error(
+                $log_prefix . "couldn't create entry against lineitem " .
+                $_->lineitem . ": " . $e->event
+            );
+            return 0;
+        }
+    }
+
+    # Create any invoice items (taxes)
+    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;
+        }
+    }
+
+    $e->xact_commit;
+    return 1;
+}
+
 # returns message object if processing should continue
 # returns false/undef value if processing should abort
 
diff --git a/Open-ILS/src/perlmods/lib/OpenILS/Utils/LooseEDI.pm b/Open-ILS/src/perlmods/lib/OpenILS/Utils/LooseEDI.pm
new file mode 100644 (file)
index 0000000..c68855c
--- /dev/null
@@ -0,0 +1,144 @@
+# The OpenILS::Utils::LooseEDI classes are an intentiaonally simplistic way to
+# represent EDI interchanges and the messages contained therein (which are in
+# turn made up of segment groups, segments, and smaller data structures).
+#
+# There is virtually no validation against EDIFACT or Editeur rules.  All we're
+# doing here is the minimum data munging against incoming JEDI that will let us
+# access segments by name without looping and searching for them (much), when
+# they're where they should be.
+#
+# Segment groups are hereinafter just "groups."  Groups can belong to other
+# groups, and segments can belong to groups, but groups cannot belong to
+# segments.
+#
+# Groups and segments at a given level always appear in
+# arrays in case there are any repeats of the the same thing at the same level.
+# Anything "less" than a segment is just copied as-is from the JEDI.
+#
+# The class you want to instantiate is OpenILS::Utils::LooseEDI::Interchange.
+# The only argument you need to give new() is the JEDI data (in string form
+# will do nicely).
+
+package OpenILS::Utils::LooseEDI::Segment; # so simple it does nothing.
+
+use strict;
+use warnings;
+
+sub new {
+    my ($class, $data) = @_;
+
+    my $self = bless $data, $class; # data is already hashref
+
+    return $self;
+}
+
+1;
+
+package OpenILS::Utils::LooseEDI::Group;
+
+use strict;
+use warnings;
+
+use OpenSRF::Utils::Logger qw/:logger/;
+
+sub new {
+    my ($class, $data) = @_;
+
+    my $self = bless {
+        data => $data
+    }, $class;
+
+    $self->load;
+
+    return $self;
+}
+
+sub load {
+    my $self = shift;
+
+    foreach (@{$self->{data}}) {
+        $logger->warn("bad element in data for " . __PACKAGE__) unless
+            @$_ == 2;
+
+        my ($left, $right) = @$_;
+        $self->{$left} ||= [];
+        push @{$self->{$left}}, $self->load_children($right);
+    }
+
+    delete $self->{data};
+}
+
+sub load_children {
+    my ($self, $thing) = @_;
+
+    if (ref $thing eq 'ARRAY') {
+        return new OpenILS::Utils::LooseEDI::Group($thing);
+    } elsif (ref $thing eq 'HASH') {
+        return new OpenILS::Utils::LooseEDI::Segment($thing);
+    } else {
+        $logger->warn("unexpected data, neither array nor hashref");
+    }
+}
+
+1;
+
+package OpenILS::Utils::LooseEDI::Message;
+
+use strict;
+use warnings;
+
+# In our unsophisticated implementation, a message is just like a segment group.
+use base 'OpenILS::Utils::LooseEDI::Group';
+
+sub message_name {
+    my ($self) = @_;
+
+    return $self->{UNH}[0]{S009}{'0065'};
+}
+
+1;
+
+package OpenILS::Utils::LooseEDI::Interchange;
+
+use strict;
+use warnings;
+
+use OpenSRF::EX qw/:try/;
+use OpenSRF::Utils::JSON;
+use OpenSRF::Utils::Logger qw/:logger/;
+
+sub new {
+    my ($class, $data) = @_;
+
+    $data = OpenSRF::Utils::JSON->JSON2perl($data) unless ref $data;
+
+    if (ref $data eq 'HASH') {
+        # Like a bad wine...
+        throw new OpenSRF::EX::Error("Interchange lacks body") unless
+            $data->{body};
+        throw new OpenSRF::EX::Error("Interchange has empty body") unless
+            ref $data->{body} eq 'ARRAY' and @{ $data->{body} };
+
+        my $self = bless {}, $class;
+
+        foreach my $part (@{ $data->{body} }) {
+            foreach my $msgname (grep /^[A-Z]/, keys %$part) {
+                $self->{$msgname} ||= [];
+                my $message =
+                    new OpenILS::Utils::LooseEDI::Message($part->{$msgname});
+                if ($msgname ne $message->message_name) {
+                    $logger->warn(
+                        "Found message thought to be named $msgname, " .
+                        "but it says " . $message->message_name
+                    );
+                }
+                push @{$self->{$msgname}}, $message;
+            }
+        }
+        return $self;
+    } else {
+        $logger->error(__PACKAGE__ . " given bad data");
+    }
+}
+
+1;
diff --git a/docs/RELEASE_NOTES_NEXT/edi-invoices.txt b/docs/RELEASE_NOTES_NEXT/edi-invoices.txt
new file mode 100644 (file)
index 0000000..f7d5418
--- /dev/null
@@ -0,0 +1,15 @@
+EDI Invoices
+============
+
+The same setup that is required today for retrieving and reacting to EDI Order
+Response messages (ORDRSP) will also react to Invoices (INVOIC).
+
+This essentially means you must have a Provider (acq.provider) configured with
+an EDI Account (acq.edi_account) containing login credentials for a vendor, you must have the edi_webrick service running (EDI translator), and you must have
+the edi_pusher script run periodically by cron.
+
+An open Evergreen invoice will be created for a each EDI Invoice message.
+Evergreen invoice entries will be created for each lineitem detected in the
+EDI message if that lineitem can be linked to a known Evergreen lineitem in
+your system.  An Evergreen invoice item will be created for a whole-invoice
+tax.