Introduce ';' as command/transaction separator
There's a slight mismatch between what users experience as a command, and how commands are defined in RevBank. Specifically, the common input "<productid> <username>" is two separate commands: the first adds the product to the cart, the second finalizes the transaction. This also means that "<productid> <username> <productid> <username>" was four separate commands, resulting in TWO transactions. That's all fine and useful, but when using this advanced input method, where input is split on whitespace, it lead to unexpected results if there are insufficient arguments for the follow-up questions of a command. For example, "take jantje 10 take pietje 10" will interpret the second "take" as the description, then "pietje" als the first command of a new transaction, and finally, "10" which is typically not a valid command. It is much more likely that the user intended two separate "take" commands and just forgot to provide the description argument, but RevBank had no way of inferring that intent. From this commit on, whenever the user intends to enter further input words beyond the one that finalizes a transaction ($cart->checkout), a ';' is required. If trailing input is present, the checkout is refused and the user gets a retry prompt. Similarly, if the user indicates the intention of having finished a command by inserting a ';' while there are insufficient words in the command line to satisfy all follow-up prompts (command arguments), the rest of the command line is rejected with a retry prompt. There is, however, still no specific requirement for a ';' separator after a command that does not finalize a transaction (e.g. "<productid> <username>" or even "<productid> x2 <productid> <username>" remains valid), or for a command that precedes a ';' to finalize a transaction (e.g. "<productid>; <username>;" is also valid). This change catches many, but not all, mistakes.
This commit is contained in:
parent
b5efbcdff9
commit
daf0077d0d
2 changed files with 57 additions and 10 deletions
|
@ -11,6 +11,12 @@ use RevBank::Users;
|
|||
use RevBank::FileIO;
|
||||
use RevBank::Cart::Entry;
|
||||
|
||||
{
|
||||
package RevBank::Cart::CheckoutProhibited;
|
||||
sub new($class, $reason) { return bless \$reason, $class; }
|
||||
sub reason($self) { return $$self; }
|
||||
}
|
||||
|
||||
sub new($class) {
|
||||
return bless { entries => [] }, $class;
|
||||
}
|
||||
|
@ -70,7 +76,21 @@ sub size($self) {
|
|||
return scalar @{ $self->{entries} };
|
||||
}
|
||||
|
||||
sub prohibit_checkout($self, $bool, $reason) {
|
||||
if ($bool) {
|
||||
$self->{prohibited} = $reason;
|
||||
} else {
|
||||
delete $self->{prohibited};
|
||||
}
|
||||
}
|
||||
|
||||
sub checkout($self, $user) {
|
||||
if ($self->{prohibited}) {
|
||||
die RevBank::Cart::CheckoutProhibited->new(
|
||||
"Cannot complete transaction: $self->{prohibited}"
|
||||
);
|
||||
}
|
||||
|
||||
if ($self->entries('refuse_checkout')) {
|
||||
$self->display;
|
||||
die "Refusing to finalize deficient transaction";
|
||||
|
|
47
revbank
47
revbank
|
@ -2,7 +2,7 @@
|
|||
|
||||
use v5.32;
|
||||
use warnings;
|
||||
use feature qw(signatures);
|
||||
use feature qw(signatures isa);
|
||||
no warnings "experimental::signatures";
|
||||
|
||||
use IO::Select;
|
||||
|
@ -17,14 +17,14 @@ use RevBank::Global;
|
|||
use RevBank::Messages;
|
||||
use RevBank::Cart;
|
||||
|
||||
our $VERSION = "4.3.0";
|
||||
our $VERSION = "5.0.0-beta";
|
||||
our %HELP1 = (
|
||||
"abort" => "Abort the current transaction",
|
||||
);
|
||||
|
||||
my @words;
|
||||
my $retry;
|
||||
my @retry;
|
||||
my @words; # input
|
||||
my $retry; # reason (text)
|
||||
my @retry; # (@accepted, $rejected, [@trailing])
|
||||
|
||||
my $one_off = 0;
|
||||
|
||||
|
@ -84,6 +84,7 @@ sub prompt($prompt, $plugins, $completions) {
|
|||
my @trailing = @{ pop @retry };
|
||||
my @rejected = pop @retry;
|
||||
my @accepted = @retry;
|
||||
s/\0SEPARATOR/;/ for @accepted, @rejected, @trailing;
|
||||
$readline->insert_text(join " ", @accepted, @rejected, @trailing);
|
||||
$readline->Attribs->{point} = @accepted ? 1 + length "@accepted" : 0;
|
||||
@retry = ();
|
||||
|
@ -131,7 +132,7 @@ RevBank::Plugins->load;
|
|||
call_hooks("startup");
|
||||
|
||||
OUTER: for (;;) {
|
||||
if (not @words) {
|
||||
if (not @words or $words[0] eq "\0SEPARATOR") {
|
||||
call_hooks("cart_changed", $cart) if $cart->changed;
|
||||
print "\n";
|
||||
}
|
||||
|
@ -169,7 +170,10 @@ OUTER: for (;;) {
|
|||
|
||||
length $input or redo PROMPT;
|
||||
|
||||
@words = ($split_input ? split(" ", $input) : $input);
|
||||
@words = ($split_input
|
||||
? map { $_ eq ';' ? "\0SEPARATOR" : $_ } grep length, split(/\s+|(;)/, $input)
|
||||
: $input
|
||||
);
|
||||
}
|
||||
|
||||
WORD: for (;;) {
|
||||
|
@ -181,8 +185,23 @@ OUTER: for (;;) {
|
|||
push @retry, $word;
|
||||
|
||||
ALL_PLUGINS: { PLUGIN: for my $plugin (@plugins) {
|
||||
next WORD if $word eq "\0SEPARATOR";
|
||||
|
||||
$cart->prohibit_checkout(
|
||||
@words && $words[0] ne "\0SEPARATOR",
|
||||
"unexpected trailing input (use ';' to separate transactions)."
|
||||
);
|
||||
|
||||
my ($rv, @rvargs) = eval { $plugin->$method($cart, $word) };
|
||||
if ($@) {
|
||||
if ($@ isa 'RevBank::Cart::CheckoutProhibited') {
|
||||
@words or die "Internal inconsistency"; # other cause than trailing input
|
||||
|
||||
push @retry, shift @words; # reject next word (first of trailing)
|
||||
push @retry, [@words];
|
||||
@words = ();
|
||||
$retry = $@->reason;
|
||||
redo OUTER;
|
||||
} elsif ($@) {
|
||||
call_hooks "plugin_fail", $plugin->id, "$method: $@";
|
||||
abort;
|
||||
}
|
||||
|
@ -191,13 +210,22 @@ OUTER: for (;;) {
|
|||
abort;
|
||||
}
|
||||
if (not ref $rv) {
|
||||
abort "Incomplete command." if $one_off and not @words;
|
||||
|
||||
if (@words and $words[0] eq "\0SEPARATOR") {
|
||||
push @retry, shift @words; # reject the ';'
|
||||
push @retry, [@words];
|
||||
@words = ();
|
||||
$retry = "Incomplete command (expected: $rv)";
|
||||
redo OUTER;
|
||||
}
|
||||
|
||||
$prompt = $rv;
|
||||
@plugins = $plugin;
|
||||
($method) = @rvargs;
|
||||
call_hooks "plugin_fail", $plugin->id, "$method: No method supplied"
|
||||
if not ref $method;
|
||||
|
||||
abort "Incomplete command." if $one_off and not @words;
|
||||
next WORD;
|
||||
}
|
||||
if ($rv == ABORT) {
|
||||
|
@ -212,7 +240,6 @@ OUTER: for (;;) {
|
|||
}
|
||||
if ($rv == REJECT) {
|
||||
my ($reason) = @rvargs;
|
||||
#abort if @words;
|
||||
if (@words) {
|
||||
call_hooks "retry", $plugin->id, $reason, @words ? 1 : 0;
|
||||
push @retry, [@words];
|
||||
|
|
Loading…
Add table
Reference in a new issue