Cleanup: use subroutine signatures, remove deprecated methods.
The signatures feature has been "experimental" since Perl 5.20 (May 2014), but expected to stay. After 8 years I'm ready to take the risk :) Have added Perl v5.28 (June 2018) as the minimum requirement, even though the current revbank should work with 5.20, to see if this bothers any users. Perl v5.28 is in Debian "buster", which is now oldstable.
This commit is contained in:
parent
1661661ffd
commit
eed0db7897
45 changed files with 233 additions and 444 deletions
|
@ -1,7 +1,10 @@
|
||||||
package RevBank::Amount;
|
package RevBank::Amount;
|
||||||
|
|
||||||
use v5.28;
|
use v5.28;
|
||||||
use warnings;
|
use warnings;
|
||||||
use experimental qw(signatures);
|
use feature qw(signatures);
|
||||||
|
no warnings qw(experimental::signatures);
|
||||||
|
|
||||||
use Carp qw(carp croak);
|
use Carp qw(carp croak);
|
||||||
use Scalar::Util;
|
use Scalar::Util;
|
||||||
use POSIX qw(lround);
|
use POSIX qw(lround);
|
||||||
|
|
|
@ -1,50 +1,41 @@
|
||||||
package RevBank::Cart;
|
package RevBank::Cart;
|
||||||
use strict;
|
|
||||||
|
use v5.28;
|
||||||
|
use warnings;
|
||||||
|
use feature qw(signatures);
|
||||||
|
no warnings qw(experimental::signatures);
|
||||||
|
|
||||||
use Carp ();
|
use Carp ();
|
||||||
use List::Util ();
|
use List::Util ();
|
||||||
use RevBank::Global;
|
use RevBank::Global;
|
||||||
use RevBank::Cart::Entry;
|
use RevBank::Cart::Entry;
|
||||||
|
|
||||||
sub new {
|
sub new($class) {
|
||||||
my ($class) = @_;
|
|
||||||
return bless { entries => [] }, $class;
|
return bless { entries => [] }, $class;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub add_entry {
|
sub add_entry($self, $entry) {
|
||||||
my ($self, $entry) = @_;
|
|
||||||
|
|
||||||
$self->_call_old_hooks("add", $entry);
|
|
||||||
RevBank::Plugins::call_hooks("add_entry", $self, $entry);
|
RevBank::Plugins::call_hooks("add_entry", $self, $entry);
|
||||||
|
|
||||||
push @{ $self->{entries} }, $entry;
|
push @{ $self->{entries} }, $entry;
|
||||||
$self->{changed}++;
|
$self->{changed}++;
|
||||||
$self->_call_old_hooks("added", $entry);
|
|
||||||
RevBank::Plugins::call_hooks("added_entry", $self, $entry);
|
RevBank::Plugins::call_hooks("added_entry", $self, $entry);
|
||||||
|
|
||||||
return $entry;
|
return $entry;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub add {
|
sub add($self, $amount, $description, $data = {}) {
|
||||||
# Deprecated interface: ->add($user, ...)
|
Carp::croak "Unitialized amount; possibly a deprecated call style (\$cart->add(undef, ...))"
|
||||||
if (defined $_[3] and not ref $_[3]) {
|
if not defined $amount;
|
||||||
return shift->old_add(@_);
|
Carp::croak "Non-hash data argument; possibly a deprecated call style (\$cart->add(\$user, ...)"
|
||||||
}
|
if @_ == 4 and not ref $data;
|
||||||
|
Carp::croak "Missing description; possibly a deprecated call style (\$cart->add(\$entry); use add_entry instead)"
|
||||||
|
if not defined $description;
|
||||||
|
|
||||||
# ->add($entry)
|
|
||||||
if (@_ == 2) {
|
|
||||||
my ($self, $entry) = @_;
|
|
||||||
return $self->add_entry($entry);
|
|
||||||
}
|
|
||||||
|
|
||||||
# ->add($amount, ...)
|
|
||||||
my ($self, $amount, $description, $data) = @_;
|
|
||||||
return $self->add_entry(RevBank::Cart::Entry->new($amount, $description, $data));
|
return $self->add_entry(RevBank::Cart::Entry->new($amount, $description, $data));
|
||||||
}
|
}
|
||||||
|
|
||||||
sub delete {
|
sub delete($self, $entry) {
|
||||||
Carp::croak("\$cart->delete(\$user, \$index) is no longer supported") if @_ > 2;
|
|
||||||
|
|
||||||
my ($self, $entry) = @_;
|
|
||||||
my $entries = $self->{entries};
|
my $entries = $self->{entries};
|
||||||
|
|
||||||
my $oldnum = @$entries;
|
my $oldnum = @$entries;
|
||||||
|
@ -54,27 +45,20 @@ sub delete {
|
||||||
return $oldnum - @$entries;
|
return $oldnum - @$entries;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub empty {
|
sub empty($self) {
|
||||||
my ($self) = @_;
|
|
||||||
|
|
||||||
$self->{entries} = [];
|
$self->{entries} = [];
|
||||||
$self->{changed}++;
|
$self->{changed}++;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub display {
|
sub display($self, $prefix = "") {
|
||||||
my ($self, $prefix) = @_;
|
|
||||||
$prefix //= "";
|
|
||||||
say "$prefix$_" for map $_->as_printable, @{ $self->{entries} };
|
say "$prefix$_" for map $_->as_printable, @{ $self->{entries} };
|
||||||
}
|
}
|
||||||
|
|
||||||
sub size {
|
sub size($self) {
|
||||||
my ($self) = @_;
|
|
||||||
return scalar @{ $self->{entries} };
|
return scalar @{ $self->{entries} };
|
||||||
}
|
}
|
||||||
|
|
||||||
sub checkout {
|
sub checkout($self, $user) {
|
||||||
my ($self, $user) = @_;
|
|
||||||
|
|
||||||
if ($self->entries('refuse_checkout')) {
|
if ($self->entries('refuse_checkout')) {
|
||||||
warn "Refusing to finalize deficient transaction.\n";
|
warn "Refusing to finalize deficient transaction.\n";
|
||||||
$self->display;
|
$self->display;
|
||||||
|
@ -108,17 +92,13 @@ sub checkout {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub entries {
|
sub entries($self, $attribute = undef) {
|
||||||
my ($self, $attribute) = @_;
|
|
||||||
|
|
||||||
my @entries = @{ $self->{entries} };
|
my @entries = @{ $self->{entries} };
|
||||||
return grep $_->has_attribute($attribute), @entries if defined $attribute;
|
return grep $_->has_attribute($attribute), @entries if defined $attribute;
|
||||||
return @entries;
|
return @entries;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub changed {
|
sub changed($self) {
|
||||||
my ($self) = @_;
|
|
||||||
|
|
||||||
my $changed = 0;
|
my $changed = 0;
|
||||||
for my $entry ($self->entries('changed')) {
|
for my $entry ($self->entries('changed')) {
|
||||||
$entry->attribute('changed', undef);
|
$entry->attribute('changed', undef);
|
||||||
|
@ -128,78 +108,8 @@ sub changed {
|
||||||
return $changed;
|
return $changed;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub sum {
|
sub sum($self) {
|
||||||
my ($self) = @_;
|
|
||||||
return List::Util::sum(map $_->{amount} * $_->quantity, @{ $self->{entries} });
|
return List::Util::sum(map $_->{amount} * $_->quantity, @{ $self->{entries} });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
### Old stuff, to be deleted in a future version:
|
|
||||||
|
|
||||||
sub _call_old_hooks {
|
|
||||||
my ($self, $hook, $entry) = @_;
|
|
||||||
|
|
||||||
my $data = $entry->{attributes};
|
|
||||||
|
|
||||||
for (1 .. $entry->quantity) {
|
|
||||||
for ($entry, $entry->contras) {
|
|
||||||
my $item = {
|
|
||||||
%$data,
|
|
||||||
amount => $_->{amount},
|
|
||||||
description => $_->{description},
|
|
||||||
};
|
|
||||||
|
|
||||||
RevBank::Plugins::call_hooks($hook, $self, $_->{user}, $item);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
sub as_strings {
|
|
||||||
my ($self) = @_;
|
|
||||||
Carp::carp("Plugin uses deprecated \$cart->as_strings");
|
|
||||||
|
|
||||||
return map $_->as_loggable, @{ $self->{entries} };
|
|
||||||
}
|
|
||||||
|
|
||||||
sub is_multi_user {
|
|
||||||
Carp::carp("\$cart->is_multi_user is no longer supported, ignoring");
|
|
||||||
}
|
|
||||||
|
|
||||||
sub select_items {
|
|
||||||
my ($self, $key) = @_;
|
|
||||||
Carp::carp("Plugin uses deprecated \$cart->select_items");
|
|
||||||
|
|
||||||
my @matches;
|
|
||||||
for my $entry (@{ $self->{entries} }) {
|
|
||||||
my %attributes = %{ $entry->{attributes} };
|
|
||||||
for (1 .. $entry->quantity) {
|
|
||||||
for my $item ($entry, $entry->contras) {
|
|
||||||
push @matches, { %attributes, %$item }
|
|
||||||
if @_ == 1 # No key or match given: match everything
|
|
||||||
or @_ == 2 and $entry->has_attribute($key) # Just a key
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return @matches;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub old_add {
|
|
||||||
my ($self, $user, $amount, $description, $data) = @_;
|
|
||||||
|
|
||||||
Carp::carp("Plugin uses deprecated old-style call to \$cart->add");
|
|
||||||
|
|
||||||
$data->{COMPATIBILITY} = 1;
|
|
||||||
|
|
||||||
my $entry = RevBank::Cart::Entry->new(
|
|
||||||
defined $user ? 0 : $amount,
|
|
||||||
$description,
|
|
||||||
$data
|
|
||||||
);
|
|
||||||
$entry->add_contra($user, $amount, $description) if defined $user;
|
|
||||||
$entry->{FORCE} = 1;
|
|
||||||
|
|
||||||
return $self->add_entry($entry);
|
|
||||||
}
|
|
||||||
|
|
||||||
1;
|
1;
|
||||||
|
|
|
@ -1,16 +1,14 @@
|
||||||
use strict;
|
|
||||||
|
|
||||||
package RevBank::Cart::Entry;
|
package RevBank::Cart::Entry;
|
||||||
|
|
||||||
|
use v5.28;
|
||||||
|
use warnings;
|
||||||
|
use feature qw(signatures);
|
||||||
|
no warnings qw(experimental::signatures);
|
||||||
|
|
||||||
use Carp qw(carp croak);
|
use Carp qw(carp croak);
|
||||||
use List::Util ();
|
use List::Util ();
|
||||||
|
|
||||||
sub new {
|
sub new($class, $amount, $description, $attributes = {}) {
|
||||||
my ($class, $amount, $description, $attributes) = @_;
|
|
||||||
|
|
||||||
@_ >= 3 or croak "Not enough arguments for RevBank::Cart::Entry->new";
|
|
||||||
$attributes //= {};
|
|
||||||
|
|
||||||
$amount = RevBank::Amount->parse_string($amount) if not ref $amount;
|
$amount = RevBank::Amount->parse_string($amount) if not ref $amount;
|
||||||
|
|
||||||
my $self = {
|
my $self = {
|
||||||
|
@ -26,9 +24,7 @@ sub new {
|
||||||
return bless $self, $class;
|
return bless $self, $class;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub add_contra {
|
sub add_contra($self, $user, $amount, $description) {
|
||||||
my ($self, $user, $amount, $description) = @_;
|
|
||||||
|
|
||||||
$amount = RevBank::Amount->parse_string($amount) if not ref $amount;
|
$amount = RevBank::Amount->parse_string($amount) if not ref $amount;
|
||||||
|
|
||||||
$description =~ s/\$you/$self->{user}/g if defined $self->{user};
|
$description =~ s/\$you/$self->{user}/g if defined $self->{user};
|
||||||
|
@ -42,26 +38,20 @@ sub add_contra {
|
||||||
$self->attribute('changed', 1);
|
$self->attribute('changed', 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
sub has_attribute {
|
sub has_attribute($self, $key) {
|
||||||
my ($self, $key) = @_;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
exists $self->{attributes}->{$key}
|
exists $self->{attributes}->{$key}
|
||||||
and defined $self->{attributes}->{$key}
|
and defined $self->{attributes}->{$key}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
sub attribute {
|
sub attribute($self, $key, $new = undef) {
|
||||||
my ($self, $key, $new) = @_;
|
|
||||||
|
|
||||||
my $ref = \$self->{attributes}->{$key};
|
my $ref = \$self->{attributes}->{$key};
|
||||||
$$ref = $new if @_ > 2;
|
$$ref = $new if @_ > 2;
|
||||||
return $$ref;
|
return $$ref;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub quantity {
|
sub quantity($self, $new = undef) {
|
||||||
my ($self, $new) = @_;
|
|
||||||
|
|
||||||
my $ref = \$self->{quantity};
|
my $ref = \$self->{quantity};
|
||||||
if (defined $new) {
|
if (defined $new) {
|
||||||
$new >= 0 or croak "Quantity must be positive";
|
$new >= 0 or croak "Quantity must be positive";
|
||||||
|
@ -72,22 +62,16 @@ sub quantity {
|
||||||
return $$ref;
|
return $$ref;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub multiplied {
|
sub multiplied($self) {
|
||||||
my ($self) = @_;
|
|
||||||
|
|
||||||
return $self->{quantity} != 1;
|
return $self->{quantity} != 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub contras {
|
sub contras($self) {
|
||||||
my ($self) = @_;
|
|
||||||
|
|
||||||
# Shallow copy suffices for now, because there is no depth.
|
# Shallow copy suffices for now, because there is no depth.
|
||||||
return map +{ %$_ }, @{ $self->{contras} };
|
return map +{ %$_ }, @{ $self->{contras} };
|
||||||
}
|
}
|
||||||
|
|
||||||
sub as_printable {
|
sub as_printable($self) {
|
||||||
my ($self) = @_;
|
|
||||||
|
|
||||||
$self->sanity_check;
|
$self->sanity_check;
|
||||||
|
|
||||||
my @s;
|
my @s;
|
||||||
|
@ -113,9 +97,7 @@ sub as_printable {
|
||||||
return @s;
|
return @s;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub as_loggable {
|
sub as_loggable($self) {
|
||||||
my ($self) = @_;
|
|
||||||
|
|
||||||
croak "Loggable called before set_user" if not defined $self->{user};
|
croak "Loggable called before set_user" if not defined $self->{user};
|
||||||
$self->sanity_check;
|
$self->sanity_check;
|
||||||
|
|
||||||
|
@ -143,9 +125,7 @@ sub as_loggable {
|
||||||
return @s;
|
return @s;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub user {
|
sub user($self, $new = undef) {
|
||||||
my ($self, $new) = @_;
|
|
||||||
|
|
||||||
if (defined $new) {
|
if (defined $new) {
|
||||||
croak "User can only be set once" if defined $self->{user};
|
croak "User can only be set once" if defined $self->{user};
|
||||||
|
|
||||||
|
@ -156,9 +136,7 @@ sub user {
|
||||||
return $self->{user};
|
return $self->{user};
|
||||||
}
|
}
|
||||||
|
|
||||||
sub sanity_check {
|
sub sanity_check($self) {
|
||||||
my ($self) = @_;
|
|
||||||
|
|
||||||
# Turnover and journals are implicit contras, so (for now) a zero sum is
|
# Turnover and journals are implicit contras, so (for now) a zero sum is
|
||||||
# not required. However, in a transaction with contras, one should at least
|
# not required. However, in a transaction with contras, one should at least
|
||||||
# not try to issue money that does not exist.
|
# not try to issue money that does not exist.
|
||||||
|
|
|
@ -1,5 +1,10 @@
|
||||||
package RevBank::Global;
|
package RevBank::Global;
|
||||||
use strict;
|
|
||||||
|
use v5.28;
|
||||||
|
use warnings;
|
||||||
|
use feature qw(signatures);
|
||||||
|
no warnings qw(experimental::signatures);
|
||||||
|
|
||||||
use POSIX qw(strftime);
|
use POSIX qw(strftime);
|
||||||
use RevBank::Amount;
|
use RevBank::Amount;
|
||||||
|
|
||||||
|
@ -14,8 +19,7 @@ sub import {
|
||||||
*{"$caller\::NEXT"} = sub () { \4 };
|
*{"$caller\::NEXT"} = sub () { \4 };
|
||||||
*{"$caller\::DONE"} = sub () { \5 };
|
*{"$caller\::DONE"} = sub () { \5 };
|
||||||
*{"$caller\::parse_user"} = \&RevBank::Users::parse_user;
|
*{"$caller\::parse_user"} = \&RevBank::Users::parse_user;
|
||||||
*{"$caller\::parse_amount"} = sub {
|
*{"$caller\::parse_amount"} = sub ($amount) {
|
||||||
my ($amount) = @_;
|
|
||||||
defined $amount or return undef;
|
defined $amount or return undef;
|
||||||
length $amount or return undef;
|
length $amount or return undef;
|
||||||
|
|
||||||
|
@ -32,7 +36,7 @@ sub import {
|
||||||
*{"$caller\::say"} = sub {
|
*{"$caller\::say"} = sub {
|
||||||
print @_, "\n";
|
print @_, "\n";
|
||||||
};
|
};
|
||||||
*{"$caller\::now"} = sub {
|
*{"$caller\::now"} = sub () {
|
||||||
return strftime '%Y-%m-%d_%H:%M:%S', localtime;
|
return strftime '%Y-%m-%d_%H:%M:%S', localtime;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,10 @@
|
||||||
package RevBank::Messages;
|
package RevBank::Messages;
|
||||||
|
|
||||||
|
use v5.28;
|
||||||
|
use warnings;
|
||||||
|
use feature qw(signatures);
|
||||||
|
no warnings qw(experimental::signatures);
|
||||||
|
|
||||||
use RevBank::Global;
|
use RevBank::Global;
|
||||||
use base 'RevBank::Plugin';
|
use base 'RevBank::Plugin';
|
||||||
|
|
||||||
|
@ -16,13 +22,11 @@ sub hook_startup {
|
||||||
say "\e[0m\n\n\nWelcome to the RevBank Shell, version $::VERSION\n";
|
say "\e[0m\n\n\nWelcome to the RevBank Shell, version $::VERSION\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_plugin_fail {
|
sub hook_plugin_fail($class, $plugin, $error, @) {
|
||||||
my ($class, $plugin, $error) = @_;
|
|
||||||
warn "Plugin '$plugin' failed: $error\n";
|
warn "Plugin '$plugin' failed: $error\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_cart_changed {
|
sub hook_cart_changed($class, $cart, @) {
|
||||||
my ($class, $cart) = @_;
|
|
||||||
$cart->size or return;
|
$cart->size or return;
|
||||||
say "Pending:";
|
say "Pending:";
|
||||||
$cart->display;
|
$cart->display;
|
||||||
|
@ -35,23 +39,19 @@ sub hook_cart_changed {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_abort {
|
sub hook_abort($class, $cart, @) {
|
||||||
my ($class, $cart) = @_;
|
|
||||||
say "\e[1;4mABORTING TRANSACTION.\e[0m";
|
say "\e[1;4mABORTING TRANSACTION.\e[0m";
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_invalid_input {
|
sub hook_invalid_input($class, $cart, $word, @) {
|
||||||
my ($class, $cart, $word) = @_;
|
|
||||||
say "$word: No such product, user, or command.";
|
say "$word: No such product, user, or command.";
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_reject {
|
sub hook_reject($class, $plugin, $reason, $abort, @) {
|
||||||
my ($class, $plugin, $reason, $abort) = @_;
|
|
||||||
say $abort ? $reason : "$reason Enter 'abort' to abort.";
|
say $abort ? $reason : "$reason Enter 'abort' to abort.";
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_user_balance {
|
sub hook_user_balance($class, $username, $old, $delta, $new, @) {
|
||||||
my ($class, $username, $old, $delta, $new) = @_;
|
|
||||||
my $sign = $delta->cents >= 0 ? '+' : '-';
|
my $sign = $delta->cents >= 0 ? '+' : '-';
|
||||||
my $rood = $new->cents < 0 ? '31;' : '';
|
my $rood = $new->cents < 0 ? '31;' : '';
|
||||||
my $abs = $delta->abs;
|
my $abs = $delta->abs;
|
||||||
|
@ -61,8 +61,7 @@ sub hook_user_balance {
|
||||||
printf "New balance for $username: $old $sign $abs = \e[${rood}1m$new\e[0m$warn\n",
|
printf "New balance for $username: $old $sign $abs = \e[${rood}1m$new\e[0m$warn\n",
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_user_created {
|
sub hook_user_created($class, $username, @) {
|
||||||
my ($class, $username) = @_;
|
|
||||||
say "New account '$username' created.";
|
say "New account '$username' created.";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,12 +1,16 @@
|
||||||
package RevBank::Plugin;
|
package RevBank::Plugin;
|
||||||
use strict;
|
|
||||||
|
use v5.28;
|
||||||
|
use warnings;
|
||||||
|
use feature qw(signatures);
|
||||||
|
no warnings qw(experimental::signatures);
|
||||||
|
|
||||||
require RevBank::Global;
|
require RevBank::Global;
|
||||||
|
|
||||||
sub new {
|
sub new($class) {
|
||||||
my ($class) = @_;
|
|
||||||
return bless { }, $class;
|
return bless { }, $class;
|
||||||
}
|
}
|
||||||
sub command {
|
sub command($self, $cart, $command, @) {
|
||||||
return RevBank::Global::NEXT();
|
return RevBank::Global::NEXT();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,10 @@
|
||||||
package RevBank::Plugins;
|
package RevBank::Plugins;
|
||||||
use strict;
|
|
||||||
|
use v5.28;
|
||||||
|
use warnings;
|
||||||
|
use feature qw(signatures);
|
||||||
|
no warnings qw(experimental::signatures);
|
||||||
|
|
||||||
use RevBank::Eval;
|
use RevBank::Eval;
|
||||||
use RevBank::Plugin;
|
use RevBank::Plugin;
|
||||||
use RevBank::Global;
|
use RevBank::Global;
|
||||||
|
@ -8,17 +13,16 @@ our @EXPORT = qw(call_hooks load_plugins);
|
||||||
|
|
||||||
my @plugins;
|
my @plugins;
|
||||||
|
|
||||||
sub _read_file {
|
sub _read_file($fn) {
|
||||||
local (@ARGV) = @_;
|
local @ARGV = ($fn);
|
||||||
readline *ARGV;
|
readline *ARGV;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub call_hooks {
|
sub call_hooks($hook, @args) {
|
||||||
my $hook = shift;
|
|
||||||
my $method = "hook_$hook";
|
my $method = "hook_$hook";
|
||||||
for my $class (@plugins) {
|
for my $class (@plugins) {
|
||||||
if ($class->can($method)) {
|
if ($class->can($method)) {
|
||||||
my ($rv, $message) = $class->$method(@_);
|
my ($rv, $message) = $class->$method(@args);
|
||||||
|
|
||||||
if (defined $rv and ref $rv) {
|
if (defined $rv and ref $rv) {
|
||||||
main::abort($message) if $rv == ABORT;
|
main::abort($message) if $rv == ABORT;
|
||||||
|
@ -28,12 +32,12 @@ sub call_hooks {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
sub register {
|
sub register(@new_plugins) {
|
||||||
call_hooks("register", $_) for @_;
|
call_hooks("register", $_) for @new_plugins;
|
||||||
push @plugins, @_;
|
push @plugins, @new_plugins;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub load {
|
sub load($class) {
|
||||||
my @config = _read_file('revbank.plugins');
|
my @config = _read_file('revbank.plugins');
|
||||||
chomp @config;
|
chomp @config;
|
||||||
s/#.*//g for @config;
|
s/#.*//g for @config;
|
||||||
|
@ -48,12 +52,13 @@ sub load {
|
||||||
}
|
}
|
||||||
RevBank::Eval::clean_eval(qq[
|
RevBank::Eval::clean_eval(qq[
|
||||||
use strict;
|
use strict;
|
||||||
|
use feature qw(signatures);
|
||||||
|
no warnings 'experimental::signatures';
|
||||||
package $package;
|
package $package;
|
||||||
BEGIN { RevBank::Global->import; }
|
BEGIN { RevBank::Global->import; }
|
||||||
our \@ISA = qw(RevBank::Plugin);
|
our \@ISA = qw(RevBank::Plugin);
|
||||||
our \%ATTR;
|
our \%ATTR;
|
||||||
sub MODIFY_CODE_ATTRIBUTES {
|
sub MODIFY_CODE_ATTRIBUTES(\$class, \$sub, \@attrs) {
|
||||||
my (\$class, \$sub, \@attrs) = \@_;
|
|
||||||
\$ATTR{ \$sub } = "\@attrs";
|
\$ATTR{ \$sub } = "\@attrs";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -79,7 +84,7 @@ sub load {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
sub new {
|
sub new($class) {
|
||||||
return map $_->new, @plugins;
|
return map $_->new, @plugins;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -45,12 +45,12 @@ There is no protection against infinite loops. Be careful!
|
||||||
because that's canonicalised.
|
because that's canonicalised.
|
||||||
|
|
||||||
Don't do this:
|
Don't do this:
|
||||||
$cart->add($u, $a, "Bad example");
|
$entry->add_contra($u, $a, "Bad example");
|
||||||
|
|
||||||
But do this:
|
But do this:
|
||||||
$u = parse_user($u) or return REJECT, "$u: No such user.";
|
$u = parse_user($u) or return REJECT, "$u: No such user.";
|
||||||
$a = parse_amount($a) or return REJECT, "$a: Invalid amount.";
|
$a = parse_amount($a) or return REJECT, "$a: Invalid amount.";
|
||||||
$cart->add($u, $a, 'Good, except that $a is special in Perl :)');
|
$entry->add_contra($u, $a, 'Good, except that $a is special in Perl :)');
|
||||||
|
|
||||||
There are two kinds of plugin methods: input methods and hooks. A plugin may
|
There are two kinds of plugin methods: input methods and hooks. A plugin may
|
||||||
define one C<command> input method, and can have any number of hooks.
|
define one C<command> input method, and can have any number of hooks.
|
||||||
|
@ -118,31 +118,34 @@ Hooks are called as class methods. The return value MUST be either C<ABORT>,
|
||||||
which causes the ongoing transaction to be aborted, or a non-reference, which
|
which causes the ongoing transaction to be aborted, or a non-reference, which
|
||||||
will be ignored.
|
will be ignored.
|
||||||
|
|
||||||
|
Hooks SHOULD have a dummy C<@> parameter at the end of their signatures,
|
||||||
|
so they don't break when more information is added
|
||||||
|
|
||||||
The following hooks are available, with their respective arguments:
|
The following hooks are available, with their respective arguments:
|
||||||
|
|
||||||
=over 10
|
=over 10
|
||||||
|
|
||||||
=item hook_register $class, $plugin
|
=item hook_register($class, $plugin, @)
|
||||||
|
|
||||||
Called when a new plugin is registered.
|
Called when a new plugin is registered.
|
||||||
|
|
||||||
=item hook_abort $class, $cart
|
=item hook_abort($class, $cart, @)
|
||||||
|
|
||||||
Called when a transaction is being aborted, right before the shopping cart is
|
Called when a transaction is being aborted, right before the shopping cart is
|
||||||
emptied.
|
emptied.
|
||||||
|
|
||||||
=item hook_prompt $class, $cart, $prompt
|
=item hook_prompt($class, $cart, $prompt, @)
|
||||||
|
|
||||||
Called just before the user is prompted for input interactively. The prompt
|
Called just before the user is prompted for input interactively. The prompt
|
||||||
MAY be altered by the plugin.
|
MAY be altered by the plugin.
|
||||||
|
|
||||||
=item hook_input $class, $cart, $input, $split_input
|
=item hook_input($class, $cart, $input, $split_input, @)
|
||||||
|
|
||||||
Called when user input was given. C<$split_input> is a boolean that is true
|
Called when user input was given. C<$split_input> is a boolean that is true
|
||||||
if the input will be split on whitespace, rather than treated as a whole.
|
if the input will be split on whitespace, rather than treated as a whole.
|
||||||
The input MAY be altered by the plugin.
|
The input MAY be altered by the plugin.
|
||||||
|
|
||||||
=item hook_add $class, $cart, $user, $item
|
=item hook_add($class, $cart, $user, $item, @)
|
||||||
|
|
||||||
Called when something is added to the cart. Of course, like in C<< $cart->add
|
Called when something is added to the cart. Of course, like in C<< $cart->add
|
||||||
>>, C<$user> will be undef if the product is added for the current user.
|
>>, C<$user> will be undef if the product is added for the current user.
|
||||||
|
@ -153,32 +156,32 @@ item going into the cart!
|
||||||
|
|
||||||
Be careful to avoid infinite loops if you add new stuff.
|
Be careful to avoid infinite loops if you add new stuff.
|
||||||
|
|
||||||
=item hook_checkout $class, $cart, $user, $transaction_id
|
=item hook_checkout($class, $cart, $user, $transaction_id, @)
|
||||||
|
|
||||||
Called when the transaction is finalized, before accounts are updated.
|
Called when the transaction is finalized, before accounts are updated.
|
||||||
|
|
||||||
=item hook_checkout_done $class, $cart, $user, $transaction_id
|
=item hook_checkout_done($class, $cart, $user, $transaction_id, @)
|
||||||
|
|
||||||
Called when the transaction is finalized, after accounts were updated.
|
Called when the transaction is finalized, after accounts were updated.
|
||||||
|
|
||||||
=item hook_reject $class, $plugin, $reason, $abort
|
=item hook_reject($class, $plugin, $reason, $abort, @)
|
||||||
|
|
||||||
Called when input is rejected by a plugin. C<$abort> is true when the
|
Called when input is rejected by a plugin. C<$abort> is true when the
|
||||||
transaction will be aborted because of the rejection.
|
transaction will be aborted because of the rejection.
|
||||||
|
|
||||||
=item hook_invalid_input $class, $cart, $word
|
=item hook_invalid_input($class, $cart, $word, @)
|
||||||
|
|
||||||
Called when input was not recognised by any of the plugins.
|
Called when input was not recognised by any of the plugins.
|
||||||
|
|
||||||
=item hook_plugin_fail $class, $plugin, $error
|
=item hook_plugin_fail($class, $plugin, $error, @)
|
||||||
|
|
||||||
Called when a plugin fails.
|
Called when a plugin fails.
|
||||||
|
|
||||||
=item hook_user_created $class, $username
|
=item hook_user_created($class, $username, @)
|
||||||
|
|
||||||
Called when a new user account was created.
|
Called when a new user account was created.
|
||||||
|
|
||||||
=item hook_user_balance $class, $username, $old, $delta, $new, $transaction_id
|
=item hook_user_balance($class, $username, $old, $delta, $new, $transaction_id, @)
|
||||||
|
|
||||||
Called when a user account is updated.
|
Called when a user account is updated.
|
||||||
|
|
||||||
|
@ -189,7 +192,7 @@ C<RevBank::Messages>. Such a hack might look like:
|
||||||
|
|
||||||
undef &RevBank::Messages::hook_abort;
|
undef &RevBank::Messages::hook_abort;
|
||||||
|
|
||||||
sub hook_abort {
|
sub hook_abort($class, $cart, @) {
|
||||||
print "This message is much better!\n"
|
print "This message is much better!\n"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,11 +1,16 @@
|
||||||
package RevBank::Users;
|
package RevBank::Users;
|
||||||
use strict;
|
|
||||||
|
use v5.28;
|
||||||
|
use warnings;
|
||||||
|
use feature qw(signatures);
|
||||||
|
no warnings qw(experimental::signatures);
|
||||||
|
|
||||||
use RevBank::Global;
|
use RevBank::Global;
|
||||||
use RevBank::Plugins;
|
use RevBank::Plugins;
|
||||||
|
|
||||||
my $filename = "revbank.accounts";
|
my $filename = "revbank.accounts";
|
||||||
|
|
||||||
sub _read {
|
sub _read() {
|
||||||
my @users;
|
my @users;
|
||||||
open my $fh, $filename or die $!;
|
open my $fh, $filename or die $!;
|
||||||
/\S/ and push @users, [split " "] while readline $fh;
|
/\S/ and push @users, [split " "] while readline $fh;
|
||||||
|
@ -13,22 +18,19 @@ sub _read {
|
||||||
return { map { lc($_->[0]) => $_ } @users };
|
return { map { lc($_->[0]) => $_ } @users };
|
||||||
}
|
}
|
||||||
|
|
||||||
sub names {
|
sub names() {
|
||||||
return map $_->[0], values %{ _read() };
|
return map $_->[0], values %{ _read() };
|
||||||
}
|
}
|
||||||
|
|
||||||
sub balance {
|
sub balance($username) {
|
||||||
my ($name) = @_;
|
return _read()->{ lc $username }->[1];
|
||||||
return _read()->{ lc $name }->[1];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sub since {
|
sub since($username) {
|
||||||
my ($name) = @_;
|
return _read()->{ lc $username }->[3];
|
||||||
return _read()->{ lc $name }->[3];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sub create {
|
sub create($username) {
|
||||||
my ($username) = @_;
|
|
||||||
open my $fh, '>>', $filename or die $!;
|
open my $fh, '>>', $filename or die $!;
|
||||||
my $now = now();
|
my $now = now();
|
||||||
print {$fh} "$username 0.00 $now\n" or die $!;
|
print {$fh} "$username 0.00 $now\n" or die $!;
|
||||||
|
@ -36,8 +38,7 @@ sub create {
|
||||||
RevBank::Plugins::call_hooks("user_created", $username);
|
RevBank::Plugins::call_hooks("user_created", $username);
|
||||||
}
|
}
|
||||||
|
|
||||||
sub update {
|
sub update($username, $delta, $transaction_id) {
|
||||||
my ($username, $delta, $transaction_id) = @_;
|
|
||||||
open my $in, 'revbank.accounts' or die $!;
|
open my $in, 'revbank.accounts' or die $!;
|
||||||
open my $out, ">.revbank.$$" or die $!;
|
open my $out, ">.revbank.$$" or die $!;
|
||||||
my $old;
|
my $old;
|
||||||
|
@ -75,8 +76,7 @@ sub update {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
sub parse_user {
|
sub parse_user($username) {
|
||||||
my ($username) = @_;
|
|
||||||
my $users = _read();
|
my $users = _read();
|
||||||
return undef if not exists $users->{ lc $username };
|
return undef if not exists $users->{ lc $username };
|
||||||
return $users->{ lc $username }->[0];
|
return $users->{ lc $username }->[0];
|
||||||
|
|
|
@ -2,9 +2,7 @@
|
||||||
|
|
||||||
HELP "adduser <name>" => "Create an account";
|
HELP "adduser <name>" => "Create an account";
|
||||||
|
|
||||||
sub command :Tab(adduser) {
|
sub command :Tab(adduser) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
$command eq 'adduser' or return NEXT;
|
$command eq 'adduser' or return NEXT;
|
||||||
|
|
||||||
if ($cart->size) {
|
if ($cart->size) {
|
||||||
|
@ -14,9 +12,7 @@ sub command :Tab(adduser) {
|
||||||
return "Name for the new account", \&username;
|
return "Name for the new account", \&username;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub username {
|
sub username($self, $cart, $name, @) {
|
||||||
my ($self, $cart, $name) = @_;
|
|
||||||
|
|
||||||
return REJECT, "Sorry, whitespace is not allowed."
|
return REJECT, "Sorry, whitespace is not allowed."
|
||||||
if $name =~ /\s/;
|
if $name =~ /\s/;
|
||||||
|
|
||||||
|
|
|
@ -5,8 +5,7 @@
|
||||||
undef;
|
undef;
|
||||||
};
|
};
|
||||||
|
|
||||||
sub hook_abort {
|
sub hook_abort($class, $cart, $reason, @) {
|
||||||
my ($self, $cart, $reason) = @_;
|
|
||||||
return if not $reason or not @$reason;
|
return if not $reason or not @$reason;
|
||||||
return if "@$reason" eq '^C';
|
return if "@$reason" eq '^C';
|
||||||
|
|
||||||
|
|
|
@ -5,8 +5,6 @@
|
||||||
# Don't just edit this plugin. Instead, COPY this file and add yours to
|
# Don't just edit this plugin. Instead, COPY this file and add yours to
|
||||||
# revbank.plugins
|
# revbank.plugins
|
||||||
|
|
||||||
sub hook_beep {
|
sub hook_beep($class, @) {
|
||||||
my ($class) = @_;
|
|
||||||
|
|
||||||
print "\a";
|
print "\a";
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,17 +4,13 @@
|
||||||
|
|
||||||
HELP "deposit <amount>" => "Deposit into an account";
|
HELP "deposit <amount>" => "Deposit into an account";
|
||||||
|
|
||||||
sub command :Tab(deposit) {
|
sub command :Tab(deposit) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
$command eq 'deposit' or return NEXT;
|
$command eq 'deposit' or return NEXT;
|
||||||
|
|
||||||
return "Amount to deposit into your account", \&amount;
|
return "Amount to deposit into your account", \&amount;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub amount :Tab(13.37,42) {
|
sub amount :Tab(13.37,42) ($self, $cart, $amount, @) {
|
||||||
my ($self, $cart, $amount) = @_;
|
|
||||||
|
|
||||||
$self->{amount} = parse_amount($amount)
|
$self->{amount} = parse_amount($amount)
|
||||||
or return REJECT, "Invalid amount";
|
or return REJECT, "Invalid amount";
|
||||||
|
|
||||||
|
@ -27,9 +23,7 @@ sub amount :Tab(13.37,42) {
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub how :Tab(&how_tab) {
|
sub how :Tab(&how_tab) ($self, $cart, $input, @) {
|
||||||
my ($self, $cart, $input) = @_;
|
|
||||||
|
|
||||||
my %methods = %{ $self->{deposit_methods} };
|
my %methods = %{ $self->{deposit_methods} };
|
||||||
|
|
||||||
my $how = $self->{how} = $methods{$input}
|
my $how = $self->{how} = $methods{$input}
|
||||||
|
@ -45,14 +39,11 @@ sub how :Tab(&how_tab) {
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub how_tab {
|
sub how_tab($self, @) {
|
||||||
my ($self) = @_;
|
|
||||||
return keys %{ $self->{deposit_methods} };
|
return keys %{ $self->{deposit_methods} };
|
||||||
}
|
}
|
||||||
|
|
||||||
sub how_prompt {
|
sub how_prompt($self, $cart, $input, @) {
|
||||||
my ($self, $cart, $input) = @_;
|
|
||||||
|
|
||||||
my $how = $self->{how};
|
my $how = $self->{how};
|
||||||
|
|
||||||
push @{ $how->{answers} }, $input;
|
push @{ $how->{answers} }, $input;
|
||||||
|
|
|
@ -23,9 +23,7 @@ use List::Util qw(sum);
|
||||||
my $iban = "NL99ABCD1234567890";
|
my $iban = "NL99ABCD1234567890";
|
||||||
my $beneficiary = "Account Name";
|
my $beneficiary = "Account Name";
|
||||||
|
|
||||||
sub hook_checkout {
|
sub hook_checkout($class, $cart, $user, $transaction_id, @) {
|
||||||
my ($class, $cart, $user, $transaction_id) = @_;
|
|
||||||
|
|
||||||
my @entries = $cart->entries("is_deposit");
|
my @entries = $cart->entries("is_deposit");
|
||||||
|
|
||||||
my $amount = sum map $_->{amount}, grep $_->attribute('method') eq 'iban', @entries;
|
my $amount = sum map $_->{amount}, grep $_->attribute('method') eq 'iban', @entries;
|
||||||
|
|
|
@ -1,8 +1,6 @@
|
||||||
#!perl
|
#!perl
|
||||||
|
|
||||||
sub hook_deposit_methods {
|
sub hook_deposit_methods($class, $message, $hash, @) {
|
||||||
my ($class, $message, $hash) = @_;
|
|
||||||
|
|
||||||
$$message = <<"END";
|
$$message = <<"END";
|
||||||
|
|
||||||
Please type one of the following:
|
Please type one of the following:
|
||||||
|
|
|
@ -4,9 +4,7 @@ HELP "dinnerbonus" => "Add fee for cooking supplies";
|
||||||
|
|
||||||
my $bonus = 1.00;
|
my $bonus = 1.00;
|
||||||
|
|
||||||
sub command :Tab(kookbonus,dinnerbonus) {
|
sub command :Tab(kookbonus,dinnerbonus) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
my @users = map $_->{user}, map $_->contras, $cart->entries('is_take');
|
my @users = map $_->{user}, map $_->contras, $cart->entries('is_take');
|
||||||
|
|
||||||
(@users and $command eq 'kookpotje') # common mistake promoted to feature
|
(@users and $command eq 'kookpotje') # common mistake promoted to feature
|
||||||
|
|
16
plugins/give
16
plugins/give
|
@ -2,35 +2,27 @@
|
||||||
|
|
||||||
HELP "give <account> <amount> [<reason>]" => "Transfer money to user's account";
|
HELP "give <account> <amount> [<reason>]" => "Transfer money to user's account";
|
||||||
|
|
||||||
sub command :Tab(give) {
|
sub command :Tab(give) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
return NEXT if $command ne 'give';
|
return NEXT if $command ne 'give';
|
||||||
|
|
||||||
return "Beneficiary", \&beneficiary;
|
return "Beneficiary", \&beneficiary;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub beneficiary :Tab(USERS) {
|
sub beneficiary :Tab(USERS) ($self, $cart, $input, @) {
|
||||||
my ($self, $cart, $input) = @_;
|
|
||||||
|
|
||||||
$self->{beneficiary} = parse_user($input)
|
$self->{beneficiary} = parse_user($input)
|
||||||
or return REJECT, "$input: No such user.";
|
or return REJECT, "$input: No such user.";
|
||||||
|
|
||||||
return "Amount to give to $self->{beneficiary}", \&amount;
|
return "Amount to give to $self->{beneficiary}", \&amount;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub amount {
|
sub amount($self, $cart, $input, @) {
|
||||||
my ($self, $cart, $input) = @_;
|
|
||||||
|
|
||||||
$self->{amount} = parse_amount($input)
|
$self->{amount} = parse_amount($input)
|
||||||
or return REJECT, "$input: Invalid amount.";
|
or return REJECT, "$input: Invalid amount.";
|
||||||
|
|
||||||
return "Why are you giving $self->{amount} to $self->{beneficiary}, or enter your username to end", \&reason;
|
return "Why are you giving $self->{amount} to $self->{beneficiary}, or enter your username to end", \&reason;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub reason :Tab(whatevah) {
|
sub reason :Tab(whatevah) ($self, $cart, $input, @) {
|
||||||
my ($self, $cart, $input) = @_;
|
|
||||||
|
|
||||||
my $beneficiary = $self->{beneficiary};
|
my $beneficiary = $self->{beneficiary};
|
||||||
my $amount = $self->{amount};
|
my $amount = $self->{amount};
|
||||||
|
|
||||||
|
|
|
@ -2,9 +2,7 @@
|
||||||
|
|
||||||
HELP "grandtotal" => "Summary of all accounts";
|
HELP "grandtotal" => "Summary of all accounts";
|
||||||
|
|
||||||
sub command :Tab(grandtotal) {
|
sub command :Tab(grandtotal) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
return NEXT if $command ne 'grandtotal';
|
return NEXT if $command ne 'grandtotal';
|
||||||
|
|
||||||
my $pos = 0;
|
my $pos = 0;
|
||||||
|
|
|
@ -8,9 +8,7 @@ my $bold = "\e[1m";
|
||||||
my $underline = "\e[4m";
|
my $underline = "\e[4m";
|
||||||
my $off = "\e[0m";
|
my $off = "\e[0m";
|
||||||
|
|
||||||
sub command :Tab(help,wtf,omgwtfbbq) {
|
sub command :Tab(help,wtf,omgwtfbbq) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
return NEXT if $command !~ /^(?:help|wtf|omgwtfbbq)$/;
|
return NEXT if $command !~ /^(?:help|wtf|omgwtfbbq)$/;
|
||||||
|
|
||||||
# GNU less(1) and more(1) are a bad choice to present to total newbies who
|
# GNU less(1) and more(1) are a bad choice to present to total newbies who
|
||||||
|
|
37
plugins/idle
37
plugins/idle
|
@ -2,28 +2,27 @@
|
||||||
|
|
||||||
my $timeout = 10;
|
my $timeout = 10;
|
||||||
|
|
||||||
sub hook_prompt_idle {
|
sub hook_prompt_idle($class, $cart, $plugin, $seconds, $readline, @) {
|
||||||
my ($class, $cart, $plugin, $seconds, $readline) = @_;
|
return unless $seconds >= $timeout and $cart->size and not $plugin;
|
||||||
if ($seconds >= $timeout and $cart->size and not $plugin) {
|
|
||||||
call_hooks("beep");
|
|
||||||
|
|
||||||
return if $seconds > $timeout; # text only once
|
call_hooks("beep");
|
||||||
|
|
||||||
my $text = $readline->copy_text;
|
return if $seconds > $timeout; # text only once
|
||||||
my $point = $readline->{point};
|
|
||||||
|
|
||||||
$readline->save_prompt;
|
my $text = $readline->copy_text;
|
||||||
$readline->replace_line("");
|
my $point = $readline->{point};
|
||||||
$readline->redisplay;
|
|
||||||
|
|
||||||
my $help = $cart->entries('refuse_checkout')
|
$readline->save_prompt;
|
||||||
? "Enter 'abort' to abort."
|
$readline->replace_line("");
|
||||||
: "Enter username to pay/finish or 'abort' to abort.";
|
$readline->redisplay;
|
||||||
print "\e[33;2;1mTransaction incomplete.\e[0m $help\n";
|
|
||||||
|
|
||||||
$readline->restore_prompt;
|
my $help = $cart->entries('refuse_checkout')
|
||||||
$readline->replace_line($text);
|
? "Enter 'abort' to abort."
|
||||||
$readline->{point} = $point;
|
: "Enter username to pay/finish or 'abort' to abort.";
|
||||||
$readline->redisplay;
|
print "\e[33;2;1mTransaction incomplete.\e[0m $help\n";
|
||||||
}
|
|
||||||
|
$readline->restore_prompt;
|
||||||
|
$readline->replace_line($text);
|
||||||
|
$readline->{point} = $point;
|
||||||
|
$readline->redisplay;
|
||||||
}
|
}
|
||||||
|
|
27
plugins/log
27
plugins/log
|
@ -9,48 +9,41 @@ sub _log {
|
||||||
}
|
}
|
||||||
|
|
||||||
my %buffer;
|
my %buffer;
|
||||||
sub hook_abort {
|
sub hook_abort(@) {
|
||||||
_log("ABORT");
|
_log("ABORT");
|
||||||
}
|
}
|
||||||
sub hook_prompt {
|
sub hook_prompt($class, $cart, $prompt, @) {
|
||||||
my ($class, $cart, $prompt) = @_;
|
|
||||||
$buffer{prompt} = $prompt;
|
$buffer{prompt} = $prompt;
|
||||||
}
|
}
|
||||||
sub hook_input {
|
|
||||||
my ($class, $cart, $input, $split_input) = @_;
|
sub hook_input($class, $cart, $input, $split_input, @) {
|
||||||
$input //= "(UNDEF)";
|
$input //= "(UNDEF)";
|
||||||
_log("PROMPT $buffer{prompt} >> $input");
|
_log("PROMPT $buffer{prompt} >> $input");
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_reject {
|
sub hook_reject($class, $plugin, $reason, $abort, @) {
|
||||||
my ($class, $plugin, $reason, $abort) = @_;
|
|
||||||
_log("REJECT [$plugin] $reason");
|
_log("REJECT [$plugin] $reason");
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_retry {
|
sub hook_retry($class, $plugin, $reason, $abort, @) {
|
||||||
my ($class, $plugin, $reason, $abort) = @_;
|
|
||||||
_log("RETRY [$plugin] $reason");
|
_log("RETRY [$plugin] $reason");
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_user_created {
|
sub hook_user_created($class, $username, @) {
|
||||||
my ($class, $username) = @_;
|
|
||||||
_log("NEWUSER $username");
|
_log("NEWUSER $username");
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_user_balance {
|
sub hook_user_balance($class, $user, $old, $delta, $new, $transaction_id, @) {
|
||||||
my ($class, $user, $old, $delta, $new, $transaction_id) = @_;
|
|
||||||
my $lost = $delta < 0 ? "lost" : "got";
|
my $lost = $delta < 0 ? "lost" : "got";
|
||||||
$delta = $delta->abs;
|
$delta = $delta->abs;
|
||||||
$_ = $_->string("+") for $old, $new;
|
$_ = $_->string("+") for $old, $new;
|
||||||
_log("BALANCE $transaction_id $user had $old, $lost $delta, now has $new");
|
_log("BALANCE $transaction_id $user had $old, $lost $delta, now has $new");
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_checkout {
|
sub hook_checkout($class, $cart, $username, $transaction_id, @) {
|
||||||
my ($class, $cart, $username, $transaction_id) = @_;
|
|
||||||
_log("CHECKOUT $transaction_id $_") for map $_->as_loggable, $cart->entries;
|
_log("CHECKOUT $transaction_id $_") for map $_->as_loggable, $cart->entries;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_register {
|
sub hook_register($class, $plugin, @) {
|
||||||
my ($class, $plugin) = @_;
|
|
||||||
_log("REGISTER $plugin");
|
_log("REGISTER $plugin");
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@ HELP "market" => "Edit market list";
|
||||||
|
|
||||||
my $filename = 'revbank.market';
|
my $filename = 'revbank.market';
|
||||||
|
|
||||||
sub _read_market {
|
sub _read_market() {
|
||||||
open my $fh, '<', $filename or die "$filename: $!";
|
open my $fh, '<', $filename or die "$filename: $!";
|
||||||
my %market;
|
my %market;
|
||||||
while (readline $fh) {
|
while (readline $fh) {
|
||||||
|
@ -22,9 +22,7 @@ sub _read_market {
|
||||||
return \%market;
|
return \%market;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub command :Tab(market,&tab) {
|
sub command :Tab(market,&tab) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
if ($command eq 'market') {
|
if ($command eq 'market') {
|
||||||
system $ENV{EDITOR} || 'vi', $filename;
|
system $ENV{EDITOR} || 'vi', $filename;
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
|
|
|
@ -9,23 +9,20 @@ HELP "pfand" => "Pfand zurueck";
|
||||||
# The file format for 'revbank.pfand' is simply two whitespace separated
|
# The file format for 'revbank.pfand' is simply two whitespace separated
|
||||||
# columns: product id and pfand amount.
|
# columns: product id and pfand amount.
|
||||||
|
|
||||||
sub _read_pfand {
|
sub _read_pfand() {
|
||||||
open my $fh, 'revbank.pfand' or die $!;
|
open my $fh, 'revbank.pfand' or die $!;
|
||||||
return {
|
return {
|
||||||
map { split " " } grep /\S/, grep !/^\s*#/, readline $fh
|
map { split " " } grep /\S/, grep !/^\s*#/, readline $fh
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
sub command :Tab(pfand) {
|
sub command :Tab(pfand) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
return NEXT if $command ne 'pfand';
|
return NEXT if $command ne 'pfand';
|
||||||
|
|
||||||
return "Pfand zurueck fuer", \&product;
|
return "Pfand zurueck fuer", \&product;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub product :Tab(&tab) {
|
sub product :Tab(&tab) ($self, $cart, $product, @) {
|
||||||
my ($self, $cart, $product) = @_;
|
|
||||||
my $pfand = parse_amount(_read_pfand->{ $product })
|
my $pfand = parse_amount(_read_pfand->{ $product })
|
||||||
or return REJECT, "Invalid pfand amount for $product";
|
or return REJECT, "Invalid pfand amount for $product";
|
||||||
|
|
||||||
|
@ -41,8 +38,7 @@ sub tab {
|
||||||
return keys %{ _read_pfand() };
|
return keys %{ _read_pfand() };
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_add_entry {
|
sub hook_add_entry ($class, $cart, $entry, @) {
|
||||||
my ($class, $cart, $entry) = @_;
|
|
||||||
return if $entry->has_attribute('is_return');
|
return if $entry->has_attribute('is_return');
|
||||||
return if not $entry->has_attribute('product_id');
|
return if not $entry->has_attribute('product_id');
|
||||||
|
|
||||||
|
|
|
@ -6,9 +6,7 @@ my $limit = 200;
|
||||||
my $err_limit = "Repetition is limited at $limit items.";
|
my $err_limit = "Repetition is limited at $limit items.";
|
||||||
my $err_pfand = "Plugins 'pfand' and 'repeat' cannot be combined.";
|
my $err_pfand = "Plugins 'pfand' and 'repeat' cannot be combined.";
|
||||||
|
|
||||||
sub command {
|
sub command($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
return ABORT, $err_pfand if $cart->entries('is_pfand');
|
return ABORT, $err_pfand if $cart->entries('is_pfand');
|
||||||
|
|
||||||
my ($post) = $command =~ /^\+(\d+)?$/
|
my ($post) = $command =~ /^\+(\d+)?$/
|
||||||
|
@ -30,9 +28,7 @@ sub command {
|
||||||
return "Add to previous product", \&add;
|
return "Add to previous product", \&add;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub add {
|
sub add($self, $cart, $arg, @) {
|
||||||
my ($self, $cart, $arg) = @_;
|
|
||||||
|
|
||||||
$arg =~ /^\d+$/ and $arg > 0
|
$arg =~ /^\d+$/ and $arg > 0
|
||||||
or return REJECT, "Invalid value.";
|
or return REJECT, "Invalid value.";
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ HELP "edit" => "Edit product list";
|
||||||
|
|
||||||
my $filename = 'revbank.products';
|
my $filename = 'revbank.products';
|
||||||
|
|
||||||
sub _read_products {
|
sub _read_products() {
|
||||||
open my $fh, '<', $filename or die "$filename: $!";
|
open my $fh, '<', $filename or die "$filename: $!";
|
||||||
my %products;
|
my %products;
|
||||||
while (readline $fh) {
|
while (readline $fh) {
|
||||||
|
@ -22,9 +22,7 @@ sub _read_products {
|
||||||
return \%products;
|
return \%products;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub command :Tab(edit,&tab) {
|
sub command :Tab(edit,&tab) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
if ($command eq 'edit') {
|
if ($command eq 'edit') {
|
||||||
system $ENV{EDITOR} || 'vi', $filename;
|
system $ENV{EDITOR} || 'vi', $filename;
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
|
|
|
@ -10,9 +10,7 @@ my $err_postfix = "Addition/substraction is only supported the other way around.
|
||||||
my $limit = 200;
|
my $limit = 200;
|
||||||
my $err_limit = "Repetition is limited at $limit items.";
|
my $err_limit = "Repetition is limited at $limit items.";
|
||||||
|
|
||||||
sub command {
|
sub command($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
return ABORT, $err_pfand if $cart->entries('is_pfand');
|
return ABORT, $err_pfand if $cart->entries('is_pfand');
|
||||||
|
|
||||||
my ($lhs, $op, $rhs) = $command =~ /^(\d+)?([x*+-])(\d+)?$/
|
my ($lhs, $op, $rhs) = $command =~ /^(\d+)?([x*+-])(\d+)?$/
|
||||||
|
@ -86,9 +84,7 @@ sub command {
|
||||||
return "Multiply previous product by", \&repeat;
|
return "Multiply previous product by", \&repeat;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub repeat {
|
sub repeat($self, $cart, $arg, @) {
|
||||||
my ($self, $cart, $arg) = @_;
|
|
||||||
|
|
||||||
$arg =~ /^\d+$/ and $arg > 0
|
$arg =~ /^\d+$/ and $arg > 0
|
||||||
or return REJECT, "Invalid value.";
|
or return REJECT, "Invalid value.";
|
||||||
|
|
||||||
|
@ -98,9 +94,7 @@ sub repeat {
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub plusminus {
|
sub plusminus($self, $cart, $arg, @) {
|
||||||
my ($self, $cart, $arg) = @_;
|
|
||||||
|
|
||||||
$arg =~ /^\d+$/ and $arg > 0
|
$arg =~ /^\d+$/ and $arg > 0
|
||||||
or return REJECT, "Invalid value.";
|
or return REJECT, "Invalid value.";
|
||||||
|
|
||||||
|
@ -119,8 +113,7 @@ sub plusminus {
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_added_entry {
|
sub hook_added_entry($class, $cart, $entry, @) {
|
||||||
my ($self, $cart, $entry) = @_;
|
|
||||||
$cart->size >= 2 or return;
|
$cart->size >= 2 or return;
|
||||||
|
|
||||||
my @entries = $cart->entries;
|
my @entries = $cart->entries;
|
||||||
|
|
|
@ -2,9 +2,7 @@
|
||||||
|
|
||||||
HELP "restart" => "Attempt to restart the RevBank shell";
|
HELP "restart" => "Attempt to restart the RevBank shell";
|
||||||
|
|
||||||
sub command :Tab(restart) {
|
sub command :Tab(restart) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
return NEXT if $command ne 'restart';
|
return NEXT if $command ne 'restart';
|
||||||
|
|
||||||
no warnings;
|
no warnings;
|
||||||
|
@ -15,9 +13,7 @@ sub command :Tab(restart) {
|
||||||
return ABORT, "exec() failed. You'll have to restart revbank yourself :P";
|
return ABORT, "exec() failed. You'll have to restart revbank yourself :P";
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_input {
|
sub hook_input($class, $cart, $input, $split_input, @) {
|
||||||
my ($self, $cart, $input, $split_input) = @_;
|
|
||||||
|
|
||||||
return if defined $input;
|
return if defined $input;
|
||||||
|
|
||||||
no warnings;
|
no warnings;
|
||||||
|
|
|
@ -7,17 +7,13 @@
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
sub command :Tab(barcode) {
|
sub command :Tab(barcode) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
return NEXT if $command ne "barcode";
|
return NEXT if $command ne "barcode";
|
||||||
|
|
||||||
return "Barcode data", \&data;
|
return "Barcode data", \&data;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub data {
|
sub data($self, $cart, $input, @) {
|
||||||
my ($self, $cart, $input) = @_;
|
|
||||||
|
|
||||||
$cart->add(
|
$cart->add(
|
||||||
-0.07,
|
-0.07,
|
||||||
"Barcode <$input>",
|
"Barcode <$input>",
|
||||||
|
@ -27,9 +23,7 @@ sub data {
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_checkout {
|
sub hook_checkout($class, $cart, $username, $transaction_id, @) {
|
||||||
my ($class, $cart, $username, $transaction_id) = @_;
|
|
||||||
|
|
||||||
my @barcodes;
|
my @barcodes;
|
||||||
for my $entry ($cart->entries('is_barcode')) {
|
for my $entry ($cart->entries('is_barcode')) {
|
||||||
push @barcodes, ($entry->attribute('barcode_data')) x $entry->quantity;
|
push @barcodes, ($entry->attribute('barcode_data')) x $entry->quantity;
|
||||||
|
|
|
@ -7,9 +7,7 @@ my %bounties = (
|
||||||
4 => [ 15, "Bedankt voor 't poetsen van alle tafels" ],
|
4 => [ 15, "Bedankt voor 't poetsen van alle tafels" ],
|
||||||
);
|
);
|
||||||
|
|
||||||
sub command :Tab(BOUNTY1,BOUNTY2,BOUNTY3,BOUNTY4) {
|
sub command :Tab(BOUNTY1,BOUNTY2,BOUNTY3,BOUNTY4) ($self, $cart, $command, @) {
|
||||||
my ($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]);
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
#!perl
|
#!perl
|
||||||
|
|
||||||
sub hook_user_balance {
|
sub hook_user_balance($class, $username, $old, $delta, $new, $transaction_id, @) {
|
||||||
my ($class, $username, $old, $delta, $new, $transaction_id) = @_;
|
|
||||||
my $msg = "$transaction_id ($username)";
|
my $msg = "$transaction_id ($username)";
|
||||||
$msg =~ s/[^\x20-\x7E]//g;
|
$msg =~ s/[^\x20-\x7E]//g;
|
||||||
$msg =~ s/'//g;
|
$msg =~ s/'//g;
|
||||||
|
|
|
@ -5,8 +5,7 @@ use JSON;
|
||||||
my $ua = LWP::UserAgent->new(agent => "revbank");
|
my $ua = LWP::UserAgent->new(agent => "revbank");
|
||||||
my $backend_url = "https://deposit.revspace.nl/mollie.php";
|
my $backend_url = "https://deposit.revspace.nl/mollie.php";
|
||||||
|
|
||||||
sub backend_call {
|
sub backend_call($hash) {
|
||||||
my ($hash) = @_;
|
|
||||||
#$hash->{test} = 1; # use mollie test environment
|
#$hash->{test} = 1; # use mollie test environment
|
||||||
|
|
||||||
my $response = $ua->post($backend_url, $hash);
|
my $response = $ua->post($backend_url, $hash);
|
||||||
|
@ -20,9 +19,7 @@ sub backend_call {
|
||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub command {
|
sub command($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
# currently 10 characters after the underscore, but it's not documented.
|
# currently 10 characters after the underscore, but it's not documented.
|
||||||
my ($id) = $command =~ /^(tr_[A-Za-z0-9]{10,12})$/ or return NEXT;
|
my ($id) = $command =~ /^(tr_[A-Za-z0-9]{10,12})$/ or return NEXT;
|
||||||
|
|
||||||
|
@ -46,8 +43,7 @@ sub command {
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_abort {
|
sub hook_abort($class, $cart, $reason, @) {
|
||||||
my ($class, $cart, $reason) = @_;
|
|
||||||
# Opportunistic; ignore failures. Can't do anything about it anyway.
|
# Opportunistic; ignore failures. Can't do anything about it anyway.
|
||||||
|
|
||||||
my @ids = map $_->attribute('mollie_id'), $cart->entries('mollie_id');
|
my @ids = map $_->attribute('mollie_id'), $cart->entries('mollie_id');
|
||||||
|
@ -55,8 +51,7 @@ sub hook_abort {
|
||||||
for @ids;
|
for @ids;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_checkout {
|
sub hook_checkout($class, $cart, $user, $transaction_id, @) {
|
||||||
my ($class, $cart, $user, $transaction_id) = @_;
|
|
||||||
# Opportunistic; ignore failures. Can't do anything about it anyway.
|
# Opportunistic; ignore failures. Can't do anything about it anyway.
|
||||||
|
|
||||||
my @ids = map $_->attribute('mollie_id'), $cart->entries('mollie_id');
|
my @ids = map $_->attribute('mollie_id'), $cart->entries('mollie_id');
|
||||||
|
|
|
@ -2,8 +2,7 @@
|
||||||
|
|
||||||
use Net::MQTT::Simple "mosquitto.space.revspace.nl";
|
use Net::MQTT::Simple "mosquitto.space.revspace.nl";
|
||||||
|
|
||||||
sub hook_checkout {
|
sub hook_checkout($class, $cart, $user, $transaction_id, @) {
|
||||||
my ($class, $cart, $user, $transaction_id) = @_;
|
|
||||||
my $filename = "revbank.sales";
|
my $filename = "revbank.sales";
|
||||||
my @entries = $cart->entries('product_id') or return;
|
my @entries = $cart->entries('product_id') or return;
|
||||||
my %already_retained;
|
my %already_retained;
|
||||||
|
|
|
@ -2,17 +2,15 @@
|
||||||
|
|
||||||
use POSIX qw(strftime);
|
use POSIX qw(strftime);
|
||||||
|
|
||||||
sub _box {
|
sub _box(@lines) {
|
||||||
print(
|
print(
|
||||||
"#" x 79, "\n",
|
"#" x 79, "\n",
|
||||||
(map { sprintf("## %-73s ##\n", $_) } @_),
|
(map { sprintf("## %-73s ##\n", $_) } @lines),
|
||||||
"#" x 79, "\n"
|
"#" x 79, "\n"
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_checkout_done {
|
sub hook_checkout_done($class, $cart, $user, $transaction_id, @) {
|
||||||
my ($class, $cart, $user, $transaction_id) = @_;
|
|
||||||
|
|
||||||
defined $user or return; # hacks like 'undo' don't have an acting user
|
defined $user or return; # hacks like 'undo' don't have an acting user
|
||||||
|
|
||||||
my $balance = RevBank::Users::balance($user) or return;
|
my $balance = RevBank::Users::balance($user) or return;
|
||||||
|
|
|
@ -4,14 +4,11 @@ use List::Util ();
|
||||||
|
|
||||||
HELP "split <account>..." => "Split the bill with others";
|
HELP "split <account>..." => "Split the bill with others";
|
||||||
|
|
||||||
sub _select_split {
|
sub _select_split($cart) {
|
||||||
my ($cart) = @_;
|
|
||||||
grep $_->{amount} < 0, $cart->entries
|
grep $_->{amount} < 0, $cart->entries
|
||||||
}
|
}
|
||||||
|
|
||||||
sub command :Tab(take,steal,split) {
|
sub command :Tab(take,steal,split) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
$command eq 'split' or return NEXT;
|
$command eq 'split' or return NEXT;
|
||||||
|
|
||||||
$self->{users} = [];
|
$self->{users} = [];
|
||||||
|
@ -25,9 +22,7 @@ sub command :Tab(take,steal,split) {
|
||||||
return "User to take from (not yourself)", \&arg;
|
return "User to take from (not yourself)", \&arg;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub arg :Tab(USERS) {
|
sub arg :Tab(USERS) ($self, $cart, $arg, @) {
|
||||||
my ($self, $cart, $arg) = @_;
|
|
||||||
|
|
||||||
my $users = $self->{users};
|
my $users = $self->{users};
|
||||||
|
|
||||||
if (@$users and $arg eq $self->{split_finish}) {
|
if (@$users and $arg eq $self->{split_finish}) {
|
||||||
|
|
|
@ -2,9 +2,7 @@
|
||||||
|
|
||||||
HELP "cash" => "Checkout without a user account";
|
HELP "cash" => "Checkout without a user account";
|
||||||
|
|
||||||
sub command :Tab(cash) {
|
sub command :Tab(cash) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
return NEXT if $command ne 'cash';
|
return NEXT if $command ne 'cash';
|
||||||
|
|
||||||
call_hooks("checkout", $cart, 'cash', 0); # Fake checkout
|
call_hooks("checkout", $cart, 'cash', 0); # Fake checkout
|
||||||
|
@ -13,9 +11,7 @@ sub command :Tab(cash) {
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_checkout {
|
sub hook_checkout($class, $cart, $user, $transaction_id, @) {
|
||||||
my ($class, $cart, $user, $transaction_id) = @_;
|
|
||||||
|
|
||||||
# Hack42 for some reason used the dutch word in their revbank1 hack.
|
# Hack42 for some reason used the dutch word in their revbank1 hack.
|
||||||
my $filename = -e("revbank.voorraad")
|
my $filename = -e("revbank.voorraad")
|
||||||
? "revbank.voorraad"
|
? "revbank.voorraad"
|
||||||
|
|
|
@ -1,8 +1,6 @@
|
||||||
#!perl
|
#!perl
|
||||||
|
|
||||||
sub command :Tab(tail) {
|
sub command :Tab(tail) ($self, $cart, $command, @) = @_;
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
return NEXT if $command ne 'tail';
|
return NEXT if $command ne 'tail';
|
||||||
|
|
||||||
my $n = (`tput lines 2>/dev/null` || 13) - 3;
|
my $n = (`tput lines 2>/dev/null` || 13) - 3;
|
||||||
|
|
13
plugins/take
13
plugins/take
|
@ -2,9 +2,7 @@
|
||||||
|
|
||||||
HELP "take <account>... <amount> <reason>" => "Transfer money from them to you";
|
HELP "take <account>... <amount> <reason>" => "Transfer money from them to you";
|
||||||
|
|
||||||
sub command :Tab(take,steal) {
|
sub command :Tab(take,steal) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
$command eq 'take' or $command eq 'steal'
|
$command eq 'take' or $command eq 'steal'
|
||||||
or return NEXT;
|
or return NEXT;
|
||||||
|
|
||||||
|
@ -13,9 +11,7 @@ sub command :Tab(take,steal) {
|
||||||
return "User to take from", \&arg;
|
return "User to take from", \&arg;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub arg :Tab(USERS) {
|
sub arg :Tab(USERS) ($self, $cart, $arg, @) {
|
||||||
my ($self, $cart, $arg) = @_;
|
|
||||||
|
|
||||||
my @users = @{ $self->{users} };
|
my @users = @{ $self->{users} };
|
||||||
my $amount = parse_amount($arg);
|
my $amount = parse_amount($arg);
|
||||||
|
|
||||||
|
@ -46,9 +42,8 @@ sub arg :Tab(USERS) {
|
||||||
return "User to take from, or total amount to finish", \&arg;
|
return "User to take from, or total amount to finish", \&arg;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub reason :Tab(bbq,NOABORT) { # finish
|
# finish
|
||||||
my ($self, $cart, $reason) = @_;
|
sub reason :Tab(bbq,NOABORT) ($self, $cart, $reason, @) {
|
||||||
|
|
||||||
return REJECT, "'$reason' is a username, not a description :)."
|
return REJECT, "'$reason' is a username, not a description :)."
|
||||||
if parse_user($reason);
|
if parse_user($reason);
|
||||||
return REJECT, "'$reason' is an amount, not a description :)."
|
return REJECT, "'$reason' is an amount, not a description :)."
|
||||||
|
|
12
plugins/undo
12
plugins/undo
|
@ -4,9 +4,7 @@ HELP "undo <transactionID>" => "Undo a transaction";
|
||||||
|
|
||||||
my $filename = ".revbank.undo";
|
my $filename = ".revbank.undo";
|
||||||
|
|
||||||
sub command :Tab(undo) {
|
sub command :Tab(undo) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
$command eq 'undo' or return NEXT;
|
$command eq 'undo' or return NEXT;
|
||||||
|
|
||||||
$cart->size and return ABORT, "Undo is not available mid-transaction.";
|
$cart->size and return ABORT, "Undo is not available mid-transaction.";
|
||||||
|
@ -14,9 +12,7 @@ sub command :Tab(undo) {
|
||||||
return "Transaction ID", \&undo;
|
return "Transaction ID", \&undo;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub undo {
|
sub undo($self, $cart, $tid, @) {
|
||||||
my ($self, $cart, $tid) = @_;
|
|
||||||
|
|
||||||
open my $in, '<', $filename or die "$filename: $!";
|
open my $in, '<', $filename or die "$filename: $!";
|
||||||
open my $out, '>', "$filename.$$" or die "$filename.$$: $!";
|
open my $out, '>', "$filename.$$" or die "$filename.$$: $!";
|
||||||
my $description = "Undo $tid";
|
my $description = "Undo $tid";
|
||||||
|
@ -47,9 +43,7 @@ sub undo {
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_user_balance {
|
sub hook_user_balance($class, $username, $old, $delta, $new, $transaction_id, @) {
|
||||||
my ($class, $username, $old, $delta, $new, $transaction_id) = @_;
|
|
||||||
|
|
||||||
open my $fh, '>>', $filename or die "$filename: $!";
|
open my $fh, '>>', $filename or die "$filename: $!";
|
||||||
print {$fh} join " ", $transaction_id, $username, -$delta, now(), "\n";
|
print {$fh} join " ", $transaction_id, $username, -$delta, now(), "\n";
|
||||||
close $fh or die "$filename: $!";
|
close $fh or die "$filename: $!";
|
||||||
|
|
|
@ -2,16 +2,14 @@
|
||||||
|
|
||||||
HELP "unlisted" => "Buy unlisted product (manual entry)";
|
HELP "unlisted" => "Buy unlisted product (manual entry)";
|
||||||
|
|
||||||
sub command :Tab(unlisted,donate) {
|
sub command :Tab(unlisted,donate) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
$command eq 'unlisted' or $command eq 'donate' or return NEXT;
|
$command eq 'unlisted' or $command eq 'donate' or return NEXT;
|
||||||
$self->{command} = $command;
|
$self->{command} = $command;
|
||||||
|
|
||||||
return "Amount to deduct from your account", \&amount;
|
return "Amount to deduct from your account", \&amount;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub amount {
|
sub amount($self, $cart, $arg, @) {
|
||||||
my ($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') {
|
||||||
|
@ -22,8 +20,7 @@ sub amount {
|
||||||
return "Please provide a short description", \&description;
|
return "Please provide a short description", \&description;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub description {
|
sub description($self, $cart, $desc, @) {
|
||||||
my ($self, $cart, $desc) = @_;
|
|
||||||
$cart->add(-$self->{amount}, $desc);
|
$cart->add(-$self->{amount}, $desc);
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
#!perl
|
#!perl
|
||||||
|
|
||||||
sub command {
|
sub command($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
if ($command =~ m[^https?://]) {
|
if ($command =~ m[^https?://]) {
|
||||||
print "This is not a browser...";
|
print "This is not a browser...";
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
|
|
|
@ -4,9 +4,7 @@ HELP "<account>" => "[Pay with your account and] show balance";
|
||||||
HELP "list" => "List accounts and balances";
|
HELP "list" => "List accounts and balances";
|
||||||
HELP "shame" => "Display Hall of Shame (negative balances)";
|
HELP "shame" => "Display Hall of Shame (negative balances)";
|
||||||
|
|
||||||
sub command :Tab(list,ls,shame,USERS) {
|
sub command :Tab(list,ls,shame,USERS) ($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
return $self->list if $command eq 'list';
|
return $self->list if $command eq 'list';
|
||||||
return $self->list if $command eq 'ls';
|
return $self->list if $command eq 'ls';
|
||||||
return $self->shame if $command eq 'shame';
|
return $self->shame if $command eq 'shame';
|
||||||
|
@ -21,9 +19,7 @@ sub command :Tab(list,ls,shame,USERS) {
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_checkout {
|
sub hook_checkout($class, $cart, $user, $transaction_id, @) {
|
||||||
my ($class, $cart, $user, $transaction_id) = @_;
|
|
||||||
|
|
||||||
if ($cart->changed) {
|
if ($cart->changed) {
|
||||||
say "Done:";
|
say "Done:";
|
||||||
$cart->display;
|
$cart->display;
|
||||||
|
@ -31,26 +27,24 @@ sub hook_checkout {
|
||||||
say "Transaction ID: $transaction_id";
|
say "Transaction ID: $transaction_id";
|
||||||
}
|
}
|
||||||
|
|
||||||
sub list {
|
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 -pe's/( -[\\d.]+)/\\e[31;1m\$1\\e[0m/' | more";
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub shame {
|
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 -pe's/( -[\\d.]+)/\\e[31;1m\$1\\e[0m/' | more";
|
||||||
return ACCEPT;
|
return ACCEPT;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub recent {
|
sub _recent($n, $u) {
|
||||||
my ($n, $u) = @_;
|
|
||||||
$n += 0;
|
$n += 0;
|
||||||
print "Last $n transactions for $u:\n";
|
print "Last $n transactions for $u:\n";
|
||||||
system "perl -lane'lc(\$F[3]) eq lc(qq[\Q$u\E]) or next; s/CHECKOUT\\s+\\S+\\s+\\S+\\s+// or next; s/ #// or next; s/_/ /; print' .revbank.log | tail -n$n";
|
system "perl -lane'lc(\$F[3]) eq lc(qq[\Q$u\E]) or next; s/CHECKOUT\\s+\\S+\\s+\\S+\\s+// or next; s/ #// or next; s/_/ /; print' .revbank.log | tail -n$n";
|
||||||
}
|
}
|
||||||
|
|
||||||
sub balance {
|
sub balance($self, $u) {
|
||||||
my ($self, $u) = @_;
|
_recent(10, $u);
|
||||||
recent(10, $u);
|
|
||||||
printf "Balance for $u is \e[1m%+.2f\e[0m\n", RevBank::Users::balance($u);
|
printf "Balance for $u is \e[1m%+.2f\e[0m\n", RevBank::Users::balance($u);
|
||||||
say "NB: Products/amounts/commands FIRST, username LAST.";
|
say "NB: Products/amounts/commands FIRST, username LAST.";
|
||||||
return ABORT;
|
return ABORT;
|
||||||
|
|
|
@ -3,9 +3,7 @@
|
||||||
sub command { print "@_\n"; NEXT }
|
sub command { print "@_\n"; NEXT }
|
||||||
|
|
||||||
|
|
||||||
sub hook_user_balance {
|
sub hook_user_balance($class, $username, $old, $delta, $new, @) {
|
||||||
my ($class, $username, $old, $delta, $new) = @_;
|
|
||||||
|
|
||||||
print "c: $class\n";
|
print "c: $class\n";
|
||||||
print "u: $username\n";
|
print "u: $username\n";
|
||||||
print "o: $old\n";
|
print "o: $old\n";
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
|
|
||||||
use Time::HiRes qw(sleep);
|
use Time::HiRes qw(sleep);
|
||||||
|
|
||||||
sub _read_warnings {
|
sub _read_warnings() {
|
||||||
open my $fh, 'revbank.warnings' or die $!;
|
open my $fh, 'revbank.warnings' or die $!;
|
||||||
return map {
|
return map {
|
||||||
my ($regex, $products, $text) = m[^
|
my ($regex, $products, $text) = m[^
|
||||||
|
@ -29,8 +29,7 @@ sub _read_warnings {
|
||||||
} grep /\S/, grep !/^\s*#/, readline $fh;
|
} grep /\S/, grep !/^\s*#/, readline $fh;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub hook_add_entry {
|
sub hook_add_entry($class, $cart, $entry, @) {
|
||||||
my ($class, $cart, $entry) = @_;
|
|
||||||
return if not $entry->has_attribute('product_id'); # skip unlisted, deposit, give, take
|
return if not $entry->has_attribute('product_id'); # skip unlisted, deposit, give, take
|
||||||
|
|
||||||
my @warnings = map {
|
my @warnings = map {
|
||||||
|
|
|
@ -2,9 +2,7 @@
|
||||||
|
|
||||||
HELP "<amount>" => "Withdraw or enter price manually";
|
HELP "<amount>" => "Withdraw or enter price manually";
|
||||||
|
|
||||||
sub command {
|
sub command($self, $cart, $command, @) {
|
||||||
my ($self, $cart, $command) = @_;
|
|
||||||
|
|
||||||
my $amount = parse_amount($command);
|
my $amount = parse_amount($command);
|
||||||
defined $amount or return NEXT;
|
defined $amount or return NEXT;
|
||||||
|
|
||||||
|
|
12
revbank
12
revbank
|
@ -1,6 +1,10 @@
|
||||||
#!/usr/bin/perl -w
|
#!/usr/bin/perl
|
||||||
|
|
||||||
|
use v5.28;
|
||||||
|
use warnings;
|
||||||
|
use feature qw(signatures);
|
||||||
|
no warnings "experimental::signatures";
|
||||||
|
|
||||||
use strict;
|
|
||||||
use attributes;
|
use attributes;
|
||||||
use IO::Select;
|
use IO::Select;
|
||||||
use List::Util ();
|
use List::Util ();
|
||||||
|
@ -45,9 +49,7 @@ $select->add(\*STDIN);
|
||||||
|
|
||||||
my $cart = RevBank::Cart->new;
|
my $cart = RevBank::Cart->new;
|
||||||
|
|
||||||
sub prompt {
|
sub prompt($prompt, $plugins, $completions) {
|
||||||
my ($prompt, $plugins, $completions) = @_;
|
|
||||||
|
|
||||||
if ($prompt) {
|
if ($prompt) {
|
||||||
$prompt =~ s/$/:/ if $prompt !~ /[?>]$/;
|
$prompt =~ s/$/:/ if $prompt !~ /[?>]$/;
|
||||||
$prompt .= " ";
|
$prompt .= " ";
|
||||||
|
|
Loading…
Add table
Reference in a new issue