revbank/lib/RevBank/Products.pm
Juerd Waalboer b22cc4c997 Move price calculation from products plugin to RevBank::Products
- Adds price tag calculation. Addons tagged #OPAQUE are excluded from the
price tag.
- BREAKING CHANGE: instead of abusing $product->{price} for a percent,
$product->{percent} is no longer a boolean but the actual percent, so
$product->{price} is the calculated amount.

The total price of a product is now calculated in two places, once when
reading the product list, and once as the result of adding the entry and
its contras when adding the product. Although this involves some
duplication and the sums are calculated in different ways, it hinges on
the existing assertion to make sure that the entry is balanced to ensure
that both sums are the same. Because of that, this code duplication
actually strengthens the integrity.
2024-12-26 01:36:55 +01:00

182 lines
5.5 KiB
Perl

package RevBank::Products;
use v5.32;
use warnings;
use experimental 'signatures'; # stable since 5.36
use RevBank::Prompt;
use RevBank::Global;
use Exporter qw(import);
our @EXPORT = qw(read_products);
sub read_products($filename = "revbank.products", $default_contra = "+sales/products") {
state %cache; # $filename => \%products
state %mtimes; # $filename => mtime
my $mtime = \$mtimes{$filename};
return $cache{$filename} if $$mtime and $$mtime == -M $filename;
my %products;
my $linenr = 0;
my $warnings = 0;
$$mtime = -M $filename;
for my $line (slurp $filename) {
$linenr++;
next if $line =~ m[
^\s*\# # comment line
|
^\s*$ # empty line, or only whitespace
]x;
my @split = RevBank::Prompt::split_input($line);
if (not @split or ref $split[0] or grep /\0/, @split) {
warn "Invalid value in $filename line $linenr.\n";
next;
}
my ($ids, $p, $desc, @extra) = @split;
my @addon_ids;
my %tags;
my $compat = 0;
if (@split == 1 and ref $split[0]) {
$compat = 1;
} else {
for (@extra) {
if (/^\+(.*)/) {
push @addon_ids, $1;
} elsif (/^\#(\w+)(=(.*))?/) {
$tags{$1} = $2 ? $3 : 1;
} else {
$compat = 1;
last;
}
}
}
if ($compat) {
$warnings++;
warn "$filename line $linenr: can't parse as new format; assuming old format.\n" if $warnings < 4;
warn "Too many warnings; suppressing the rest. See UPGRADING.md for instructions.\n" if $warnings == 4;
($ids, $p, $desc) = split " ", $line, 3;
@addon_ids = ();
unshift @addon_ids, $1 while $desc =~ s/\s+ \+ (\S+)$//x;
}
my $canonical = join " ", map RevBank::Prompt::reconstruct($_), $ids, $p, $desc, @extra;
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 $linenr.\n";
next;
}
$percent = $sign * (0 + $price);
$price = undef; # calculated later
} else {
$price = $sign * eval { parse_amount($price) };
if (not defined $price) {
warn "Invalid price for '$ids[0]' at $filename line $linenr.\n";
next;
}
}
for my $id (@ids) {
warn "Product '$id' redefined at $filename line $linenr (original at line $products{$id}{line}).\n" if exists $products{$id};
$products{$id} = {
id => $ids[0],
description => $desc,
contra => $contra || $default_contra,
_addon_ids => \@addon_ids,
line => $linenr,
tags => \%tags,
config => $canonical,
percent => $percent,
price => $price, # base price
# The following are calculated below, for top-level products only:
# tag_price => base price + sum of transparent addons
# hidden_fees => sum of opaque addons
# total_price => tag_price + hidden_fees
};
}
}
# Resolve addons
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} } }; # shallow copy to overwrite ->{price} later
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} };
}
}
# Calculate tag and total price
PRODUCT: for my $product (values %products) {
next if $product->{id} =~ /^\+/;
my $tag_price = $product->{price} || 0;
my $hidden = 0;
my @seen = ($product);
for my $addon (@{ $product->{addons} }) {
if ($addon->{percent}) {
my $sum = List::Util::sum map {
$_->{price}
} grep {
$_->{contra} eq $addon->{contra}
} @seen;
$addon->{price} = $addon->{percent} / 100 * $sum;
}
if ($addon->{tags}{OPAQUE}) {
$hidden += $addon->{price};
} else {
$tag_price += $addon->{price};
}
push @seen, $addon;
}
$product->{tag_price} = $tag_price;
$product->{hidden_fees} = $hidden;
$product->{total_price} = $tag_price + $hidden;
}
return $cache{$filename} = \%products;
}
1;