"all" and "each" transaction mode distinction; added missing batch_processor view
This commit is contained in:
parent
df9c0b4065
commit
f54f940f74
@ -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,
|
||||||
Commands::charge_account($fh);
|
$self->param("txn-mode") // "all"
|
||||||
});
|
);
|
||||||
|
Commands::charge_account($fh);
|
||||||
$self->redirect_to('home');
|
$self->redirect_to('home');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
57
templates/account/batch_processor.html.ep
Normal file
57
templates/account/batch_processor.html.ep
Normal 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
|
320
trsr
320
trsr
@ -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 });
|
||||||
my $record = $charger->($account, $billId);
|
cond_txn_each {
|
||||||
if ( !$billId and my $t = $OPTS{target} ) {
|
my $record = $charger->($account, $billId);
|
||||||
$db->make_transfers( $record->ID => $t );
|
if ( !$billId and my $t = $OPTS{target} ) {
|
||||||
}
|
$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 });
|
||||||
while ( my $account = $accounts->next ) {
|
cond_txn_all {
|
||||||
$charger->( $account, sprintf $billId );
|
while ( my $account = $accounts->next ) {
|
||||||
}
|
cond_txn_each { $charger->( $account, sprintf $billId ); }
|
||||||
|
}
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
else {
|
else {
|
||||||
@ -134,113 +159,147 @@ 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]
|
||||||
// croak "charge_account() expects open file handle to read from"
|
and local *ARGV = (
|
||||||
if exists $_[0];
|
openhandle($_[0])
|
||||||
|
// croak "charge_account() expects open file handle to read from"
|
||||||
|
)
|
||||||
|
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;
|
|
||||||
my ($date, $account, $debit, $credit, $purpose) = $ACCOUNT
|
|
||||||
? do { my @columns = split /[, \t][ ]*/, $line, 4;
|
|
||||||
splice @columns, 1, 0, $ACCOUNT;
|
|
||||||
@columns
|
|
||||||
}
|
|
||||||
: split /[, \t][ ]*/, $line, 5
|
|
||||||
;
|
|
||||||
|
|
||||||
$account = $db->resultset("Account")->find($account)
|
_strip_comment_from($line) // next;
|
||||||
// die "User not found: $account\n";
|
|
||||||
|
|
||||||
chomp $credit if !defined $purpose;
|
my @columns
|
||||||
|
= split /[, \t][ ]*/, $line,
|
||||||
|
defined($ACCOUNT) ? 4 : 5;
|
||||||
|
|
||||||
_append_adj_lines_to($purpose);
|
splice @columns, 1, 0, $ACCOUNT // ();
|
||||||
|
|
||||||
if ( $debit =~ /^\d/ ) {
|
eval { cond_txn_each { _inner_charge(@columns) } } // do {
|
||||||
|
if ( $@ ) {
|
||||||
my $tgt = $credit;
|
die $@ if $abort_on_exception;
|
||||||
my $billId = $purpose =~ s{ \A ([[:print:]]+?) : \s* }{}xms
|
warn $@;
|
||||||
? $1
|
|
||||||
: croak qq{Missing ID in debit purpose: $purpose}
|
|
||||||
;
|
|
||||||
|
|
||||||
if ( $purpose =~ s{ \s* << \s* (.+?) \r? \n? \z }{}xms ) {
|
|
||||||
push @transfers, [
|
|
||||||
map { /\D/ ? $targets{ $_ } : $_ } split /,\s*/, $1
|
|
||||||
] => $billId;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( my ($var, $op, $assigned)
|
|
||||||
= $tgt =~ m{^([A-Za-z]\w+)([?=])(\d+)$}
|
|
||||||
) {
|
|
||||||
if ( $op eq q{=} ) {
|
|
||||||
$tgt = $targets{ $var } = $assigned;
|
|
||||||
}
|
}
|
||||||
else {
|
};
|
||||||
$tgt = select_target_credit_from($assigned, $var);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
elsif ( $tgt =~ /^[a-z]/i ) {
|
|
||||||
$tgt = $targets{ $tgt }
|
|
||||||
// croak "Target credit not assigned: $tgt!\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
$account->add_to_debits({
|
|
||||||
billId => $billId,
|
|
||||||
date => $date,
|
|
||||||
purpose => $purpose,
|
|
||||||
value => integer_money_value($debit),
|
|
||||||
targetCredit => $tgt,
|
|
||||||
});
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
elsif ( $credit =~ /^\d/ ) {
|
}
|
||||||
|
}
|
||||||
|
|
||||||
my $tgt = $purpose =~ s{ \s* >> \s* (.+) \r? \n? \z }{}xms && $1;
|
sub _inner_charge {
|
||||||
|
my ($date, $account, $debit, $credit, $purpose) = @_;
|
||||||
|
|
||||||
my $c = $account->add_to_credits({
|
$account = $db->resultset("Account")->find($account)
|
||||||
date => $date,
|
// croak "There is no account with that name: $account";
|
||||||
account => $account,
|
|
||||||
purpose => $purpose,
|
|
||||||
value => integer_money_value($credit)
|
|
||||||
});
|
|
||||||
|
|
||||||
if ( $debit =~ /^[a-z]/i ) { $targets{ $debit } = $c->credId; }
|
chomp $credit if !defined $purpose;
|
||||||
elsif ( length $debit && $debit ne q{+} ) {
|
|
||||||
croak "Invalid field input: $debit neither number nor name";
|
|
||||||
}
|
|
||||||
|
|
||||||
push @transfers, $c->credId => $tgt if $tgt;
|
|
||||||
|
|
||||||
|
{ # 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;
|
||||||
}
|
}
|
||||||
else {
|
_strip_comment_from($purpose);
|
||||||
die "Cannot be parsed: $debit $credit\n";
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( @transfers ) {
|
my @transfer;
|
||||||
my $t = $db->make_transfers( @transfers );
|
|
||||||
if ( $t ) {
|
if ( $debit =~ /^\d/ ) {
|
||||||
printf "Transferred %d in total.\n", $t;
|
|
||||||
|
my $tgt = $credit;
|
||||||
|
my $billId = $purpose =~ s{ \A ([[:print:]]+?) : \s* }{}xms
|
||||||
|
? $1
|
||||||
|
: croak qq{Missing ID in debit purpose: $purpose}
|
||||||
|
;
|
||||||
|
|
||||||
|
if ( $purpose =~ s{ (?: \t | \s* << \s* ) (.+?) \r? \n? \z }{}xms ) {
|
||||||
|
push @transfer, [
|
||||||
|
map { /\D/ ? $targets{ $_ } : $_ } split /,\s*/, $1
|
||||||
|
] => $billId;
|
||||||
}
|
}
|
||||||
else {
|
|
||||||
print "No transfers could be made.\n";
|
if ( my ($var, $op, $assigned)
|
||||||
|
= $tgt =~ m{^([A-Za-z]\w+)([?=])(\w+)$}
|
||||||
|
) {
|
||||||
|
|
||||||
|
if ( defined( my $c = $targets{$var} ) ) {
|
||||||
|
croak "$var already assigned to $c";
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( $op eq q{=} ) {
|
||||||
|
$tgt = $targets{ $var } = $assigned;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$tgt = select_target_credit_from($assigned, $var);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
elsif ( $tgt =~ /^[a-z]/i ) {
|
||||||
|
$tgt = $targets{ $tgt }
|
||||||
|
// croak "Target credit not assigned: $tgt!\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
$account->add_to_debits({
|
||||||
|
billId => $billId,
|
||||||
|
date => $date,
|
||||||
|
purpose => $purpose,
|
||||||
|
value => integer_money_value($debit),
|
||||||
|
targetCredit => $tgt,
|
||||||
|
});
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
elsif ( $credit =~ /^\d/ ) {
|
||||||
|
|
||||||
|
my $tgt = $purpose =~ s{ \s* >> \s* (.+) \r? \n? \z }{}xms && $1;
|
||||||
|
|
||||||
|
my $c = $account->add_to_credits({
|
||||||
|
date => $date,
|
||||||
|
account => $account,
|
||||||
|
purpose => $purpose,
|
||||||
|
value => integer_money_value($credit)
|
||||||
|
});
|
||||||
|
|
||||||
|
if ( $debit =~ /^[a-z]/i ) { $targets{ $debit } = $c->credId; }
|
||||||
|
elsif ( length $debit && $debit ne q{+} ) {
|
||||||
|
croak "Invalid field input: $debit neither number nor name";
|
||||||
|
}
|
||||||
|
|
||||||
|
push @transfer, $c->credId => $tgt if $tgt;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
else {
|
||||||
|
croak "Cannot be parsed: $debit $credit\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
$db->make_transfers(@transfer) if @transfer;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sub _strip_comment_from {
|
sub _strip_comment_from {
|
||||||
@ -252,26 +311,11 @@ 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;
|
|
||||||
}
|
|
||||||
|
|
||||||
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;
|
return $val;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub select_target_credit_from {
|
sub select_target_credit_from {
|
||||||
@ -340,24 +384,27 @@ sub make_transfers {
|
|||||||
|
|
||||||
my $last_input = q{ };
|
my $last_input = q{ };
|
||||||
|
|
||||||
DEBIT:
|
cond_txn_all {
|
||||||
while ( my $s = $suggestions->next ) {
|
|
||||||
printf " - [%s] %s: %s? (%s)\b\b", $s->date, $s->billId,
|
|
||||||
$s->purpose, $last_input;
|
|
||||||
chomp( my $input = <STDIN> );
|
|
||||||
$input = $last_input if !length($input);
|
|
||||||
if ( lc($input) =~ /^y(es)?$/i ) {
|
|
||||||
$transfers->create({
|
|
||||||
credId => $c->credId, billId => $s->billId
|
|
||||||
});
|
|
||||||
}
|
|
||||||
elsif ( !$input || lc($input) !~ /^no?$/i ) {
|
|
||||||
print "Input invalid!\n";
|
|
||||||
redo DEBIT;
|
|
||||||
}
|
|
||||||
$last_input = $input;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
DEBIT:
|
||||||
|
while ( my $s = $suggestions->next ) {
|
||||||
|
printf " - [%s] %s: %s? (%s)\b\b", $s->date, $s->billId,
|
||||||
|
$s->purpose, $last_input;
|
||||||
|
chomp( my $input = <STDIN> );
|
||||||
|
$input = $last_input if !length($input);
|
||||||
|
if ( lc($input) =~ /^y(es)?$/i ) {
|
||||||
|
$transfers->create({
|
||||||
|
credId => $c->credId, billId => $s->billId
|
||||||
|
});
|
||||||
|
}
|
||||||
|
elsif ( !$input || lc($input) !~ /^no?$/i ) {
|
||||||
|
print "Input invalid!\n";
|
||||||
|
redo DEBIT;
|
||||||
|
}
|
||||||
|
$last_input = $input;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -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,
|
||||||
|
Loading…
Reference in New Issue
Block a user