Bump to v3.4; make all transactions balanced using hidden accounts

See UPGRADING.md for details.
This commit is contained in:
Juerd Waalboer 2022-06-11 18:51:05 +02:00
parent e3a04a0e36
commit 441bf05fde
14 changed files with 116 additions and 40 deletions

View file

@ -1,3 +1,20 @@
# (2022-06-11) RevBank 3.4
RevBank now has built-in hidden accounts and balanced transactions. These
accounts will be made automatically, and hidden from the user interface. Apart
from that, they function as normal accounts (for now).
If you have scripts that parse `.revbank.log` or `revbank.products`, you may
want to ignore all accounts that start with `-` or `+`.
In the hopefully very unlikely event that you have existing user accounts that
start with `-` or `+`, those will have to be renamed manually, as such accounts
are no longer accessible.
For your custom plugins, you may want to add `->add_contra` calls to every
`$cart->add` call that does not already have them. Unbalanced transactions will
probably be deprecated in a future version.
# (2022-06-04) RevBank 3.3 # (2022-06-04) RevBank 3.3
Raw amounts without a command are no longer supported. There was already an Raw amounts without a command are no longer supported. There was already an

View file

@ -66,11 +66,7 @@ sub checkout($self, $user) {
return; return;
} }
if ($user =~ /^[-+]/) { $user = RevBank::Users::assert_user($user);
# Hidden internal accounts
my $canonical = RevBank::Users::parse_user($user);
$user = $canonical // RevBank::Users::create($user);
}
my $entries = $self->{entries}; my $entries = $self->{entries};

View file

@ -27,6 +27,7 @@ sub new($class, $amount, $description, $attributes = {}) {
sub add_contra($self, $user, $amount, $description) { sub add_contra($self, $user, $amount, $description) {
$amount = RevBank::Amount->parse_string($amount) if not ref $amount; $amount = RevBank::Amount->parse_string($amount) if not ref $amount;
$user = RevBank::Users::assert_user($user);
$description =~ s/\$you/$self->{user}/g if defined $self->{user}; $description =~ s/\$you/$self->{user}/g if defined $self->{user};
@ -84,7 +85,7 @@ sub as_printable($self) {
push @s, sprintf "%8s %s", $self->{amount}->string_flipped, $self->{description}; push @s, sprintf "%8s %s", $self->{amount}->string_flipped, $self->{description};
for my $c ($self->contras) { for my $c ($self->contras) {
next if RevBank::Users::is_hidden($c->{user}); next if RevBank::Users::is_hidden($c->{user}) and not $ENV{REVBANK_DEBUG};
push @s, sprintf( push @s, sprintf(
"%11s %s %s", "%11s %s %s",

View file

@ -55,7 +55,7 @@ sub hook_reject($class, $plugin, $reason, $abort, @) {
} }
sub hook_user_balance($class, $username, $old, $delta, $new, @) { sub hook_user_balance($class, $username, $old, $delta, $new, @) {
return if hidden $username; return if hidden $username and not $ENV{REVBANK_DEBUG};
my $sign = $delta->cents >= 0 ? '+' : '-'; my $sign = $delta->cents >= 0 ? '+' : '-';
my $rood = $new->cents < 0 ? '31;' : ''; my $rood = $new->cents < 0 ? '31;' : '';
@ -67,7 +67,7 @@ sub hook_user_balance($class, $username, $old, $delta, $new, @) {
} }
sub hook_user_created($class, $username, @) { sub hook_user_created($class, $username, @) {
return if hidden $username; return if hidden $username and not $ENV{REVBANK_DEBUG};
say "New account '$username' created."; say "New account '$username' created.";
} }

View file

@ -7,6 +7,7 @@ no warnings qw(experimental::signatures);
use RevBank::Global; use RevBank::Global;
use RevBank::Plugins; use RevBank::Plugins;
use Carp ();
my $filename = "revbank.accounts"; my $filename = "revbank.accounts";
@ -77,16 +78,30 @@ sub update($username, $delta, $transaction_id) {
); );
} }
sub parse_user($username) {
my $users = _read();
return undef if not exists $users->{ lc $username };
return $users->{ lc $username }->[0];
}
sub is_hidden($username) { sub is_hidden($username) {
return $username =~ /^[-+]/; return $username =~ /^[-+]/;
} }
sub parse_user($username) {
return undef if is_hidden($username);
my $users = _read();
return exists $users->{ lc $username }
? $users->{ lc $username }->[0]
: undef;
}
sub assert_user($username) {
my $users = _read();
return exists $users->{ lc $username }
? $users->{ lc $username }->[0]
: (is_hidden($username)
? create($username)
: Carp::croak("Account '$username' does not exist")
);
}
1; 1;

View file

@ -19,7 +19,10 @@ sub amount :Tab(13.37,42) ($self, $cart, $amount, @) {
return $message . "How are we receiving this $amount?", \&how return $message . "How are we receiving this $amount?", \&how
if keys %{ $self->{deposit_methods} }; if keys %{ $self->{deposit_methods} };
$cart->add(+$self->{amount}, "Deposit", { is_deposit => 1 }); $cart
->add(+$self->{amount}, "Deposit", { is_deposit => 1 })
->add_contra("-deposits/other", -$self->{amount}, "Deposited by \$you");
return ACCEPT; return ACCEPT;
} }
@ -35,7 +38,15 @@ sub how :Tab(&how_tab) ($self, $cart, $input, @) {
return shift @{ $how->{prompts} }, \&how_prompt; return shift @{ $how->{prompts} }, \&how_prompt;
} }
$cart->add(+$self->{amount}, $how->{description}, { is_deposit => 1, method => $how->{_key} }); my $contra =
$how->{_key} eq 'cash' ? '-cash'
: $how->{_key} eq 'reimburse' ? '-expenses/reimbursed'
: "-deposits/$how->{_key}";
$cart
->add(+$self->{amount}, $how->{description}, { is_deposit => 1, method => $how->{_key} })
->add_contra($contra, -$self->{amount}, "$how->{description} by \$you");
return ACCEPT; return ACCEPT;
} }
@ -53,7 +64,11 @@ sub how_prompt($self, $cart, $input, @) {
} }
my $desc = sprintf $how->{description}, @{ $how->{answers} }; my $desc = sprintf $how->{description}, @{ $how->{answers} };
my $contra = $how->{_key} eq 'cash' ? '-cash' : "-deposits/$how->{_key}";
$cart
->add(+$self->{amount}, $desc, { is_deposit => 1, method => $how->{_key} })
->add_contra($contra, -$self->{amount}, "$desc by \$you");
$cart->add(+$self->{amount}, $desc, { is_deposit => 1, method => $how->{_key} });
return ACCEPT; return ACCEPT;
} }

