summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorivan <ivan>2011-08-11 21:37:53 +0000
committerivan <ivan>2011-08-11 21:37:53 +0000
commit41f298b3615f21d2441b4503bcd8146ee5aa21e2 (patch)
tree9fd5ae86fbad25cc414df121aa4ae755a19961d8
parenta78d8300b4be8ff4fbb4b5c64139d4c442b16c2f (diff)
add a bulk_simple price plan that behaves more intuitively for intial invoices, RT#14015
-rw-r--r--FS/FS/part_pkg/bulk.pm150
-rw-r--r--FS/FS/part_pkg/bulk_Common.pm119
-rw-r--r--FS/FS/part_pkg/bulk_simple.pm45
3 files changed, 198 insertions, 116 deletions
diff --git a/FS/FS/part_pkg/bulk.pm b/FS/FS/part_pkg/bulk.pm
index 4ac73892c..fd96f8bc2 100644
--- a/FS/FS/part_pkg/bulk.pm
+++ b/FS/FS/part_pkg/bulk.pm
@@ -1,146 +1,64 @@
package FS::part_pkg::bulk;
+use base qw( FS::part_pkg::bulk_Common );
use strict;
-use vars qw(@ISA $DEBUG $me %info);
+use vars qw($DEBUG $me %info);
use Date::Format;
-use FS::part_pkg::flat;
+use List::Util qw( max );
use FS::Conf;
-@ISA = qw(FS::part_pkg::flat);
-
$DEBUG = 0;
$me = '[FS::part_pkg::bulk]';
%info = (
- 'name' => 'Bulk billing based on number of active services',
- 'inherit_fields' => [ 'global_Mixin' ],
+ 'name' => 'Bulk billing based on number of active services (during billing period)',
+ 'inherit_fields' => [ 'bulk_Common', 'global_Mixin' ],
'fields' => {
- 'svc_setup_fee' => { 'name' => 'Setup fee for each new service',
- 'default' => 0,
- },
- 'svc_recur_fee' => { 'name' => 'Recurring fee for each service',
- 'default' => 0,
- },
- 'summarize_svcs'=> { 'name' => 'Show a count of services on the invoice, '.
- 'instead of a detailed list',
- 'type' => 'checkbox',
- },
'no_prorate' => { 'name' => 'Don\'t prorate recurring fees on services '.
'active for a partial month',
'type' => 'checkbox',
},
},
- 'fieldorder' => [ 'svc_setup_fee', 'svc_recur_fee',
- 'summarize_svcs', 'no_prorate' ],
- 'weight' => 50,
+ 'fieldorder' => [ 'no_prorate' ],
+ 'weight' => 51,
);
-sub price_info {
- my $self = shift;
- my $str = $self->SUPER::price_info;
- my $svc_setup_fee = $self->option('svc_setup_fee');
- my $svc_recur_fee = $self->option('svc_recur_fee');
- my $conf = new FS::Conf;
- my $money_char = $conf->config('money_char') || '$';
- $str .= " , bulk" if $str;
- $str .= ": $money_char" . $svc_setup_fee . " one-time per service"
- if $svc_setup_fee;
- $str .= ", " if ($svc_setup_fee && $svc_recur_fee);
- $str .= $money_char . $svc_recur_fee . " recurring per service"
- if $svc_recur_fee;
- $str;
+
+sub _bulk_cust_svc {
+ my( $self, $cust_pkg, $sdate ) = @_;
+ # END START
+ $cust_pkg->h_cust_svc( $$sdate, $cust_pkg->last_bill );
}
-#some false laziness-ish w/agent.pm... not a lot
-sub calc_recur {
- my($self, $cust_pkg, $sdate, $details ) = @_;
+sub _bulk_setup {
+ my( $self, $cust_pkg, $h_cust_svc ) = @_;
+ ($h_cust_svc->date_inserted < $cust_pkg->last_bill)
+ ? $self->option('svc_setup_fee')
+ : 0;
+}
+
+sub _bulk_recur {
+ my( $self, $cust_pkg, $h_cust_svc, $sdate ) = @_;
- my $conf = new FS::Conf;
- my $money_char = $conf->config('money_char') || '$';
-
- my $svc_setup_fee = $self->option('svc_setup_fee');
+ return ($self->option('svc_recur_fee'), '')
+ if $self->option('no_prorate',1);
my $last_bill = $cust_pkg->last_bill;
+ my $svc_start = max( $h_cust_svc->date_inserted, $last_bill);
+ my $svc_end = $h_cust_svc->date_deleted;
+ $svc_end = ( !$svc_end || $svc_end > $$sdate ) ? $$sdate : $svc_end;
- return sprintf("%.2f", $self->base_recur($cust_pkg, $sdate) )
- unless $$sdate > $last_bill;
-
- my $total_svc_charge = 0;
- my %n_setup = ();
- my %n_recur = ();
- my %part_svc_label = ();
-
- my $summarize = $self->option('summarize_svcs',1);
-
- warn "$me billing for bulk services from ". time2str('%x', $last_bill).
- " to ". time2str('%x', $$sdate). "\n"
- if $DEBUG;
-
- # END START
- foreach my $h_cust_svc ( $cust_pkg->h_cust_svc( $$sdate, $last_bill ) ) {
-
- my @label = $h_cust_svc->label_long( $$sdate, $last_bill );
- die "fatal: no historical label found, wtf?" unless scalar(@label); #?
- my $svc_details = $label[0]. ': '. $label[1]. ': ';
- $part_svc_label{$h_cust_svc->svcpart} ||= $label[0];
-
- my $svc_charge = 0;
-
- my $svc_start = $h_cust_svc->date_inserted;
- if ( $svc_start < $last_bill ) {
- $svc_start = $last_bill;
- } elsif ( $svc_setup_fee ) {
- $svc_charge += $svc_setup_fee;
- $svc_details .= $money_char. sprintf('%.2f setup, ', $svc_setup_fee);
- $n_setup{$h_cust_svc->svcpart}++;
- }
-
- my $svc_end = $h_cust_svc->date_deleted;
- $svc_end = ( !$svc_end || $svc_end > $$sdate ) ? $$sdate : $svc_end;
-
- my $recur_charge;
- if ( $self->option('no_prorate',1) ) {
- $recur_charge = $self->option('svc_recur_fee');
- }
- else {
- $recur_charge = $self->option('svc_recur_fee')
- * ( $svc_end - $svc_start )
- / ( $$sdate - $last_bill );
- }
-
- $svc_details .= $money_char. sprintf('%.2f', $recur_charge ).
- ' ('. time2str('%x', $svc_start).
- ' - '. time2str('%x', $svc_end ). ')'
- if $recur_charge;
-
- $svc_charge += $recur_charge;
- $n_recur{$h_cust_svc->svcpart}++;
- push @$details, $svc_details if !$summarize;
- $total_svc_charge += $svc_charge;
-
- }
- if ( $summarize ) {
- foreach my $svcpart (keys %part_svc_label) {
- push @$details, sprintf('Setup fee: %d @ '.$money_char.'%.2f',
- $n_setup{$svcpart}, $svc_setup_fee )
- if $svc_setup_fee and $n_setup{$svcpart};
- push @$details, sprintf('%d services @ '.$money_char.'%.2f',
- $n_recur{$svcpart}, $self->option('svc_recur_fee') )
- if $n_recur{$svcpart};
- }
- }
-
- sprintf('%.2f', $self->base_recur($cust_pkg, $sdate) + $total_svc_charge );
-}
+ my $recur_charge = $self->option('svc_recur_fee')
+ * ( $svc_end - $svc_start )
+ / ( $$sdate - $last_bill );
-sub can_discount { 0; }
+ return (0, '') unless $recur_charge;
-sub hide_svc_detail {
- 1;
-}
+ my $svc_details .= ' ('. time2str('%x', $svc_start).
+ ' - '. time2str('%x', $svc_end ). ')';
+
+ ( $recur_charge, $svc_details );
-sub is_free_options {
- qw( setup_fee recur_fee svc_setup_fee svc_recur_fee );
}
1;
diff --git a/FS/FS/part_pkg/bulk_Common.pm b/FS/FS/part_pkg/bulk_Common.pm
new file mode 100644
index 000000000..f8f0ab881
--- /dev/null
+++ b/FS/FS/part_pkg/bulk_Common.pm
@@ -0,0 +1,119 @@
+package FS::part_pkg::bulk_Common;
+use base qw( FS::part_pkg::flat );
+
+use strict;
+use vars qw($DEBUG $me %info);
+use Date::Format;
+use FS::Conf;
+
+$DEBUG = 0;
+$me = '[FS::part_pkg::bulk_Common]';
+
+%info = (
+ 'disabled' => 1,
+ 'inherit_fields' => [ 'global_Mixin' ],
+ 'fields' => {
+ 'svc_setup_fee' => { 'name' => 'Setup fee for each new service',
+ 'default' => 0,
+ },
+ 'svc_recur_fee' => { 'name' => 'Recurring fee for each service',
+ 'default' => 0,
+ },
+ 'summarize_svcs'=> { 'name' => 'Show a count of services on the invoice, '.
+ 'instead of a detailed list',
+ 'type' => 'checkbox',
+ },
+ 'no_prorate' => { 'name' => 'Don\'t prorate recurring fees on services '.
+ 'active for a partial month',
+ 'type' => 'checkbox',
+ },
+ },
+ 'fieldorder' => [ 'svc_setup_fee', 'svc_recur_fee',
+ 'summarize_svcs', 'no_prorate' ],
+ 'weight' => 51,
+);
+
+sub price_info {
+ my $self = shift;
+ my $str = $self->SUPER::price_info;
+ my $svc_setup_fee = $self->option('svc_setup_fee');
+ my $svc_recur_fee = $self->option('svc_recur_fee');
+ my $conf = new FS::Conf;
+ my $money_char = $conf->config('money_char') || '$';
+ $str .= " , bulk" if $str;
+ $str .= ": $money_char" . $svc_setup_fee . " one-time per service"
+ if $svc_setup_fee;
+ $str .= ", " if ($svc_setup_fee && $svc_recur_fee);
+ $str .= $money_char . $svc_recur_fee . " recurring per service"
+ if $svc_recur_fee;
+ $str;
+}
+
+#some false laziness-ish w/agent.pm... not a lot
+sub calc_recur {
+ my($self, $cust_pkg, $sdate, $details ) = @_;
+
+ my $conf = new FS::Conf;
+ my $money_char = $conf->config('money_char') || '$';
+
+ my $last_bill = $cust_pkg->last_bill;
+
+ my $total_svc_charge = 0;
+ my %n_setup = ();
+ my %n_recur = ();
+ my %part_svc_label = ();
+
+ my $summarize = $self->option('summarize_svcs',1);
+
+ foreach my $cust_svc ( $self->_bulk_cust_svc( $cust_pkg, $sdate ) ) {
+
+ my @label = $cust_svc->label_long( $$sdate, $last_bill );
+ die "fatal: no label found, wtf?" unless scalar(@label); #?
+ my $svc_details = $label[0]. ': '. $label[1]. ': ';
+ $part_svc_label{$cust_svc->svcpart} ||= $label[0];
+
+ my $svc_charge = 0;
+
+ my $setup = $self->_bulk_setup($cust_pkg, $cust_svc);
+ if ( $setup ) {
+ $svc_charge += $setup;
+ $svc_details .= $money_char. sprintf('%.2f setup, ', $setup);
+ $n_setup{$cust_svc->svcpart}++;
+ }
+
+ my( $recur, $r_details ) = $self->_bulk_recur($cust_pkg, $cust_svc, $sdate);
+ if ( $recur ) {
+ $svc_charge += $recur;
+ $svc_details .= $money_char. sprintf('%.2f', $recur). $r_details;
+ $n_recur{$cust_svc->svcpart}++;
+ push @$details, $svc_details if !$summarize;
+ }
+
+ $total_svc_charge += $svc_charge;
+
+ }
+
+ if ( $summarize ) {
+ foreach my $svcpart (keys %part_svc_label) {
+ push @$details, sprintf('Setup fee: %d @ '.$money_char.'%.2f',
+ $n_setup{$svcpart}, $self->option('svc_setup_fee') )
+ if $self->option('svc_setup_fee') and $n_setup{$svcpart};
+ push @$details, sprintf('%d services @ '.$money_char.'%.2f',
+ $n_recur{$svcpart}, $self->option('svc_recur_fee') )
+ if $n_recur{$svcpart};
+ }
+ }
+
+ sprintf('%.2f', $self->base_recur($cust_pkg, $sdate) + $total_svc_charge );
+}
+
+sub can_discount { 0; }
+
+sub hide_svc_detail { 1; }
+
+sub is_free_options {
+ qw( setup_fee recur_fee svc_setup_fee svc_recur_fee );
+}
+
+1;
+
diff --git a/FS/FS/part_pkg/bulk_simple.pm b/FS/FS/part_pkg/bulk_simple.pm
new file mode 100644
index 000000000..93944cc0a
--- /dev/null
+++ b/FS/FS/part_pkg/bulk_simple.pm
@@ -0,0 +1,45 @@
+package FS::part_pkg::bulk_simple;
+use base qw( FS::part_pkg::bulk_Common );
+
+use strict;
+use vars qw($DEBUG $me %info);
+use Date::Format;
+use FS::Conf;
+use FS::cust_svc_option;
+
+$DEBUG = 0;
+$me = '[FS::part_pkg::bulk]';
+
+%info = (
+ 'name' => 'Bulk billing based on number of active services (at invoice generation)',
+ 'inherit_fields' => [ 'bulk_Common', 'global_Mixin' ],
+ 'weight' => 50,
+);
+
+sub _bulk_cust_svc {
+ my( $self, $cust_pkg, $sdate ) = @_;
+ $cust_pkg->cust_svc;
+}
+
+sub _bulk_setup {
+ my( $self, $cust_pkg, $cust_svc ) = @_;
+ return 0 if $cust_svc->option('bulk_setup');
+
+ my $bulk_setup = new FS::cust_svc_option {
+ 'svcnum' => $cust_svc->svcnum,
+ 'optionname' => 'bulk_setup',
+ 'optionvalue' => time, #invoice date?
+ };
+ my $error = $bulk_setup->insert;
+ die $error if $error;
+
+ $self->option('svc_setup_fee');
+}
+
+sub _bulk_recur {
+ my( $self, $cust_pkg, $cust_svc ) = @_;
+ ( $self->option('svc_recur_fee'), '' );
+}
+
+1;
+