"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;
|
||||
require "$Bin/trsr" or die;
|
||||
$self->app->db->storage->txn_do(sub {
|
||||
Commands::act_on_other_db($self->app->db);
|
||||
Commands->import(
|
||||
$self->app->db,
|
||||
$self->param("txn-mode") // "all"
|
||||
);
|
||||
Commands::charge_account($fh);
|
||||
});
|
||||
$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
|
174
trsr
174
trsr
@ -1,22 +1,43 @@
|
||||
#!/usr/bin/env perl
|
||||
use strict;
|
||||
|
||||
my (%OPTS, $ACCOUNT, %targets);
|
||||
my ($db, %OPTS, $ACCOUNT, %targets);
|
||||
|
||||
package Commands {
|
||||
use Text::Wrap qw(wrap);
|
||||
use Carp qw(croak);
|
||||
use Scalar::Util qw(openhandle);
|
||||
|
||||
my $db;
|
||||
use TrsrDB \$db;
|
||||
use open qw/:std :encoding(utf8)/;
|
||||
|
||||
sub act_on_other_db {
|
||||
$db = shift;
|
||||
sub import {
|
||||
$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");
|
||||
|
||||
_require_value();
|
||||
@ -40,17 +61,21 @@ sub single_charge {
|
||||
|
||||
if ( $ACCOUNT ) {
|
||||
my $account = $accounts->find({ ID => $ACCOUNT });
|
||||
cond_txn_each {
|
||||
my $record = $charger->($account, $billId);
|
||||
if ( !$billId and my $t = $OPTS{target} ) {
|
||||
$db->make_transfers( $record->ID => $t );
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
elsif ( $billId =~ s{ @(\w+) }{ %s }xms ) {
|
||||
$accounts = $accounts->search({ type => $1 });
|
||||
cond_txn_all {
|
||||
while ( my $account = $accounts->next ) {
|
||||
$charger->( $account, sprintf $billId );
|
||||
cond_txn_each { $charger->( $account, sprintf $billId ); }
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
else {
|
||||
@ -134,40 +159,75 @@ sub _require_target {
|
||||
}
|
||||
|
||||
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"
|
||||
if exists $_[0];
|
||||
)
|
||||
and my $abort_on_exception = 1
|
||||
;
|
||||
|
||||
# Possible line formats:
|
||||
# -------------------------------------------------------
|
||||
#
|
||||
# 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
|
||||
# ^^^ Field may be empty, yet not stripped
|
||||
# Please note: Omit Account if specified on commandline.
|
||||
# ^^^ Field may be empty, yet not omitted
|
||||
#
|
||||
# Please note: Omit Account if it is specified on commandline.
|
||||
#
|
||||
|
||||
my @transfers;
|
||||
cond_txn_all {
|
||||
|
||||
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
|
||||
|
||||
my @columns
|
||||
= split /[, \t][ ]*/, $line,
|
||||
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)
|
||||
// die "User not found: $account\n";
|
||||
// croak "There is no account with that name: $account";
|
||||
|
||||
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/ ) {
|
||||
|
||||
@ -177,21 +237,27 @@ sub charge_account {
|
||||
: croak qq{Missing ID in debit purpose: $purpose}
|
||||
;
|
||||
|
||||
if ( $purpose =~ s{ \s* << \s* (.+?) \r? \n? \z }{}xms ) {
|
||||
push @transfers, [
|
||||
if ( $purpose =~ s{ (?: \t | \s* << \s* ) (.+?) \r? \n? \z }{}xms ) {
|
||||
push @transfer, [
|
||||
map { /\D/ ? $targets{ $_ } : $_ } split /,\s*/, $1
|
||||
] => $billId;
|
||||
}
|
||||
|
||||
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{=} ) {
|
||||
$tgt = $targets{ $var } = $assigned;
|
||||
}
|
||||
else {
|
||||
$tgt = select_target_credit_from($assigned, $var);
|
||||
}
|
||||
|
||||
}
|
||||
elsif ( $tgt =~ /^[a-z]/i ) {
|
||||
$tgt = $targets{ $tgt }
|
||||
@ -224,23 +290,16 @@ sub charge_account {
|
||||
croak "Invalid field input: $debit neither number nor name";
|
||||
}
|
||||
|
||||
push @transfers, $c->credId => $tgt if $tgt;
|
||||
push @transfer, $c->credId => $tgt if $tgt;
|
||||
|
||||
}
|
||||
|
||||
else {
|
||||
die "Cannot be parsed: $debit $credit\n";
|
||||
}
|
||||
croak "Cannot be parsed: $debit $credit\n";
|
||||
}
|
||||
|
||||
if ( @transfers ) {
|
||||
my $t = $db->make_transfers( @transfers );
|
||||
if ( $t ) {
|
||||
printf "Transferred %d in total.\n", $t;
|
||||
}
|
||||
else {
|
||||
print "No transfers could be made.\n";
|
||||
}
|
||||
}
|
||||
$db->make_transfers(@transfer) if @transfer;
|
||||
|
||||
}
|
||||
|
||||
sub _strip_comment_from {
|
||||
@ -252,28 +311,13 @@ sub _strip_comment_from {
|
||||
|
||||
sub integer_money_value {
|
||||
my $val = shift;
|
||||
die "Not a number: $val" if $val =~ /[^\d.]/;
|
||||
$val =~ s{ ( \. (\d\d) ) \z }{}xms;
|
||||
$val *= 100;
|
||||
if ( $1 ) { $val += $2 }
|
||||
die "Not a positive money value (cent): $val" if $val =~ /[^\d.]/;
|
||||
if ( $val =~ s{ ( \. (\d\d)? ) \z }{}xms ) {
|
||||
($val *= 100) += $2 // 0;
|
||||
}
|
||||
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 {
|
||||
my ($account, $var) = @_;
|
||||
|
||||
@ -340,6 +384,8 @@ sub make_transfers {
|
||||
|
||||
my $last_input = q{ };
|
||||
|
||||
cond_txn_all {
|
||||
|
||||
DEBIT:
|
||||
while ( my $s = $suggestions->next ) {
|
||||
printf " - [%s] %s: %s? (%s)\b\b", $s->date, $s->billId,
|
||||
@ -360,6 +406,7 @@ sub make_transfers {
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
sub balance_status {
|
||||
my ($with_report) = @_;
|
||||
@ -433,15 +480,19 @@ return 1 if caller;
|
||||
package main;
|
||||
|
||||
use Getopt::Long;
|
||||
use TrsrDB \$db;
|
||||
|
||||
%OPTS = (
|
||||
'account' => \$ACCOUNT,
|
||||
'target-credit' => \%targets
|
||||
'target-credit' => \%targets,
|
||||
'txn-mode' => \&Commands::set_txn_mode,
|
||||
);
|
||||
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'
|
||||
);
|
||||
$DB::single=1;
|
||||
delete @OPTS{'account', 'target-credit', 'txn-mode'};
|
||||
|
||||
my %dispatcher = (
|
||||
charge => \&Commands::charge_account,
|
||||
@ -466,8 +517,9 @@ my %dispatcher = (
|
||||
exec qw{morbo server -w TrsrDB -w templates/};
|
||||
},
|
||||
sql => sub {
|
||||
exec "sqlite3" => -cmd => "PRAGMA foreign_keys=ON; PRAGMA recursive_triggers=ON;"
|
||||
=> -line => $ENV{TRSRDB_SQLITE_FILE};
|
||||
exec "sqlite3",
|
||||
-cmd => "PRAGMA foreign_keys=ON; PRAGMA recursive_triggers=ON;",
|
||||
'-line', $ENV{TRSRDB_SQLITE_FILE};
|
||||
},
|
||||
status => \&Commands::balance_status,
|
||||
transfer => \&Commands::make_transfers,
|
||||
|
Loading…
Reference in New Issue
Block a user