View file

@ -27,7 +27,9 @@ sub product :Tab(&tab) ($self, $cart, $product, @) {
or return REJECT, "Invalid pfand amount for $product"; or return REJECT, "Invalid pfand amount for $product";
if ($pfand) { if ($pfand) {
$cart->add(+$pfand, "Pfand zurueck", { is_return => 1 }); $cart
->add(+$pfand, "Pfand zurueck", { is_return => 1 })
->add_contra("-pfand", -$pfand, "Pfand fuer \$you");
} else { } else {
say "$product: Kein Pfand"; say "$product: Kein Pfand";
} }
@ -44,7 +46,9 @@ sub hook_add_entry ($class, $cart, $entry, @) {
my $pfand = _read_pfand->{ $entry->attribute('product_id') } or return; my $pfand = _read_pfand->{ $entry->attribute('product_id') } or return;
$cart->add(-$pfand, "Pfand", { is_pfand => 1 }); $cart
->add(-$pfand, "Pfand", { is_pfand => 1 })
->add_contra("-pfand", +$pfand, "Pfand von \$you");
return; return;
} }

View file

@ -42,11 +42,17 @@ sub command :Tab(edit,&tab) ($self, $cart, $command, @) {
return ACCEPT; return ACCEPT;
} }
$cart->add( $cart
-$price, ->add(
$product->{description}, -$price,
{ product_id => $product->{id}, plugin => $self->id } $product->{description},
); { product_id => $product->{id}, plugin => $self->id }
)
->add_contra(
"+sales/products",
+$price,
"\$you bought $product->{description}"
);
return ACCEPT; return ACCEPT;
} }

View file

