diff options
Diffstat (limited to 'FS/FS/Cron/bill.pm')
| -rw-r--r-- | FS/FS/Cron/bill.pm | 150 | 
1 files changed, 150 insertions, 0 deletions
| diff --git a/FS/FS/Cron/bill.pm b/FS/FS/Cron/bill.pm new file mode 100644 index 000000000..7de2ff2f6 --- /dev/null +++ b/FS/FS/Cron/bill.pm @@ -0,0 +1,150 @@ +package FS::Cron::bill; + +use strict; +use vars qw( @ISA @EXPORT_OK ); +use Exporter; +use Date::Parse; +use FS::UID qw(dbh); +use FS::Record qw(qsearchs); +use FS::cust_main; +use FS::part_event; +use FS::part_event_condition; + +@ISA = qw( Exporter ); +@EXPORT_OK = qw ( bill ); + +sub bill { + +  my %opt = @_; + +  my $check_freq = $opt{'check_freq'} || '1d'; + +  my $debug = 0; +  $debug = 1 if $opt{'v'}; +  $debug = $opt{'l'} if $opt{'l'}; +  +  $FS::cust_main::DEBUG = $debug; +  #$FS::cust_event::DEBUG = $opt{'l'} if $opt{'l'}; + +  my @search = (); + +  push @search, "cust_main.payby    = '". $opt{'p'}. "'" +    if $opt{'p'}; +  push @search, "cust_main.agentnum =  ". $opt{'a'} +    if $opt{'a'}; + +  if ( @ARGV ) { +    push @search, "( ". +      join(' OR ', map "cust_main.custnum = $_", @ARGV ). +    " )"; +  } + +  ### +  # generate where_pkg/where_event search clause +  ### + +  #we're at now now (and later). +  my($time)= $opt{'d'} ? str2time($opt{'d'}) : $^T; +  $time += $opt{'y'} * 86400 if $opt{'y'}; + +  my $invoice_time = $opt{'n'} ? $^T : $time; + +  # select * from cust_main where +  my $where_pkg = <<"END"; +    0 < ( select count(*) from cust_pkg +            where cust_main.custnum = cust_pkg.custnum +              and ( cancel is null or cancel = 0 ) +              and (    setup is null or setup =  0 +                    or bill  is null or bill  <= $time  +                    or ( expire is not null and expire <= $^T ) +                    or ( adjourn is not null and adjourn <= $^T ) +                  ) +        ) +END + +  my $where_event = join(' OR ', map { +    my $eventtable = $_; + +    my $join  = FS::part_event_condition->join_conditions_sql(  $eventtable ); +    my $where = FS::part_event_condition->where_conditions_sql( $eventtable, +                                                                'time'=>$time, +                                                              ); + +    my $are_part_event =  +      "0 < ( SELECT COUNT(*) FROM part_event $join +               WHERE check_freq = '$check_freq' +                 AND eventtable = '$eventtable' +                 AND ( disabled = '' OR disabled IS NULL ) +                 AND $where +           ) +      "; + +    if ( $eventtable eq 'cust_main' ) {  +      $are_part_event; +    } else { +      "0 < ( SELECT COUNT(*) FROM $eventtable +               WHERE cust_main.custnum = $eventtable.custnum +                 AND $are_part_event +           ) +      "; +    } + +  } FS::part_event->eventtables); + +  push @search, "( $where_pkg OR $where_event )"; + +  ### +  # get a list of custnums +  ### + +  warn "searching for customers:\n". join("\n", @search). "\n" +    if $opt{'v'} || $opt{'l'}; + +  my $sth = dbh->prepare( +    "SELECT custnum FROM cust_main". +    " WHERE ". join(' AND ', @search) +  ) or die dbh->errstr; + +  $sth->execute or die $sth->errstr; + +  my @custnums = map { $_->[0] } @{ $sth->fetchall_arrayref }; + +  ### +  # for each custnum, queue or make one customer object and bill +  # (one at a time, to reduce memory footprint with large #s of customers) +  ### +   +  foreach my $custnum ( @custnums ) { + +    if ( $opt{'m'} ) { + +      #add job to queue that calls bill_and_collect with options +        my $queue = new FS::queue { +          'job'    => 'FS::cust_main::queued_bill', +          'secure' => 'Y', +        }; +        my $error = $queue->insert( +        'custnum'      => $custnum, +        'time'         => $time, +        'invoice_time' => $invoice_time, +        'check_freq'   => $check_freq, +        'resetup'      => $opt{'s'} ? $opt{'s'} : 0, +      ); + +    } else { + +      my $cust_main = qsearchs( 'cust_main', { 'custnum' => $custnum } ); + +      $cust_main->bill_and_collect( +        'time'         => $time, +        'invoice_time' => $invoice_time, +        'check_freq'   => $check_freq, +        'resetup'      => $opt{'s'}, +        'debug'        => $debug, +      ); + +    } + +  } + +} | 
