1 package FS::ClientAPI::MyAccount;
5 use Digest::MD5 qw(md5_hex);
7 use Cache::SharedMemoryCache; #store in db?
8 use FS::CGI qw(small_custview); #doh
10 use FS::Record qw(qsearchs);
17 use FS::ClientAPI; #hmm
18 FS::ClientAPI->register_handlers(
19 'MyAccount/login' => \&login,
20 'MyAccount/customer_info' => \&customer_info,
21 'MyAccount/edit_info' => \&edit_info,
22 'MyAccount/invoice' => \&invoice,
23 'MyAccount/cancel' => \&cancel,
24 'MyAccount/list_pkgs' => \&list_pkgs,
25 'MyAccount/order_pkg' => \&order_pkg,
26 'MyAccount/cancel_pkg' => \&cancel_pkg,
29 use vars qw( @cust_main_editable_fields );
30 @cust_main_editable_fields = qw(
31 first last company address1 address2 city
32 county state zip country daytime night fax
33 ship_first ship_last ship_company ship_address1 ship_address2 ship_city
34 ship_state ship_zip ship_country ship_daytime ship_night ship_fax
38 my $cache = new Cache::SharedMemoryCache();
40 #false laziness w/FS::ClientAPI::passwd::passwd (needs to handle encrypted pw)
44 my $svc_domain = qsearchs('svc_domain', { 'domain' => $p->{'domain'} } )
45 or return { error => "Domain not found" };
48 ( length($p->{'password'}) < 13
49 && qsearchs( 'svc_acct', { 'username' => $p->{'username'},
50 'domsvc' => $svc_domain->svcnum,
51 '_password' => $p->{'password'} } )
53 || qsearchs( 'svc_acct', { 'username' => $p->{'username'},
54 'domsvc' => $svc_domain->svcnum,
55 '_password' => $p->{'password'} } );
57 unless ( $svc_acct ) { return { error => 'Incorrect password.' } }
60 'svcnum' => $svc_acct->svcnum,
63 my $cust_pkg = $svc_acct->cust_svc->cust_pkg;
65 my $cust_main = $cust_pkg->cust_main;
66 $session->{'custnum'} = $cust_main->custnum;
71 $session_id = md5_hex(md5_hex(time(). {}. rand(). $$))
72 } until ( ! defined $cache->get($session_id) ); #just in case
74 $cache->set( $session_id, $session, '1 hour' );
76 return { 'error' => '',
77 'session_id' => $session_id,
83 my $session = $cache->get($p->{'session_id'})
84 or return { 'error' => "Can't resume session" }; #better error message
88 my $custnum = $session->{'custnum'};
90 if ( $custnum ) { #customer record
92 my $cust_main = qsearchs('cust_main', { 'custnum' => $custnum } )
93 or return { 'error' => "unknown custnum $custnum" };
95 $return{balance} = $cust_main->balance;
100 date => time2str("%b %o, %Y", $_->_date),
103 } $cust_main->open_cust_bill;
104 $return{open_invoices} = \@open;
106 my $conf = new FS::Conf;
107 $return{small_custview} =
108 small_custview( $cust_main, $conf->config('defaultcountry') );
110 $return{name} = $cust_main->first. ' '. $cust_main->get('last');
112 for (@cust_main_editable_fields) {
113 $return{$_} = $cust_main->get($_);
116 } else { #no customer record
118 my $svc_acct = qsearchs('svc_acct', { 'svcnum' => $session->{'svcnum'} } )
119 or die "unknown svcnum";
120 $return{name} = $svc_acct->email;
125 return { 'error' => '',
126 'custnum' => $custnum,
134 my $session = $cache->get($p->{'session_id'})
135 or return { 'error' => "Can't resume session" }; #better error message
137 my $custnum = $session->{'custnum'}
138 or return { 'error' => "no customer record" };
140 my $cust_main = qsearchs('cust_main', { 'custnum' => $custnum } )
141 or return { 'error' => "unknown custnum $custnum" };
143 my $new = new FS::cust_main { $cust_main->hash };
144 $new->set( $_ => $p->{$_} )
145 foreach grep { exists $p->{$_} } @cust_main_editable_fields;
146 my $error = $new->replace($cust_main);
147 return { 'error' => $error } if $error;
150 return { 'error' => '' };
155 my $session = $cache->get($p->{'session_id'})
156 or return { 'error' => "Can't resume session" }; #better error message
158 my $custnum = $session->{'custnum'};
160 my $invnum = $p->{'invnum'};
162 my $cust_bill = qsearchs('cust_bill', { 'invnum' => $invnum,
163 'custnum' => $custnum } )
164 or return { 'error' => "Can't find invnum" };
168 return { 'error' => '',
170 'invoice_text' => join('', $cust_bill->print_text ),
177 my $session = $cache->get($p->{'session_id'})
178 or return { 'error' => "Can't resume session" }; #better error message
180 my $custnum = $session->{'custnum'};
182 my $cust_main = qsearchs('cust_main', { 'custnum' => $custnum } )
183 or return { 'error' => "unknown custnum $custnum" };
185 my @errors = $cust_main->cancel( 'quiet'=>1 );
187 my $error = scalar(@errors) ? join(' / ', @errors) : '';
189 return { 'error' => $error };
195 my $session = $cache->get($p->{'session_id'})
196 or return { 'error' => "Can't resume session" }; #better error message
198 my $custnum = $session->{'custnum'};
200 my $cust_main = qsearchs('cust_main', { 'custnum' => $custnum } )
201 or return { 'error' => "unknown custnum $custnum" };
203 return { 'cust_pkg' => [ map { $_->hashref } $cust_main->ncancelled_pkgs ] };
209 my $session = $cache->get($p->{'session_id'})
210 or return { 'error' => "Can't resume session" }; #better error message
212 my $custnum = $session->{'custnum'};
214 my $cust_main = qsearchs('cust_main', { 'custnum' => $custnum } )
215 or return { 'error' => "unknown custnum $custnum" };
217 #false laziness w/ClientAPI/Signup.pm
219 my $cust_pkg = new FS::cust_pkg ( {
220 'custnum' => $custnum,
221 'pkgpart' => $p->{'pkgpart'},
223 my $error = $cust_pkg->check;
224 return { 'error' => $error } if $error;
226 my $svc_acct = new FS::svc_acct ( {
227 'svcpart' => $p->{'svcpart'} || $cust_pkg->part_pkg->svcpart('svc_acct'),
228 map { $_ => $p->{$_} }
229 qw( username _password sec_phrase popnum ),
234 while ( length($p->{"snarf_machine$snarfnum"}) ) {
235 my $acct_snarf = new FS::acct_snarf ( {
236 'machine' => $p->{"snarf_machine$snarfnum"},
237 'protocol' => $p->{"snarf_protocol$snarfnum"},
238 'username' => $p->{"snarf_username$snarfnum"},
239 '_password' => $p->{"snarf_password$snarfnum"},
242 push @acct_snarf, $acct_snarf;
244 $svc_acct->child_objects( \@acct_snarf );
246 my $y = $svc_acct->setdefault; # arguably should be in new method
247 return { 'error' => $y } if $y && !ref($y);
249 $error = $svc_acct->check;
250 return { 'error' => $error } if $error;
253 tie my %hash, 'Tie::RefHash';
254 %hash = ( $cust_pkg => [ $svc_acct ] );
256 $error = $cust_main->order_pkgs( \%hash, '', 'noexport' => 1 );
257 return { 'error' => $error } if $error;
259 my $conf = new FS::Conf;
260 if ( $conf->exists('signup_server-realtime') ) {
262 my $old_balance = $cust_main->balance;
264 my $bill_error = $cust_main->bill;
265 $cust_main->apply_payments;
266 $cust_main->apply_credits;
267 $bill_error = $cust_main->collect;
269 if ( $cust_main->balance > $old_balance ) {
270 $cust_pkg->cancel('quiet'=>1);
271 return { 'error' => '_decline' };
280 return { error => '' };
286 my $session = $cache->get($p->{'session_id'})
287 or return { 'error' => "Can't resume session" }; #better error message
289 my $custnum = $session->{'custnum'};
291 my $cust_main = qsearchs('cust_main', { 'custnum' => $custnum } )
292 or return { 'error' => "unknown custnum $custnum" };
294 my $pkgnum = $session->{'pkgnum'};
296 my $cust_pkg = qsearchs('cust_pkg', { 'custnum' => $custnum,
297 'pkgnum' => $pkgnum, } )
298 or return { 'error' => "unknown pkgnum $pkgnum" };
300 my $error = $cust_main->cancel( 'quiet'=>1 );
301 return { 'error' => $error };