@ -14,11 +14,19 @@ sub command :Tab(barcode) ($self, $cart, $command, @) {
} }
sub data($self, $cart, $input, @) { sub data($self, $cart, $input, @) {
$cart->add( my $price = 0.07;
-0.07,
"Barcode <$input>", $cart
{ is_barcode => 1, barcode_data => $input } ->add(
); -$price,
"Barcode <$input>",
{ is_barcode => 1, barcode_data => $input }
)
->add_contra(
"+sales/barcodes",
+$price,
"\$you bought barcode <$input>"
);
return ACCEPT; return ACCEPT;
} }

View file

@ -9,7 +9,9 @@ my %bounties = (
sub command :Tab(BOUNTY1,BOUNTY2,BOUNTY3,BOUNTY4) ($self, $cart, $command, @) { sub command :Tab(BOUNTY1,BOUNTY2,BOUNTY3,BOUNTY4) ($self, $cart, $command, @) {
if ($command =~ /BOUNTY(\d+)/) { if ($command =~ /BOUNTY(\d+)/) {
$cart->add(+$bounties{$1}[0], $bounties{$1}[1]); $cart
->add(+$bounties{$1}[0], $bounties{$1}[1])
->add_contra("-expenses/bounties", -$bounties{$1}[0], "$command by \$you");
return ACCEPT; return ACCEPT;
} }

View file

@ -35,11 +35,17 @@ sub command($self, $cart, $command, @) {
$description .= " TEST MODE ($result->{test_amount})"; $description .= " TEST MODE ($result->{test_amount})";
} }
$cart->add( $cart
+$amount, ->add(
$description, +$amount,
{ is_deposit => 1, method => 'online', mollie_id => $id, no_repeat => 1 } $description,
); { is_deposit => 1, method => 'online', mollie_id => $id, no_repeat => 1 }
)
->add_contra(
"-deposits/online",
-$amount,
"$description by \$you"
);
return ACCEPT; return ACCEPT;
} }

View file

@ -13,7 +13,10 @@ sub amount($self, $cart, $arg, @) {
$self->{amount} = parse_amount($arg) or return REJECT, "Invalid amount."; $self->{amount} = parse_amount($arg) or return REJECT, "Invalid amount.";
if ($self->{command} eq 'donate') { if ($self->{command} eq 'donate') {
$cart->add(-$self->{amount}, "Donation (THANK YOU!)"); $cart
->add(-$self->{amount}, "Donation (THANK YOU!)")
->add_contra("+donations", +$self->{amount}, "Donation by \$you");
return ACCEPT; return ACCEPT;
} }
@ -21,7 +24,10 @@ sub amount($self, $cart, $arg, @) {
} }
sub description($self, $cart, $desc, @) { sub description($self, $cart, $desc, @) {
$cart->add(-$self->{amount}, "Unlisted: $desc"); $cart
->add(-$self->{amount}, "Unlisted: $desc")
->add_contra("+sales/unlisted", +$self->{amount}, "Unlisted: $desc by \$you");
return ACCEPT; return ACCEPT;
} }

View file

@ -28,12 +28,12 @@ sub hook_checkout($class, $cart, $user, $transaction_id, @) {
} }
sub list($self) { sub list($self) {
system "sort -f revbank.accounts | grep -v ^# | perl -pe's/( -[\\d.]+)/\\e[31;1m\$1\\e[0m/' | more"; system "sort -f revbank.accounts | grep -v ^# | perl -ne's/( -[\\d.]+)/\\e[31;1m\$1\\e[0m/; print if not /^[-+]/' | more";
return ACCEPT; return ACCEPT;
} }
sub shame($self) { sub shame($self) {
system "sort -k2 -n revbank.accounts | grep -v ^# | grep -- ' -' | perl -pe's/( -[\\d.]+)/\\e[31;1m\$1\\e[0m/' | more"; system "sort -k2 -n revbank.accounts | grep -v ^# | grep -- ' -' | perl -ne's/( -[\\d.]+)/\\e[31;1m\$1\\e[0m/; print if not /^[-+]/' | more";
return ACCEPT; return ACCEPT;
} }

View file

@ -18,7 +18,7 @@ use RevBank::Global;
use RevBank::Messages; use RevBank::Messages;
use RevBank::Cart; use RevBank::Cart;
our $VERSION = "3.3"; our $VERSION = "3.4";
our %HELP1 = ( our %HELP1 = (
"abort" => "Abort the current transaction", "abort" => "Abort the current transaction",
); );