version 1.2, 2011/03/21 01:05:38 |
version 1.3, 2011/03/21 04:19:25 |
|
|
|
|
use DateTime; |
use DateTime; |
|
|
my %included_hours; |
|
|
|
my $config = RTI::Config->new(); |
my $config = RTI::Config->new(); |
|
|
#print Dump $config; exit; |
#print Dump $config; exit; |
|
|
# my $state = RTI::State->new($cust); |
|
# $invoice{state} = $state; |
|
my $lastinvdate; # = $state->{lastinvoicedte}; XXX Needs to be a DateTime |
|
|
|
#$lastinvdate = DateTime->now->subtract( months => 2 ); |
|
my $invoiceid = 1; # $state->{lastinvoice} + 1; |
my $invoiceid = 1; # $state->{lastinvoice} + 1; |
|
|
my $startdate; |
my $startdate; |
my @invoices; |
my @invoices; |
foreach my $cust ( @{ $config->get('customers') } ) { |
foreach my $cust ( @{ $config->get('customers') } ) { |
|
|
|
# my $state = RTI::State->new($cust); |
|
# $invoice{state} = $state; |
|
my $lastinvdate; # = $state->{lastinvoicedte}; XXX Needs to be a DateTime |
|
#$lastinvdate = DateTime->now->subtract( months => 2 ) |
|
# ->set( hour => 0, minute => 0, second => 0 ); |
|
|
my %invoice = ( |
my %invoice = ( |
from => $config->get('from'), |
from => $config->get('from'), |
|
info => $config->get('info'), |
to => $cust->{address}, |
to => $cust->{address}, |
rates => $cust->{rates}, |
rates => $cust->{rates}, |
match => $cust->{match}, |
match => $cust->{match}, |
); |
); |
|
|
if ( $cust->{base_rate} ) { |
if ( $cust->{base_rate} ) { |
my $date = DateTime->now; |
my $day = $cust->{day} || 1; |
my $day = $cust->{day} || 1; |
|
my $freq = $cust->{frequency} || 1; |
my $freq = $cust->{frequency} || 1; |
|
|
my $diff; |
my $day_method; |
my $per; |
my $per; |
given ( $cust->{per} ) { |
given ( $cust->{per} ) { |
when ('week') { $per = 'weeks'; $diff = $date->dow - $day; } |
when ('week') { $per = 'weeks'; $day_method = 'dow' } |
when ('month') { $per = 'months'; $diff = $date->day - $day; } |
when ('month') { $per = 'months'; $day_method = 'day' } |
default { die "Unknown per [$cust->{per}]\n" } |
default { die "Unknown per [$cust->{per}]\n" } |
} |
} |
|
|
# $day is start day, end should be one day further back |
my $lastbill |
$diff = abs($diff) + 1; |
= DateTime->now->set( hour => 0, minute => 0, second => 0 ); |
$date->subtract( days => $diff ); |
while ( $lastbill->$day_method != $day ) { |
|
$lastbill->subtract( days => 1 ); |
|
} |
|
|
|
my $date |
|
= $lastinvdate |
|
? $lastinvdate->clone->add( days => 1 ) |
|
: $lastbill->clone->subtract( $per => $freq ); |
|
|
my $title |
my $title |
= $freq == 1 |
= $freq == 1 |
? ucfirst( $cust->{per} . 'ly' ) |
? ucfirst( $cust->{per} . 'ly' ) |
|
|
|
|
my %project = ( title => $title, fees => [], ); |
my %project = ( title => $title, fees => [], ); |
|
|
# XXX need to add them until we get to where we billed already |
while ( $date < $lastbill ) { |
# if we don't know the last invoice date, assume the day before |
my $start = $date->clone; |
$lastinvdate ||= $date->clone->subtract( days => 1 ); |
|
while ( $date > $lastinvdate ) { |
$date->add( $per => $freq ); |
$invoice{enddate} = $date->clone; |
$date = $lastbill->clone if $date > $lastbill; |
|
if ( my $diff = $date->$day_method - $day ) { |
|
$date->subtract( days => $diff ); |
|
} |
|
|
|
my $end = $date->clone; |
|
|
|
$end->subtract( seconds => 1 ); |
|
|
|
$startdate = $start->clone if $startdate > $start; |
|
$invoice{start} ||= $start->clone; |
|
$invoice{end} = $end->clone; |
my %hours = ( |
my %hours = ( |
end => $date->clone, |
start => $start->clone, |
|
end => $end->clone, |
hours => { %{ $cust->{hours} } }, |
hours => { %{ $cust->{hours} } }, |
); |
); |
|
|
my $contents = ' to ' . $date->ymd; |
push @{ $invoice{hours} }, \%hours; |
$date->subtract( $per => $freq )->add( days => 1 ); |
push @{ $project{fees} }, |
$contents = $date->ymd . $contents; |
|
|
|
$invoice{startdate} ||= $date->clone; |
|
$hours{start} = $date->clone; |
|
$startdate = $date->clone; |
|
|
|
unshift @{ $invoice{hours} }, \%hours; |
|
unshift @{ $project{fees} }, |
|
{ |
{ |
count => 1, |
count => 1, |
rate => $cust->{base_rate}, |
rate => $cust->{base_rate}, |
contents => $contents, |
contents => $start->ymd . ' to ' . $end->ymd, |
}; |
}; |
|
|
# Next time, one day less |
|
$date->subtract( days => 1 ); |
|
} |
} |
|
|
if (@{ $project{fees} }) { |
if ( @{ $project{fees} } ) { |
push @{ $invoice{projects} }, \%project; |
push @{ $invoice{projects} }, \%project; |
} |
} |
} |
} |
else { |
else { |
$invoice{enddate} = DateTime->now->ymd; |
$invoice{end} = DateTime->now; |
push @{ $invoice{hours} }, |
push @{ $invoice{hours} }, |
{ end => DateTime->now, hours => $cust->{hours} }; |
{ end => DateTime->now, hours => $cust->{hours} }; |
} |
} |
|
|
|
next unless $invoice{end}; |
push @invoices, \%invoice; |
push @invoices, \%invoice; |
} |
} |
|
|
|
|
id => $ticket->id, |
id => $ticket->id, |
queue => $ticket->queue, |
queue => $ticket->queue, |
owner => $ticket->owner, |
owner => $ticket->owner, |
title => $ticket->id . ': ' . $ticket->subject, |
title => $ticket->subject, |
detail => 'Requestors: ' |
detail => 'Ticket: ' |
. join( ', ', $ticket->requestors ) |
. $ticket->id |
. ' Queue: ' |
. ' Queue: ' |
. $ticket->queue, |
. $ticket->queue |
|
. ' Requestors: ' |
|
. join( ', ', $ticket->requestors ), |
fees => [], |
fees => [], |
expenses => [], |
expenses => [], |
); |
); |
|
|
while ( my $txn = $txn_i->() ) { |
while ( my $txn = $txn_i->() ) { |
next unless $txn->time_taken; |
next unless $txn->time_taken; |
|
|
|
my ( $date, $time ) = split ' ', $txn->created; |
|
my ( $year, $month, $day ) = split '-', $date; |
|
my ( $hour, $minute, $second ) = split ':', $time; |
|
|
|
my $txn_date = DateTime->new( |
|
year => $year, |
|
month => $month, |
|
day => $day, |
|
hour => $hour, |
|
minute => $minute, |
|
second => $second, |
|
); |
|
|
|
next if $invoice->{start} && $invoice->{start} > $txn_date; |
|
next if $invoice->{end} < $txn_date; |
|
|
|
my $hours = {}; |
|
if ( $invoice->{hours} ) { |
|
foreach my $h ( @{ $invoice->{hours} } ) { |
|
next if $h->{start} && $h->{start} > $txn_date; |
|
next if $h->{end} < $txn_date; |
|
|
|
$hours = $h->{hours}; |
|
last; |
|
} |
|
} |
|
|
my $work_time = sprintf "%.03f", $txn->time_taken / 60; |
my $work_time = sprintf "%.03f", $txn->time_taken / 60; |
my $work_type = $txn->cf('WorkType'); |
my $work_type = $txn->cf('WorkType'); |
my $work_rate |
my $work_rate |
|
|
|| $invoice->{rates}{default} |
|| $invoice->{rates}{default} |
|| 0; |
|| 0; |
|
|
my $ih_type |
my $h_type |
= exists $included_hours{$work_type} |
= exists $hours->{$work_type} |
? $work_type |
? $work_type |
: 'default'; |
: 'default'; |
|
|
|
|
|
|
push @{ $project{fees} }, \%fee; |
push @{ $project{fees} }, \%fee; |
|
|
next |
next unless $hours->{$h_type} && $hours->{$h_type} > 0; |
unless $included_hours{$ih_type} && $included_hours{$ih_type} > 0; |
|
|
|
my $discount_time = 0; |
my $discount_time = 0; |
if ( $included_hours{$ih_type} > $work_time ) { |
if ( $hours->{$h_type} > $work_time ) { |
$included_hours{$ih_type} -= $work_time; |
$hours->{$h_type} -= $work_time; |
$discount_time = $work_time; |
$discount_time = $work_time; |
} |
} |
else { |
else { |
$discount_time = $included_hours{$ih_type}; |
$discount_time = $hours->{$h_type}; |
$included_hours{$ih_type} = 0; |
$hours->{$h_type} = 0; |
} |
} |
|
|
if ($discount_time) { |
if ($discount_time) { |
|
|
|
|
if ( $invoice->{past_due} ) { |
if ( $invoice->{past_due} ) { |
$invoice->{total_due} = $invoice->{total} + $invoice->{past_due}; |
$invoice->{total_due} = $invoice->{total} + $invoice->{past_due}; |
|
} |
|
|
|
foreach my $key (qw/ start end /) { |
|
if ( exists $invoice->{$key} ) { |
|
$invoice->{$key} = $invoice->{$key}->strftime('%B %d, %Y'); |
|
} |
} |
} |
|
|
$invoice->{id} = $invoiceid; |
$invoice->{id} = $invoiceid; |