X-Git-Url: http://git.freeside.biz/gitweb/?a=blobdiff_plain;f=FS%2FFS%2FMaestro.pm;h=8c33e3bf95631712568da4431ecb162525f5cc89;hb=b5c4237a34aef94976bc343c8d9e138664fc3984;hp=c1d0470324001a87fe9eaf8d7ac18b5297894f43;hpb=592c1771d2b332127113e30094bf8fd6b026e046;p=freeside.git diff --git a/FS/FS/Maestro.pm b/FS/FS/Maestro.pm index c1d047032..8c33e3bf9 100644 --- a/FS/FS/Maestro.pm +++ b/FS/FS/Maestro.pm @@ -1,10 +1,15 @@ package FS::Maestro; +use strict; use Date::Format; use FS::Conf; use FS::Record qw( qsearchs ); use FS::cust_main; +use FS::cust_pkg; +use FS::part_svc; +#i guess this is kind of deprecated in favor of service_status, but keeping it +#around until they say they don't need it. sub customer_status { my( $custnum ) = shift; #@_; my $svcnum = @_ ? shift : ''; @@ -19,122 +24,224 @@ sub customer_status { or return { 'status' => 'E', 'error' => "custnum $custnum not found" }; - my( $svc_pbx, $good_till, $outbound_service ) = ( '', '', '' ); + return service_status($svcnum) if $svcnum; + + ### + # regular customer to maestro (single package) + ### + my %result = (); - if ( $svcnum ) { - - ### - # reseller scenario to maestro (customer w/ multiple packages) - ### - # find $svc_pbx + my @cust_pkg = $cust_main->cust_pkg; - $svc_pbx = qsearchs({ - 'table' => 'svc_pbx', - 'addl_from' => ' LEFT JOIN cust_svc USING ( svcnum ) '. - ' LEFT JOIN cust_pkg USING ( pkgnum ) ', - 'hashref' => { 'svcnum' => $svcnum }, - 'extra_sql' => " AND custnum = $custnum", - }) - or return { 'status' => 'E', - 'error' => "svcnum $svcnum not found" }; + #things specific to the non-reseller scenario - #status in the reseller scenario + $result{'status'} = substr($cust_main->ucfirst_status,0,1); - my $cust_pkg = $svc_pbx->cust_svc->cust_pkg; + $result{'products'} = + [ map $_->pkgpart, grep !$_->get('cancel'), @cust_pkg ]; - $result{'status'} = substr($cust_pkg->ucfirst_status,0,1); + #find svc_pbx - # find "outbound service" y/n + my @cust_svc = map $_->cust_svc, @cust_pkg; - #XXX outbound service per-reseller ? - #my @cust_pkg = $cust_main->cust_pkg; - # - #my $conf = new FS::Conf; - #my %outbound_pkgs = map { $_=>1 } $conf->config('mc-outbound_packages'); - #my $outbound_service = - # scalar( grep { $outbound_pkgs{ $_->pkgpart } - # && !$_->get('cancel') - # } - # @cust_pkg - # ) - # ? 1 : 0; + my @cust_svc_pbx = + grep { my($n,$l,$t) = $_->label; $t eq 'svc_pbx' } + @cust_svc; - # find "good till" date/time stamp (this package) + if ( ! @cust_svc_pbx ) { + return { 'status' => 'E', + 'error' => "customer $custnum has no conference service" }; + } elsif ( scalar(@cust_svc_pbx) > 1 ) { + return { 'status' => 'E', + 'error' => + "customer $custnum has more than one conference". + " service (reseller?); specify a svcnum as a second argument", + }; + } - $good_till = time2str('%c', $cust_pkg->bill || time ); + my $cust_svc_pbx = $cust_svc_pbx[0]; - } else { + my $svc_pbx = $cust_svc_pbx->svc_x; - ### - # regular customer to maestro (single package) - ### + # find "outbound service" y/n - my @cust_pkg = $cust_main->cust_pkg; + my $conf = new FS::Conf; + my %outbound_pkgs = map { $_=>1 } $conf->config('mc-outbound_packages'); + $result{'outbound_service'} = + scalar( grep { $outbound_pkgs{ $_->pkgpart } + && !$_->get('cancel') + } + @cust_pkg + ) + ? 1 : 0; - #things specific to the non-reseller scenario + # find "good till" date/time stamp - $result{'status'} = substr($cust_main->ucfirst_status,0,1); + my @active_cust_pkg = + sort { $a->bill <=> $b->bill } + grep { !$_->get('cancel') && $_->part_pkg->freq ne '0' } + @cust_pkg; + $result{'good_till'} = time2str('%c', $active_cust_pkg[0]->bill || time ); - $result{'products'} = - [ map $_->pkgpart, grep !$_->get('cancel'), @cust_pkg ]; + return { + 'name' => $cust_main->name, + 'email' => $cust_main->invoicing_list_emailonly_scalar, + #'agentnum' => $cust_main->agentnum, + #'agent' => $cust_main->agent->agent, + 'max_lines' => $svc_pbx ? $svc_pbx->max_extensions : '', + 'max_simultaneous' => $svc_pbx ? $svc_pbx->max_simultaneous : '', + %result, + }; - #find svc_pbx +} - my @cust_svc = map $_->cust_svc, @cust_pkg; +sub service_status { + my $svcnum = shift; - my @cust_svc_pbx = - grep { my($n,$l,$t) = $_->label; $t eq 'svc_pbx' } - @cust_svc; + my $svc_pbx = qsearchs({ + 'table' => 'svc_pbx', + 'addl_from' => ' LEFT JOIN cust_svc USING ( svcnum ) '. + ' LEFT JOIN cust_pkg USING ( pkgnum ) ', + 'hashref' => { 'svcnum' => $svcnum }, + #'extra_sql' => " AND custnum = $custnum", + }) + or return { 'status' => 'E', + 'error' => "svcnum $svcnum not found" }; - if ( ! @cust_svc_pbx ) { - return { 'status' => 'E', - 'error' => "customer $custnum has no conference service" }; - } elsif ( scalar(@cust_svc_pbx) > 1 ) { - return { 'status' => 'E', - 'error' => - "customer $custnum has more than one conference". - " service (reseller?); specify a svcnum as a second argument", - }; - } + my $cust_pkg = $svc_pbx->cust_svc->cust_pkg; + my $cust_main = $cust_pkg->cust_main; - my $cust_svc_pbx = $cust_svc_pbx[0]; + my %result = (); - $svc_pbx = $cust_svc_pbx->svc_x; + #status in the reseller scenario + $result{'status'} = substr($cust_pkg->ucfirst_status,0,1); + + # find "outbound service" y/n + my @cust_pkg = $cust_main->cust_pkg; + #XXX what about outbound service per-reseller ? + my $conf = new FS::Conf; + my %outbound_pkgs = map { $_=>1 } $conf->config('mc-outbound_packages'); + $result{'outbound_service'} = + scalar( grep { $outbound_pkgs{ $_->pkgpart } + && !$_->get('cancel') + } + @cust_pkg + ) + ? 1 : 0; + + # find "good till" date/time stamp (this package) + $result{'good_till'} = time2str('%c', $cust_pkg->bill || time ); - # find "outbound service" y/n + return { + 'custnum' => $cust_main->custnum, + 'name' => $cust_main->name, + 'email' => $cust_main->invoicing_list_emailonly_scalar, + #'agentnum' => $cust_main->agentnum, + #'agent' => $cust_main->agent->agent, + 'max_lines' => $svc_pbx->max_extensions, + 'max_simultaneous' => $svc_pbx->max_simultaneous, + %result, + }; - my $conf = new FS::Conf; - my %outbound_pkgs = map { $_=>1 } $conf->config('mc-outbound_packages'); - $outbound_service = - scalar( grep { $outbound_pkgs{ $_->pkgpart } - && !$_->get('cancel') - } - @cust_pkg - ) - ? 1 : 0; +} - # find "good till" date/time stamp +#some false laziness w/ MyAccount order_pkg +sub order_pkg { + my $opt = ref($_[0]) ? shift : { @_ }; - my @active_cust_pkg = - sort { $a->bill <=> $b->bill } - grep { !$_->get('cancel') && $_->part_pkg->freq ne '0' } - @cust_pkg; - $good_till = time2str('%c', $active_cust_pkg[0]->bill || time ); + $opt->{'title'} = delete $opt->{'name'} + if !exists($opt->{'title'}) && exists($opt->{'name'}); + + my $custnum = $opt->{'custnum'}; + + my $curuser = $FS::CurrentUser::CurrentUser; + + my $cust_main = qsearchs({ + 'table' => 'cust_main', + 'hashref' => { 'custnum' => $custnum }, + 'extra_sql' => ' AND '. $curuser->agentnums_sql, + }) + or return { 'error' => "custnum $custnum not found" }; + + my $status = $cust_main->status; + #false laziness w/ClientAPI/Signup.pm + + my $cust_pkg = new FS::cust_pkg ( { + 'custnum' => $custnum, + 'pkgpart' => $opt->{'pkgpart'}, + } ); + my $error = $cust_pkg->check; + return { 'error' => $error } if $error; + + my @svc = (); + unless ( $opt->{'svcpart'} eq 'none' ) { + + my $svcpart = ''; + if ( $opt->{'svcpart'} =~ /^(\d+)$/ ) { + $svcpart = $1; + } else { + $svcpart = $cust_pkg->part_pkg->svcpart; #($svcdb); + } + + my $part_svc = qsearchs('part_svc', { 'svcpart' => $svcpart } ); + return { 'error' => "Unknown svcpart $svcpart" } unless $part_svc; + + my $svcdb = $part_svc->svcdb; + + my %fields = ( + 'svc_acct' => [ qw( username domsvc _password sec_phrase popnum ) ], + 'svc_domain' => [ qw( domain ) ], + 'svc_phone' => [ qw( phonenum pin sip_password phone_name ) ], + 'svc_external' => [ qw( id title ) ], + 'svc_pbx' => [ qw( id title ) ], + ); + + my $svc_x = "FS::$svcdb"->new( { + 'svcpart' => $svcpart, + map { $_ => $opt->{$_} } @{$fields{$svcdb}} + } ); + + #snarf processing not necessary here (or probably at all, anymore) + + my $y = $svc_x->setdefault; # arguably should be in new method + return { 'error' => $y } if $y && !ref($y); + + $error = $svc_x->check; + return { 'error' => $error } if $error; + + push @svc, $svc_x; } - return { - 'name' => $cust_main->name, - 'email' => $cust_main->invoicing_list_emailonly_scalar, - 'agentnum' => $cust_main->agentnum, - 'agent' => $cust_main->agent->agent, - 'max_lines' => $svc_pbx ? $svc_pbx->max_extensions : '', - 'max_simultaneous' => $svc_pbx ? $svc_pbx->max_simultaneous : '', - 'outbound_service' => $outbound_service, - 'good_till' => $good_till, - %result, - }; + use Tie::RefHash; + tie my %hash, 'Tie::RefHash'; + %hash = ( $cust_pkg => \@svc ); + #msgcat + $error = $cust_main->order_pkgs( \%hash, 'noexport' => 1 ); + return { 'error' => $error } if $error; + +# currently they're using this in the reseller scenario, so don't +# bill the package immediately +# my $conf = new FS::Conf; +# if ( $conf->exists('signup_server-realtime') ) { +# +# my $bill_error = _do_bop_realtime( $cust_main, $status ); +# +# if ($bill_error) { +# $cust_pkg->cancel('quiet'=>1); +# return $bill_error; +# } else { +# $cust_pkg->reexport; +# } +# +# } else { + $cust_pkg->reexport; +# } + + my $svcnum = $svc[0] ? $svc[0]->svcnum : ''; + + return { error=>'', pkgnum=>$cust_pkg->pkgnum, svcnum=>$svcnum }; }