
(Bumps version to 3.8 because admins should update the plugin list.) Deduplication didn't work on quantified additions, i.e. if you added "20x clubmate" when there was already clubmate in the cart, it would add just ONE item, and have a lingering message that the next thing would be multiplied by 20. This old bug was especially annoying if there is a barcode "20x clubmate" to scan 20 bottles (which is the size of a crate), and this is repeated. The fix also uncovered another bug: newly added entries were selected too early. There are two hooks, hook_add_entry and hook_added_entry, and of course the selection should happen in between, not before the former. No entry in UPGRADING.md, because I think it is extremely unlikely that any plugin author will have used the selection feature yet, which is very new.
149 lines
4 KiB
Perl
149 lines
4 KiB
Perl
#!perl
|
|
|
|
HELP1 "<productID>" => "Add a product to pending transaction";
|
|
|
|
my $filename = 'revbank.products';
|
|
my $default_contra = '+sales/products';
|
|
|
|
sub read_products() {
|
|
state %products;
|
|
state $mtime;
|
|
|
|
return \%products if $mtime and $mtime == -M $filename;
|
|
$mtime = -M $filename;
|
|
|
|
my $line = 0;
|
|
|
|
for (slurp $filename) {
|
|
$line++;
|
|
|
|
s/^\s+|\s+$//g; # trim
|
|
next if /^#/;
|
|
next if not length;
|
|
|
|
my ($ids, $p, $desc) = split " ", $_, 3;
|
|
my @ids = split /,/, $ids;
|
|
|
|
$p ||= "invalid";
|
|
$desc ||= "(no description)";
|
|
|
|
my ($price, $contra) = split /\@/, $p, 2;
|
|
|
|
my $sign = $price =~ s/^-// ? -1 : 1;
|
|
my $percent = $price =~ s/%$//;
|
|
|
|
if ($percent) {
|
|
if (grep !/^\+/, @ids) {
|
|
warn "Percentage invalid for non-addon at $filename line $line.\n";
|
|
next;
|
|
}
|
|
$price = 0 + $price;
|
|
} else {
|
|
$price = eval { parse_amount($price) };
|
|
if (not defined $price) {
|
|
warn "Invalid price for '$ids[0]' at $filename line $line.\n";
|
|
next;
|
|
}
|
|
}
|
|
|
|
my @addon_ids;
|
|
unshift @addon_ids, $1 while $desc =~ s/\s+ \+ (\S+)$//x;
|
|
|
|
$products{$_} = {
|
|
id => $ids[0],
|
|
price => $sign * $price,
|
|
percent => $percent,
|
|
description => $desc,
|
|
contra => $contra || $default_contra,
|
|
_addon_ids => \@addon_ids,
|
|
line => $line,
|
|
} for @ids;
|
|
}
|
|
|
|
PRODUCT: for my $product (values %products) {
|
|
my %ids_seen = ($product->{id} => 1);
|
|
my @addon_ids = @{ $product->{_addon_ids} };
|
|
|
|
while (my $addon_id = shift @addon_ids) {
|
|
$addon_id = "+$addon_id" if exists $products{"+$addon_id"};
|
|
|
|
if ($ids_seen{$addon_id}++) {
|
|
warn "Infinite addon loop for '$product->{id}' at $filename line $product->{line}.\n";
|
|
next PRODUCT;
|
|
}
|
|
|
|
my $addon = $products{$addon_id};
|
|
if (not $addon) {
|
|
warn "Addon '$addon_id' does not exist for '$product->{id}' at $filename line $product->{line}.\n";
|
|
next PRODUCT;
|
|
}
|
|
|
|
push @{ $product->{addons} }, $addon;
|
|
push @addon_ids, @{ $addon->{_addon_ids} };
|
|
}
|
|
}
|
|
|
|
return \%products;
|
|
}
|
|
|
|
sub command :Tab(&tab) ($self, $cart, $command, @) {
|
|
$command =~ /\S/ or return NEXT;
|
|
$command =~ /^\+/ and return NEXT;
|
|
|
|
my $products = read_products;
|
|
my $product = $products->{ $command } or return NEXT;
|
|
my $price = $product->{price};
|
|
|
|
my $contra_desc = "\$you bought $product->{description}";
|
|
|
|
my @addons = @{ $product->{addons} // [] };
|
|
|
|
my $display = undef;
|
|
$display = "Product" if @addons and $price->cents > 0;
|
|
$display = "Reimbursement" if @addons and $price->cents < 0;
|
|
|
|
my $entry = $cart->add(
|
|
-$price,
|
|
$product->{description},
|
|
{
|
|
product_id => $product->{id},
|
|
plugin => $self->id,
|
|
product => $product,
|
|
deduplicate => join("/", $self->id, $product->{id}),
|
|
}
|
|
);
|
|
$entry->add_contra(
|
|
$product->{contra},
|
|
+$price,
|
|
$contra_desc,
|
|
$display
|
|
);
|
|
|
|
for my $addon (@addons) {
|
|
my $addon_price = $addon->{price};
|
|
if ($addon->{percent}) {
|
|
my $sum = List::Util::sum map {
|
|
$_->{amount}
|
|
} grep {
|
|
$_->{user} eq $addon->{contra}
|
|
} $entry->contras;
|
|
|
|
$addon_price = $addon_price / 100 * $sum;
|
|
}
|
|
|
|
$entry->amount( $entry->amount - $addon_price );
|
|
|
|
$entry->add_contra(
|
|
$addon->{contra},
|
|
$addon_price,
|
|
"$addon->{description} ($contra_desc)",
|
|
$addon->{description}
|
|
);
|
|
}
|
|
|
|
return ACCEPT;
|
|
}
|
|
|
|
sub tab {
|
|
return grep !/^\+/, grep /\D/, keys %{ read_products() };
|
|
}
|