"all" and "each" transaction mode distinction; added missing batch_processor view

This commit is contained in:
Florian "flowdy" Heß 2017-02-21 22:29:13 +01:00
parent df9c0b4065
commit f54f940f74
3 changed files with 252 additions and 142 deletions

View File

@ -90,10 +90,11 @@ sub batch_processor {
; ;
open my $fh, '<', \$text; open my $fh, '<', \$text;
require "$Bin/trsr" or die; require "$Bin/trsr" or die;
$self->app->db->storage->txn_do(sub { Commands->import(
Commands::act_on_other_db($self->app->db); $self->app->db,
$self->param("txn-mode") // "all"
);
Commands::charge_account($fh); Commands::charge_account($fh);
});
$self->redirect_to('home'); $self->redirect_to('home');
} }

View File

@ -0,0 +1,57 @@
% title "Batch processor";
<form method="POST">
<h2>Incomings, outgoings, transfers</h2>
<textarea name="batch" placeholder="See help link below for a syntax description"><%=
param 'batch'
%></textarea>
% my %txn_selected = ( (param("txn-mode") // "all") => q{checked="checked"} );
<h2>Transaction mode:</h2>
<label><input type="radio" name="txn-mode" value="all" <%= $txn_selected{all} %> />
submission: keep all entries or none if there is an error</label><br>
<label><input type="radio" name="txn-mode" value="each" <%= $txn_selected{each} %> />
each entry: keep entries before any erroneous one</label>
</p>
<button type="submit">Submit</button>
</form>
% stash help => $_ for begin
<h3>What is the basic syntax?</h3>
<p>You can separate the fields of each line by a comma and optional whitespace, or at least one whitespace. The order of columns is:</p>
<ol>
<li>The booking date in format YYYY-MM-DD,</li>
<li>The account name,</li>
<li>The value if it is a debit,</li>
<li>The value if it is a credit,</li>
<li>The purpose (in the case of a debit, it must start with bill ID and colon),</li>
<li>Optional: "<< Comma-separated list of credit IDs used to pay the debt" or ">> Comma-separated list of bill IDs the booked income is used for", respectively</li>
</ol>
<h4>Multi-line purposes</h4>
<p>Multi-line purposes must either be surrounded by " (escape literal " by doubling it), or started in the next line and terminated with an empty one.</p>
<h3>How to book an incoming payment</h3>
<p>In the credit column must be a value greater than 0. Cent must be passed as a decimal part, i.e. "100" really mean 100.00, not 1.00! In the debit column you MAY input a name starting with a letter (otherwise it needs to be just "+"), by which you can refer to a credit in lines below instead of the number.</p>
<h3>How to enter a target credit</h3>
<p>A target credit is defined as initially equal 0.</p>
<h3>How to make an internal debt</h3>
<p>A debit, recognized by an amount in the debit column, can have a target credit name or number in the credit column.</p>
<h3>How to book an outgoing payment</h3>
<p>An outgoing payment must not have a target credit. Leave that field empty.</p>
% end

174
trsr
View File

@ -1,22 +1,43 @@
#!/usr/bin/env perl #!/usr/bin/env perl
use strict; use strict;
my (%OPTS, $ACCOUNT, %targets); my ($db, %OPTS, $ACCOUNT, %targets);
package Commands { package Commands {
use Text::Wrap qw(wrap); use Text::Wrap qw(wrap);
use Carp qw(croak); use Carp qw(croak);
use Scalar::Util qw(openhandle); use Scalar::Util qw(openhandle);
my $db;
use TrsrDB \$db;
use open qw/:std :encoding(utf8)/; use open qw/:std :encoding(utf8)/;
sub act_on_other_db { sub import {
$db = shift; $db = $_[1];
set_txn_mode(undef, $_[2] // "each");
} }
sub single_charge { sub cond_txn_all (&);
sub cond_txn_each (&);
sub set_txn_mode {
my (undef, $mode) = @_;
my $with_txn = sub (&) { $db->storage->txn_do(shift) };
my $no_txn = sub (&) { goto &{$_[0]} };
no warnings 'redefine';
if ( $mode eq "all" ) {
*cond_txn_all = $with_txn;
*cond_txn_each = $no_txn;
}
elsif ( $mode eq "each" ) {
*cond_txn_all = $no_txn;
*cond_txn_each = $with_txn;
}
else {
croak "Unsupported txn_mode $mode";
}
}
sub single_charge { # charge interactively
my $accounts = $db->resultset("Account"); my $accounts = $db->resultset("Account");
_require_value(); _require_value();
@ -40,17 +61,21 @@ sub single_charge {
if ( $ACCOUNT ) { if ( $ACCOUNT ) {
my $account = $accounts->find({ ID => $ACCOUNT }); my $account = $accounts->find({ ID => $ACCOUNT });
cond_txn_each {
my $record = $charger->($account, $billId); my $record = $charger->($account, $billId);
if ( !$billId and my $t = $OPTS{target} ) { if ( !$billId and my $t = $OPTS{target} ) {
$db->make_transfers( $record->ID => $t ); $db->make_transfers( $record->ID => $t );
} }
};
} }
elsif ( $billId =~ s{ @(\w+) }{ %s }xms ) { elsif ( $billId =~ s{ @(\w+) }{ %s }xms ) {
$accounts = $accounts->search({ type => $1 }); $accounts = $accounts->search({ type => $1 });
cond_txn_all {
while ( my $account = $accounts->next ) { while ( my $account = $accounts->next ) {
$charger->( $account, sprintf $billId ); cond_txn_each { $charger->( $account, sprintf $billId ); }
} }
};
} }
else { else {
@ -134,40 +159,75 @@ sub _require_target {
} }
sub charge_account { sub charge_account {
goto &single_charge if $OPTS{1}; goto &single_charge if %OPTS;
print "Enter records ...\n"; print "Enter records ...\n" if -t STDIN;
local *ARGV = openhandle($_[0]) exists $_[0]
and local *ARGV = (
openhandle($_[0])
// croak "charge_account() expects open file handle to read from" // croak "charge_account() expects open file handle to read from"
if exists $_[0]; )
and my $abort_on_exception = 1
;
# Possible line formats: # Possible line formats:
# ------------------------------------------------------- # -------------------------------------------------------
#
# Date Account Var Amount Purpose :: Credit (Payment or target) # Date Account Var Amount Purpose :: Credit (Payment or target)
# ^^^ Field may be empty, yet not stripped # ^^^ Field may be empty or '+', yet not omitted
# Date Account Amount TCr ID: Purpose :: Bill # Date Account Amount TCr ID: Purpose :: Bill
# ^^^ Field may be empty, yet not stripped # ^^^ Field may be empty, yet not omitted
# Please note: Omit Account if specified on commandline. #
# Please note: Omit Account if it is specified on commandline.
#
my @transfers; cond_txn_all {
while ( defined( my $line = <ARGV> ) ) { while ( defined( my $line = <ARGV> ) ) {
_strip_comment_from($line) // next; _strip_comment_from($line) // next;
my ($date, $account, $debit, $credit, $purpose) = $ACCOUNT
? do { my @columns = split /[, \t][ ]*/, $line, 4; my @columns
splice @columns, 1, 0, $ACCOUNT; = split /[, \t][ ]*/, $line,
@columns defined($ACCOUNT) ? 4 : 5;
splice @columns, 1, 0, $ACCOUNT // ();
eval { cond_txn_each { _inner_charge(@columns) } } // do {
if ( $@ ) {
die $@ if $abort_on_exception;
warn $@;
} }
: split /[, \t][ ]*/, $line, 5 };
; }
}
}
sub _inner_charge {
my ($date, $account, $debit, $credit, $purpose) = @_;
$account = $db->resultset("Account")->find($account) $account = $db->resultset("Account")->find($account)
// die "User not found: $account\n"; // croak "There is no account with that name: $account";
chomp $credit if !defined $purpose; chomp $credit if !defined $purpose;
_append_adj_lines_to($purpose); { # Append any consecutive lines
my ($has_more, $line);
for ( $purpose //= do { $has_more = 1; <ARGV> } ) {
(s/^"// .. s/(?<!")(")(?=\s|$)//) || $has_more or last;
last if $1;
$line = <ARGV> or last;
last if $line eq $/;
$line =~ s{^\s+}{};
$_ .= $line;
redo;
}
_strip_comment_from($purpose);
}
my @transfer;
if ( $debit =~ /^\d/ ) { if ( $debit =~ /^\d/ ) {
@ -177,21 +237,27 @@ sub charge_account {
: croak qq{Missing ID in debit purpose: $purpose} : croak qq{Missing ID in debit purpose: $purpose}
; ;
if ( $purpose =~ s{ \s* << \s* (.+?) \r? \n? \z }{}xms ) { if ( $purpose =~ s{ (?: \t | \s* << \s* ) (.+?) \r? \n? \z }{}xms ) {
push @transfers, [ push @transfer, [
map { /\D/ ? $targets{ $_ } : $_ } split /,\s*/, $1 map { /\D/ ? $targets{ $_ } : $_ } split /,\s*/, $1
] => $billId; ] => $billId;
} }
if ( my ($var, $op, $assigned) if ( my ($var, $op, $assigned)
= $tgt =~ m{^([A-Za-z]\w+)([?=])(\d+)$} = $tgt =~ m{^([A-Za-z]\w+)([?=])(\w+)$}
) { ) {
if ( defined( my $c = $targets{$var} ) ) {
croak "$var already assigned to $c";
}
if ( $op eq q{=} ) { if ( $op eq q{=} ) {
$tgt = $targets{ $var } = $assigned; $tgt = $targets{ $var } = $assigned;
} }
else { else {
$tgt = select_target_credit_from($assigned, $var); $tgt = select_target_credit_from($assigned, $var);
} }
} }
elsif ( $tgt =~ /^[a-z]/i ) { elsif ( $tgt =~ /^[a-z]/i ) {
$tgt = $targets{ $tgt } $tgt = $targets{ $tgt }
@ -224,23 +290,16 @@ sub charge_account {
croak "Invalid field input: $debit neither number nor name"; croak "Invalid field input: $debit neither number nor name";
} }
push @transfers, $c->credId => $tgt if $tgt; push @transfer, $c->credId => $tgt if $tgt;
} }
else { else {
die "Cannot be parsed: $debit $credit\n"; croak "Cannot be parsed: $debit $credit\n";
}
} }
if ( @transfers ) { $db->make_transfers(@transfer) if @transfer;
my $t = $db->make_transfers( @transfers );
if ( $t ) {
printf "Transferred %d in total.\n", $t;
}
else {
print "No transfers could be made.\n";
}
}
} }
sub _strip_comment_from { sub _strip_comment_from {
@ -252,28 +311,13 @@ sub _strip_comment_from {
sub integer_money_value { sub integer_money_value {
my $val = shift; my $val = shift;
die "Not a number: $val" if $val =~ /[^\d.]/; die "Not a positive money value (cent): $val" if $val =~ /[^\d.]/;
$val =~ s{ ( \. (\d\d) ) \z }{}xms; if ( $val =~ s{ ( \. (\d\d)? ) \z }{}xms ) {
$val *= 100; ($val *= 100) += $2 // 0;
if ( $1 ) { $val += $2 } }
return $val; return $val;
} }
sub _append_adj_lines_to {
my ($has_more, $line);
for ( $_[0] //= do { $has_more = 1; <ARGV> } ) {
if ( (s/^"// .. s/(?<!")(")(?=\s|$)//) || $has_more) {
last if $1;
$line = <ARGV>;
last if $line eq $/;
$line =~ s{^\s+}{};
$_ .= $line;
redo;
}
}
&_strip_comment_from;
}
sub select_target_credit_from { sub select_target_credit_from {
my ($account, $var) = @_; my ($account, $var) = @_;
@ -340,6 +384,8 @@ sub make_transfers {
my $last_input = q{ }; my $last_input = q{ };
cond_txn_all {
DEBIT: DEBIT:
while ( my $s = $suggestions->next ) { while ( my $s = $suggestions->next ) {
printf " - [%s] %s: %s? (%s)\b\b", $s->date, $s->billId, printf " - [%s] %s: %s? (%s)\b\b", $s->date, $s->billId,
@ -359,6 +405,7 @@ sub make_transfers {
} }
} }
}
} }
sub balance_status { sub balance_status {
@ -433,15 +480,19 @@ return 1 if caller;
package main; package main;
use Getopt::Long; use Getopt::Long;
use TrsrDB \$db;
%OPTS = ( %OPTS = (
'account' => \$ACCOUNT, 'account' => \$ACCOUNT,
'target-credit' => \%targets 'target-credit' => \%targets,
'txn-mode' => \&Commands::set_txn_mode,
); );
GetOptions( \%OPTS, GetOptions( \%OPTS,
'1', 'account|a=s', 'target|t=s', 'target-credit|tcr=i', 'source|s=s', 'txn-mode=s', 'account|a=s', 'target|t=s', 'target-credit|tcr=i', 'source|s=s',
'date|d=s', 'value|v=i', 'purpose|p=s', 'bill-id|b=s' 'date|d=s', 'value|v=i', 'purpose|p=s', 'bill-id|b=s'
); );
$DB::single=1;
delete @OPTS{'account', 'target-credit', 'txn-mode'};
my %dispatcher = ( my %dispatcher = (
charge => \&Commands::charge_account, charge => \&Commands::charge_account,
@ -466,8 +517,9 @@ my %dispatcher = (
exec qw{morbo server -w TrsrDB -w templates/}; exec qw{morbo server -w TrsrDB -w templates/};
}, },
sql => sub { sql => sub {
exec "sqlite3" => -cmd => "PRAGMA foreign_keys=ON; PRAGMA recursive_triggers=ON;" exec "sqlite3",
=> -line => $ENV{TRSRDB_SQLITE_FILE}; -cmd => "PRAGMA foreign_keys=ON; PRAGMA recursive_triggers=ON;",
'-line', $ENV{TRSRDB_SQLITE_FILE};
}, },
status => \&Commands::balance_status, status => \&Commands::balance_status,
transfer => \&Commands::make_transfers, transfer => \&Commands::make_transfers,